在Android6.0版本之后引入,用户可以对应用程序进行选择性授权,即使拒绝了这个权限,依旧可以使用该应用的其他功能。
Android所有的危险权限一共有9组24个权限:
Dangerous Permissions:
group:android.permission-group.CONTACTS
permission:android.permission.WRITE_CONTACTS
permission:android.permission.GET_ACCOUNTS
permission:android.permission.READ_CONTACTS
group:android.permission-group.PHONE
permission:android.permission.READ_CALL_LOG
permission:android.permission.READ_PHONE_STATE
permission:android.permission.CALL_PHONE
permission:android.permission.WRITE_CALL_LOG
permission:android.permission.USE_SIP
permission:android.permission.PROCESS_OUTGOING_CALLS
permission:com.android.voicemail.permission.ADD_VOICEMAIL
group:android.permission-group.CALENDAR
permission:android.permission.READ_CALENDAR
permission:android.permission.WRITE_CALENDAR
group:android.permission-group.CAMERA
permission:android.permission.CAMERA
group:android.permission-group.SENSORS
permission:android.permission.BODY_SENSORS
group:android.permission-group.LOCATION
permission:android.permission.ACCESS_FINE_LOCATION
permission:android.permission.ACCESS_COARSE_LOCATION
group:android.permission-group.STORAGE
permission:android.permission.READ_EXTERNAL_STORAGE
permission:android.permission.WRITE_EXTERNAL_STORAGE
group:android.permission-group.MICROPHONE
permission:android.permission.RECORD_AUDIO
group:android.permission-group.SMS
permission:android.permission.READ_SMS
permission:android.permission.RECEIVE_WAP_PUSH
permission:android.permission.RECEIVE_MMS
permission:android.permission.RECEIVE_SMS
permission:android.permission.SEND_SMS
permission:android.permission.READ_CELL_BROADCASTS
注意:一旦用户授权了一个权限,该权限所对应的权限组中的所有的其他权限也会同时被授权。
在程序运行时申请权限
先修改AndroidManifest.xml文件如下:
申请运行时权限核心代码:
onClick(){
//如果用户未授权拨打电话权限,申请运行时权限 if(ContextCompat.checkSelfPermission(MainActivity.this,Manifest. permission.CALL_PHONE)!=PackageManager.PERMISSION_GRANTED){
//参数1:Activity的实例;参数2:String数组,放置需要申请的权限名
//参数3:请求码,只要是唯一值就可以了
ActivityCompat.requestPermissions(MainActivity.this,new String[] {
Manifest.permission.CALL_PHONE},1);
}else{
call();
}
}
call(){
try{
Intent intent = new Intent(Intent.ACTION_CALL);
intent.setData(Uri.parse("tel:10086"));
startActivity(intent);
}catch(Exception e){
...
}
}
//调用完requestPermission()方法后,无论用户是否授权,最终都会回调到此方法中
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类。
ContentResolver类提供了与SQLiteDatabase类相似的一系列方法对数据进行CRUD操作。
ContentResolver类不接受表名参数,而是接收一个Uri参数。
这个内容URI由authority和path两部分组成,authority是用于对不同的应用作区分的,一般采用程序包名的方式来进行命名;path则是用于对同一程序中不同的表做区分的,通常都会添加到authority后面。
如:content://com.example.app.provider/table1
得到了内容URI字符串后,还需要将它解析为Uri对象才能作为参数传入:
Uri uri = Uri.parse("content://com.example.app.provider/table1");
使用上述uri查询table1表中的数据:
//参数1:指定查询某个应用程序下的某一张表;参数2:指定查询的列名;
//参数3:指定where的约束条件;参数4:为where中的占位符提供具体的值
//参数5:指定查询结果的排序方式
Cursor cursor = getContentResolver().query( uri, projection, selection, selectionArgs, sortOrder );
if(cursor!=null){
while(cursor.moveToNext()){
String column1 = cursor.getString(cursor.getColumnIndex("column1"));
int column2 = cursor.getInt(cursor.getColumnIndex("column2"));
}
cursor.close();
}
添加数据核心代码:
ContentValues values = new ContentValues();
values.put("column1","text");
values.put("column2",1);
getContentResolver().insert(uri,values);
修改数据核心代码:
ContentValues values = new ContentValues();
values.put("column1","");
getContentResolver().update(uri,values,"column1 = ? and column2 = ?",
new string[]{"text","1"});
删除数据核心代码:
getContentResolver().delete(uri,"column2 = ?",new String[]{"1"});
读取系统联系人信息:
步骤1 核心代码实现:
onCreate(){
ListView contactsView = (ListView)findViewById(...);
adapter = new ArrayAdapter(...);
contactsView.setAdapter(adapter);
//如果未授权读取联系人权限,申请授权 if(ContextCompat.checkSelfPermission(this,Manifest.permission.READ_ CONTACTS)!=PackageManager.PERMISSION_GRANTED){
ActivityCompat.requestPermissions(this,new String[]{Manifest.per mission.READ_CONTACTS},1); }else{
readContacts();
}
}
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)); contactsList.add(displayName+"\n"+number);
}
adapter.notifyDataSetChanged();
}
}catch(Exception e){
...
}finally{
if(cursor!=null){
cursor.close();
}
}
}
public void onRequestPermissionsResult(int requestCode,String[] permissions, 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:
}
}
步骤2 配置读取系统联系人的权限:
创建自己的内容提供器
自定义内容提供器核心代码:
class MyProvider extends ContentProvider{
//初始化内容提供器时调用,一般在这完成数据库的创建和升级操作
boolean onCreate(){
return false;
}
//从内容提供器中查询数据
Cursor query(Uri uri,String[] projection,String selection,String[] selectionArgs,String sortOrder){
return null;
}
//向内容提供器中添加数据,添加完成后,返回一个用于表示这条新记录的URI
Uri insert(Uri uri,ContentValues values){
return null;
}
//更新内容提供器中已有的数据,返回受影响的行数
int update(Uri uri,ContentValues values,String selection,String[] selectionArgs){
return 0;
}
//从内容提供器中删除数据,返回受影响的行数
int delete(Uri uri,String selection,String[] selectionArgs){
return 0;
}
//根据传入的内容URI来返回相应的MIME类型
String getType(Uri uri){
return null;
}
}
借助UriMatcher这个类可以轻松匹配内容URI的功能
核心代码实现:
class MyProvider extends ContentProvider{
public static final int TABLE1_DIR = 0;
public static final int TABLE1_ITEM = 1;
public static final int TABLE1_DIR = 2;
public static final int TABLE1_ITEM = 3;
private static UriMatcher uriMatcher;
static{ uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
//参数1:authority;参数2:path;参数3:自定义代码 uriMatcher.addURI("com.example.app.provider","table1",TABLE1_DIR); uriMatcher.addURI("com.example.app.provider","table1/#",TABLE1_ITEM); uriMatcher.addURI("com.example.app.provider","table2",TABLE2_DIR); uriMatcher.addURI("com.example.app.provider","table2/#",TABLE2_ITEM);
}
Cursor query(Uri uri,String[] projection,String selection,String[] selectionArgs,String sortOrder){
//调用UriMatcher的match()方法时,返回值是某个能够匹配这个Uri对象
//所对应的自定义代码
switch(uriMatcher.match(uri){
case TABLE1_DIR:
//查询table1表中的所有数据
break;
case TABLE1_ITEM:
//查询table1表中的单条数据
break;
case TABLE2_DIR:
//查询table2表中的所有数据
break;
case TABLE2_ITEM:
//查询table2表中的单条数据
break;
default:
break;
}
}
}
getType()是所有内容提供器都必须提供的方法,用于获取Uri对象所对应的MIME类型。
一个内容URI所对应的MIME字符串主要由三部分组成:
1 必须以vnd开头
2 如果内容URI以路径结尾,则后接android.cursir.dir/,如果内容URI以id结尾,则后接android.cursir.item/
3 最后接上vnd.
完善MyProvider代码:
String getType(Uri uri){
switch(uriMatcher.match(uri){
case TABLE1_DIR: return "vnd.android.cursor.dir/vnd.com.example.app.provider.table1";
case TABLE1_ITEM: return "vnd.android.cursor.item/vnd.com.example.app.provider.table1"; case TABLE2_DIR: return "vnd.android.cursor.dir/vnd.com.example.app.provider.table2";
case TABLE2_ITEM: return "vnd.android.cursor.item/vnd.com.example.app.provider.table2";
default:
break;
}
return null;
}