android的内容提供器使用方法 android6.0以上获取运行是权限方法 本地内容提供器和自定义内容提供器
2017/3/10 记载
《内容提供器》
内容提供器:主要是用于在不同应用程序之间共享数据的,打个比如,很多应用需要获取本地系统中的电话簿、GPRS定位、还有其他应用的数据,这些都是需要共享的。
运行时权限:在6.0android以下的版本,都是在安装应用的时候一次性获取所有的权限,但是在6.0之后
谷歌考虑到安全性的问题和实用性的问题,就抛弃了以前的方式,采用在应用运行的时候要用到的时候,提示获取
权限,若用户不同意就不能用,但是依然可以使用其他功能,从而避免了店大欺客的问题。
Android将现在的权限分为普通权限和危险权限,普通权限是不会威胁到用户的数据安全性问题。
就可以在安装时候允许或者由系统自动给你判定。危险权限就需要用户自行判断了,危险权限也不多如下所示:
效果图:
等。除此之外,还提供了一个PackageManger(需要了解一下)管理类,它的主要职责是管理应用程序包。 通过它,我们就可以获取应用程序信息。
做一个在运行时可以获取的权限例子RuntimePermissionProject:当点击按钮拨打电话10086,这个时候就会调用运行时权限了。
activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">
<Button
android:id="@+id/make_call"
android:text="Make Call"
android:textAllCaps="false"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
</LinearLayout>
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.ldp.com.runtimepermissionproject">
<!--添加权限-->
<uses-permission android:name="android.permission.CALL_PHONE"/>
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
MainActivity
package com.example.ldp.com.runtimepermissionproject;
import android.Manifest;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.support.annotation.NonNull;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
/*点击按钮要获取权限*/
Button buttonCall = (Button)findViewById(R.id.make_call);
buttonCall.setOnClickListener(new View.OnClickListener() {
/**ContextCompat.checkSelfPermission()检查用户是否已经给我们授权了(第二个参数是什么权限)。
* PackageManager调用应用权限组比较,给了就直接执行Call(),如果没有给权限通过
* ActivityCompat.requestPermissions(Activity的实例,申请的权限放在其中,请求码(唯一值)这里设置为1)方法请求权限
* */
@Override
public void onClick(View v) {
if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.CALL_PHONE)!=
PackageManager.PERMISSION_GRANTED){
ActivityCompat.requestPermissions(MainActivity.this,new String[]{Manifest.permission.CALL_PHONE},1);
}else{
call();
}
}
});
}
private void call(){
try {
Intent intent = new Intent(Intent.ACTION_CALL);
intent.setData(Uri.parse("tel:10086"));
startActivity(intent);
}catch (SecurityException e){
e.printStackTrace();
}
}
/*不论用户同意还是拒绝,
*activity的onRequestPermissionsResult会被回调来通知结果(通过第三个参数),grantResults,如下*/
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
switch (requestCode){
case 1:
if (grantResults.length>0&&grantResults[0]== PackageManager.PERMISSION_GRANTED){
call();
}else{
Toast.makeText(this,"You Denied the permission",Toast.LENGTH_SHORT).show();
}
break;
default:
}
}
}
效果图:因为我这里使用的是夜神模拟器,所以出现不支持的情况,不过代码没问题,可以使用本机模拟器试一下
进入正题了,内容提供器 ContentResolver
1.使用现有的内容提空气来读取相应程序中的数据
2.自己创建一个内容提供器来给应用提供数据外部访问接口
说明:
效果图:
写一个例子来通过ContentResolver来获取本地电话簿
activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">
<ListView
android:id="@+id/contacts_view"
android:layout_width="match_parent"
android:layout_height="match_parent">
</ListView>
</LinearLayout>
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.ldp.com.contenttest">
<uses-permission android:name="android.permission.READ_CONTACTS"/>
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
MainActivity
package com.example.ldp.com.contenttest;
import android.Manifest;
import android.content.pm.PackageManager;
import android.database.Cursor;
import android.provider.ContactsContract;
import android.support.annotation.NonNull;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.Switch;
import android.widget.Toast;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends AppCompatActivity {
ArrayAdapter<String> adapter;
List<String> contractsList = new ArrayList<>();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ListView constactView = (ListView) findViewById(R.id.contacts_view);
adapter = new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1,contractsList);
constactView.setAdapter(adapter);
if(ContextCompat.checkSelfPermission(this, Manifest.permission.READ_CONTACTS)!= PackageManager.PERMISSION_GRANTED){
ActivityCompat.requestPermissions(this,new String[]{Manifest.permission.READ_CONTACTS},1);
}else{
}
}
private void readContacts(){
Cursor cursor = null;
try{
cursor = getContentResolver().query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI,null,null,null,null);
if(cursor!=null){
while(cursor.moveToNext()){
String displayName = cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME));
String number = cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
contractsList.add(displayName+"\n"+number);
}
adapter.notifyDataSetChanged();
}
}catch (Exception e){
e.printStackTrace();
}finally {
if(cursor!=null){
cursor.close();
}
}
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
switch (requestCode){
case 1:
if(grantResults.length>0&&grantResults[0]==PackageManager.PERMISSION_GRANTED){
readContacts();
}else{
Toast.makeText(this,"You denied the permission",Toast.LENGTH_SHORT).show();
}
break;
default:
}
}
}
两张效果图:
获取权限:
允许权限:
上,面是使用本地内置的内容提供器的例子,接下来使用直接定义的内容提供器
直接创建内容提供器:
想要使用扩程序的共享数据功能,推荐使用继承ContentProvider(6个抽象方法)的方式创建
参数效果图:
利用UriMatcher中的addURI(authority,Path,自定义代码),可以讲一个Uri对象传入返回值是某个能够匹配
这个Uri对象所对应的自定义代码。
效果图: