Android获取设备隐私 忽略6.0权限管理

所属分类: 软件编程 / Android 阅读数: 1641
收藏 0 赞 0 分享

一.前言
(1).由于MIUI等部分国产定制系统也有权限管理,没有相关api,故无法判断用户是否允许获取联系人等隐私。在Android 6.0之后,新增权限管理可以通过官方api判断用户的运行状态;
(2).我们指定targetSdkVersion为23或者之后我们还需要在运行时请求这些所需的权限。这很重要,因为已经出现了很多开发者把targetSdkVersion飙到了最新,然后发现自己的app疯狂的崩溃,这是由于他们没有实现执行运行时权限请求的代码。当你已经把一个targeting API 为23或者之后的app发布到了Google Play上,这更是一个问题,你无法立即把那个apk的targeting API替换成更早的版本。

二.权限分析
从Android6.0开始,权限分为普通权限和许可权限。许可权限分类归组,一个权限授权之后,该组下的权限均可使用。
(1)普通权限
只需要在xml申请即可,使用方法和之前6.0以前的一样。在应用安装应用时,会默认获得许可。
(2)许可权限
可执行 $adb shell pm list permissions -d -g

同一组的任何一个权限被授权了,其他权限也自动被授权。例如,一旦WRITE_CONTACTS被授权了,app也有READ_CONTACTS和GET_ACCOUNTS了。
源码中被用来检查和请求权限的方法分别是Activity的checkSelfPermission和requestPermissions,这些方法api23引入。

三.相关方法
(1).ContextCompat.checkSelfPermission()
检查应用是否拥有该权限,被授权返回值为PERMISSION_GRANTED,否则返回PERMISSION_DENIED
(2).ActivityCompat.requestPermissions()
将弹出请求授权对话框,这个方法在M之前版本调用,OnRequestPermissionsResultCallback 直接被调用,带着正确的 PERMISSION_GRANTED或者 PERMISSION_DENIED 。
(3).AppCompatActivity.onRequestPermissionsResult()
该方法类似于Activity的OnActivityResult()的回调方法,主要接收请求授权的返回值

//版本判断 
if (Build.VERSION.SDK_INT >= 23) { 
  //减少是否拥有权限 
  int checkCallPhonePermission = ContextCompat.checkSelfPermission(getApplicationContext(), permission); 
  if (checkCallPhonePermission != PackageManager.PERMISSION_GRANTED) { 
    //弹出对话框接收权限 
    ActivityCompat.requestPermissions(BaseActivity.this, new String[]{permission}, id); 
    return; 
} 
@Override 
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { 
  super.onRequestPermissionsResult(requestCode, permissions, grantResults); 
 
  if (grantResults[0] == PackageManager.PERMISSION_GRANTED) { 
    //TODO:已授权 
  } else { 
    //TODO:用户拒绝 
  } 
} 

(4).封装

public class BaseActivity extends AppCompatActivity { 
  private Map<Integer, Runnable> allowablePermissionRunnables = new HashMap<>(); 
  private Map<Integer, Runnable> disallowablePermissionRunnables = new HashMap<>(); 
 
  @Override 
  protected void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
  } 
 
  /** 
   * 请求权限 
   * @param id 请求授权的id 唯一标识即可 
   * @param permission 请求的权限 
   * @param allowableRunnable 同意授权后的操作 
   * @param disallowableRunnable 禁止权限后的操作 
   */ 
  protected void requestPermission(int id, String permission, Runnable allowableRunnable, Runnable disallowableRunnable) { 
    if (allowableRunnable == null) { 
      throw new IllegalArgumentException("allowableRunnable == null"); 
    } 
 
    allowablePermissionRunnables.put(id, allowableRunnable); 
    if (disallowableRunnable != null) { 
      disallowablePermissionRunnables.put(id, disallowableRunnable); 
    } 
 
    //版本判断 
    if (Build.VERSION.SDK_INT >= 23) { 
      //减少是否拥有权限 
      int checkCallPhonePermission = ContextCompat.checkSelfPermission(getApplicationContext(), permission); 
      if (checkCallPhonePermission != PackageManager.PERMISSION_GRANTED) { 
        //弹出对话框接收权限 
        ActivityCompat.requestPermissions(BaseActivity.this, new String[]{permission}, id); 
        return; 
      } else { 
        allowableRunnable.run(); 
      } 
    } else { 
      allowableRunnable.run(); 
    } 
  } 
 
  @Override 
  public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { 
    super.onRequestPermissionsResult(requestCode, permissions, grantResults); 
 
    if (grantResults[0] == PackageManager.PERMISSION_GRANTED) { 
      Runnable allowRun = allowablePermissionRunnables.get(requestCode); 
      allowRun.run(); 
    } else { 
      Runnable disallowRun = disallowablePermissionRunnables.get(requestCode); 
      disallowRun.run(); 
    } 
  } 
} 
public class MainActivity extends BaseActivity implements View.OnClickListener{ 
  private Button btCallPhone; 
  private Button btContact; 
 
