https://developers.google.com/Android/guides/permissions
问题描述:一个apk在6.0之下运行正常,但是在6.0上运行一直崩溃,并且提示需要某个权限,但是该权限已经在AndroidMainfest.xml中声明。当apk安装的时候,弹出一个可以允许和拒绝权限的弹框,如果在这里运行该apk使用该权限,则apk则运行正常,否则继续崩溃。
猜测及思路:6.0也许引入了一个新的权限管理机制,apk所需要的权限可以让用户手动允许和拒绝。
崩溃的原因是apk所需要的权限被拒绝了,相应的获取操作出现了异常。那么如何在代码中检测该权限呢?如果检测到没有该权限则不执行相应的daim
android6.0棉花软糖有一个新的使app安装和自动更新更加有效的权限模式。现在的权限是在运行时请求而不是在apk被安装之前,用户可以选择允许或拒绝某一项特定的权限。为了实现这个灵活性,你的apk需要确保当用户允许或拒绝某一项特定的权限后apk运行正常。
前提:
1.API level和目标SDK设置在23+
2.确保使用了Support V4:com.android.support:support-v4:23.0.0
3.所需要的的权限还是需要在AndroidMainfest.xml中正常声明
指南:
1.如果你在AndroidMainfest.xml中声明了一个权限,那么在调用API方法之前,通过ActivityCompat或者ContextCompat的checkSerlfPermission()方法来检测你是否有相应的请求权限。如果返回false说明该权限没有被用户允许,你需要通过requestPermissions()方法重新请求这些权限。返回的结果可以在回调方法中获取
if(ActivityCompat.checkSelfPermission(this,Manifest.permission.ACCESS_FINE_LOCATION)
!=PackageManager.PERMISSION_GRANTED){
// Check Permissions Now
ActivityCompat.requestPermissions(this,
newString[]{Manifest.permission.ACCESS_FINE_LOCATION},
REQUEST_LOCATION);
}else{
// permission has been granted, continue as usual
LocationmyLocation=
LocationServices.FusedLocationApi.getLastLocation(mGoogleApiClient);
}
2.实现请求权限的回调
如果你所需要的权限没有被用户允许,requestPermission()方法将会回调询问用户是否允许这些权限。用户执行的结果将在回调onRequestPermissionsResult()中被返回。由于权限请求可能会被拒绝或者取消,所以你应该实现该回调并在调用相应API时进行检查。下面的代码只是检测了一个单独的权限请求,你也可以一次检测多个权限请求。
publicvoidonRequestPermissionsResult(intrequestCode,
String[]permissions,
int[]grantResults){
if(requestCode==REQUEST_LOCATION){
if(grantResults.length==1
&&grantResults[0]==PackageManager.PERMISSION_GRANTED){
// We can now safely use the API we requested access to
LocationmyLocation=
LocationServices.FusedLocationApi.getLastLocation(mGoogleApiClient);
}else{
// Permission was denied or request was cancelled
}
}
}
3.显示需要该权限的原因
如果你的apk的核心功能需要某个权限,但是该权限又被用户拒绝了,在再次请求权限之前,你需要展示一个额外的说明给用户。当用户明白为何需要该权限时,他们更倾向于允许该权限。
在这种请求下,你需要先调用shouldShowRequestPermissionRationale()方法,然后再调用requestPermissions()方法。如果返回true,你需要新建一个界面用于展示为什么需要该权限。
if(ActivityCompat.checkSelfPermission(this,Manifest.permission.ACCESS_FINE_LOCATION)
!=PackageManager.PERMISSION_GRANTED){
// Check Permissions Now
privatestaticfinalintREQUEST_LOCATION=2;
if(ActivityCompat.shouldShowRequestPermissionRationale(this,
Manifest.permission.ACCESS_FINE_LOCATION)){
// Display UI and wait for user interaction
}else{
ActivityCompat.requestPermissions(
this,newString[]{Manifest.permission.LOCATION_FINE},
ACCESS_FINE_LOCATION);
}
}else{
// permission has been granted, continue as usual
LocationmyLocation=
LocationServices.FusedLocationApi.getLastLocation(mGoogleApiClient);
}
4.处理异常
如果你的apk使用了GoogleApiClient,并且调用了connect()方法,GooglePlay services会检测它所需要的每一项权限。在GooglePlay service8.1及其之后,如果所需权限中的任何一项缺失,connect()将会返回false。
如果你调用connect()失败,确保你的app正确处理所产生的异常。如果缺失了Google Play service所需要的异常,你可以通过startResolutionForResult()来重新初始化来解决
@Override
publicvoidonConnectionFailed(ConnectionResultresult){
if(mResolvingError){
// Already attempting to resolve an error.
return;
}elseif(result.hasResolution()){
try{
mResolvingError=true;
result.startResolutionForResult(this,REQUEST_RESOLVE_ERROR);
}catch(SendIntentExceptione){
// There was an error with the resolution intent. Try again.
mGoogleApiClient.connect();
}
}else{
// Show dialog using GooglePlayServicesUtil.getErrorDialog()
showErrorDialog(result.getErrorCode());
mResolvingError=true;
}
}