我对Android 6(Marshmallow)中的新权限系统感到困惑。
我正在使用两个所谓的'dangeorus权限' -
<uses-permission android:name="android.permission.GET_ACCOUNTS" />
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
我在很多地方使用这些权限。
我需要访问用户帐户注册应用程序,发送反馈等。我还访问设备ID以注册应用程序。
我编写了两个函数来获取名为util的Java类中的邮件帐户列表和设备ID,并在AsyncTask中使用这些函数以及片段。
现在,我完全不知所措,如何申请这些权限!我应该在函数中询问还是每次调用这些函数?在AsyncTask中,我如何得到响应?
请帮忙。
答案 0 :(得分:3)
您不应该在AsyncTask中要求权限。实际上你甚至不应该检索deviceId或其中的一些信息 - 将其作为参数传递。
每个应用程序的设置都包含“危险”权限及其状态的列表。如果您针对sdk 23定位应用,则默认情况下它们都已禁用。因此,当您使用需要此类权限的代码时,您需要检查它是否被授予,如果没有 - 请求它。
简化方法是为这样的权限创建实用程序类:
public final class PermissionUtils {
private PermissionUtils() {
}
public static boolean checkPermissions(Context context, String... permissions) {
for (String permission : permissions) {
if (!checkPermission(context, permission)) {
return false;
}
}
return true;
}
public static boolean checkPermission(Context context, String permission) {
return ContextCompat.checkSelfPermission(context, permission) == PackageManager.PERMISSION_GRANTED;
}
public static boolean isDeviceInfoGranted(Context context) {
return checkPermission(context, Manifest.permission.READ_PHONE_STATE);
}
public static void requestPermissions(Object o, int permissionId, String... permissions) {
if (o instanceof Fragment) {
FragmentCompat.requestPermissions((Fragment) o, permissions, permissionId);
} else if (o instanceof Activity) {
ActivityCompat.requestPermissions((AppCompatActivity) o, permissions, permissionId);
}
}
}
然后像这样使用它: (在你的片段/活动中)
if (PermissionUtils.isDeviceInfoGranted(this)) {
//get deviceId and proccess your code
} else {
String[] permissions = new String[]{READ_PHONE_STATE};
PermissionUtils.requestPermissions(this, PHONE_STATE_PERMISSION_ID, permissions);
}
覆盖AppCompatActivity中权限结果的处理程序或实现OnRequestPermissionsResultCallback接口:
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
switch (requestCode) {
case PHONE_STATE_PERMISSION_ID:
boolean granted = grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED;
if (granted) {
//get deviceId and proccess your code
} else {
//nobody knows what to do
}
break;
default:
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
}
}
如您所见,我们仍有问题 - 如果用户拒绝了该怎么办?作为变体,我们可以显示解释对话框。或者(可能更好)只在登机屏幕后才要求权限。
答案 1 :(得分:2)
首先,您必须在您的activity / fragment中实现以下方法:
@Override
public void onRequestPermissionsResult(int requestCode,@NonNull String permissions[], @NonNull int[] grantResults) {
switch (requestCode) {
case Permissions.READ_CONTACTS: { //Permissions.READ_CONTACTS is an int constant
if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
getContactsFromPhone();
}
}
}
}
然后在你的onclick上获得这些联系人你应该问你是否有这个权限:
if (ContextCompat.checkSelfPermission(activity, Manifest.permission.READ_CONTACTS) != PackageManager.PERMISSION_GRANTED) {
ActivityCompat.requestPermissions(activity,
new String[]{Manifest.permission.READ_CONTACTS}, Permissions.READ_CONTACTS); //Permissions.READ_CONTACTS is an int constant
return false;
} else {
getContactsFromPhone();
}
答案 2 :(得分:0)
1)是的,每次调用这些函数时都应该检查是否授予了权限。
2)您可以检查并从主线程请求权限。我认为从AsyncTask检查和请求权限并不是一个好方法,因为权限请求是一个对话框,你必须等待asynctask线程,直到你得到用户响应。尝试检查主线程的权限,获取用户响应,然后,如果授予权限,则启动asynctask。
答案 3 :(得分:0)
Dexter是一个Android库,可以简化在运行时请求权限的过程。