  @Override 
  protected void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.activity_main); 
 
    btCallPhone = (Button) findViewById(R.id.call_phone); 
    btContact = (Button) findViewById(R.id.contact); 
 
    btCallPhone.setOnClickListener(this); 
    btContact.setOnClickListener(this); 
  } 
 
  @Override 
  public void onClick(View v) { 
    if(v == btCallPhone){ 
      //拨打电话 
      requestPermission(1, Manifest.permission.CALL_PHONE, new Runnable() { 
        @Override 
        public void run() { 
          callPhone(); 
        } 
      }, new Runnable() { 
        @Override 
        public void run() { 
          callPhoneDenied(); 
        } 
      }); 
    }else if(v == btContact){ 
      //读取联系人信息 
      requestPermission(2, Manifest.permission.WRITE_CONTACTS, new Runnable() { 
        @Override 
        public void run() { 
          readContact(); 
        } 
      }, new Runnable() { 
        @Override 
        public void run() { 
          readContactDenied(); 
        } 
      }); 
    } 
  } 
 
  private void callPhone() { 
    Toast.makeText(MainActivity.this, "CALL_PHONE OK", Toast.LENGTH_SHORT) 
        .show(); 
  } 
 
  private void callPhoneDenied() { 
    Toast.makeText(MainActivity.this, "CALL_PHONE Denied", Toast.LENGTH_SHORT) 
        .show(); 
  } 
 
  private void readContact() { 
    ContentResolver cr = getContentResolver(); 
    String str[] = {ContactsContract.CommonDataKinds.Phone.CONTACT_ID, ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME, ContactsContract.CommonDataKinds.Phone.NUMBER, 
        ContactsContract.CommonDataKinds.Phone.PHOTO_ID}; 
    Cursor cur = cr.query( 
        ContactsContract.CommonDataKinds.Phone.CONTENT_URI, str, null, 
        null, null); 
    int count = cur.getCount(); 
    cur.close(); 
 
    Toast.makeText(MainActivity.this, String.format("发现%s条", count), Toast.LENGTH_SHORT) 
        .show(); 
  } 
 
  private void readContactDenied() { 
    Toast.makeText(MainActivity.this, "Contact Denied", Toast.LENGTH_SHORT) 
        .show(); 
  } 
} 

以上就是本文的全部内容,希望对大家的学习有所帮助。

更多精彩内容其他人还在看

Android网络编程之获取网络上的Json数据实例

这篇文章主要介绍了Android网络编程之获取网络上的Json数据实例,本文用完整的代码实例讲解了在Android中读取网络中Json数据的方法,需要的朋友可以参考下
收藏 0 赞 0 分享

Android中的windowSoftInputMode属性详解

这篇文章主要介绍了Android中的windowSoftInputMode属性详解,本文对windowSoftInputMode的9个属性做了详细总结,需要的朋友可以参考下
收藏 0 赞 0 分享

Android网络编程之UDP通信模型实例

这篇文章主要介绍了Android网络编程之UDP通信模型实例,本文给出了服务端代码和客户端代码,需要的朋友可以参考下
收藏 0 赞 0 分享

Android中使用ListView实现漂亮的表格效果

这篇文章主要介绍了Android中使用ListView实现漂亮的表格效果,本文用详细的代码实例创建了一个股票行情表格,需要的朋友可以参考下
收藏 0 赞 0 分享

Android中刷新界面的二种方法

这篇文章主要介绍了Android中刷新界面的二种方法,本文使用Handler、postInvalidate两种方法实现界面刷新,需要的朋友可以参考下
收藏 0 赞 0 分享

Android SDK三种更新失败及其解决方法

这篇文章主要介绍了Android SDK三种更新失败及其解决方法,需要的朋友可以参考下
收藏 0 赞 0 分享

Android学习笔记——Menu介绍(一)

Android3.0(API level 11)开始,Android设备不再需要专门的菜单键。随着这种变化,Android app应该取消对传统6项菜单的依赖。取而代之的是提供anction bar来提供基本的用户功能
收藏 0 赞 0 分享

Android学习笔记——Menu介绍(二)

这次将继续上一篇文章没有讲完的Menu的学习,上下文菜单(Context menu)和弹出菜单(Popup menu)
收藏 0 赞 0 分享

Android学习笔记——Menu介绍(三)

今天继续昨天没有讲完的Menu的学习,主要是Popup Menu的学习,需要的朋友可以参考下
收藏 0 赞 0 分享

Android显示网络图片实例

这篇文章主要介绍了Android显示网络图片的方法,以实例形式展示了Android程序显示网络图片的方法,非常具有实用价值,需要的朋友可以参考下
收藏 0 赞 0 分享
查看更多