android 6.0 权限bug、权限处理

来源:互联网 发布:centos 7 dhcp ip配置 编辑:程序博客网 时间:2024/06/07 11:16

android 6.0 权限处理

在Android6.0(M)之前,在用户安装应用的时候会产生一个权限列表,只有用户允许这些权限后,应用才可以正常的安装,这就会产生一个问题,这些权限对用户是不具有感知性的,也就是说用户都不知道你要这些权限干什么,我明明装的是一个阅读类型的应用,你却要我拨打电话的权限,你想干嘛呢?当然绝大部分的开发者是善意的,但也避免不了一些特殊人群利用这些“漏洞”做一些不好的事情。
而在Android6.0(M)之后,用户是可以不管权限直接安装应用的,当应用需要调用某些权限的时候,会给予用户一个通知与说明,我要这些权限干什么,这样下来可以让用户有更加清醒的权限分配意识,也在一定程度上更加人性化的保护了用户的隐私,避免了“权限一刀切”。
权限的分组

在Android6.0(M)之后,对权限进行了分类,大致有这三种:

·普通权限·危险权限·特殊权限

普通权限
也就是正常权限,是对手机的一些正常操作,对用户的隐私没有太大影响的权限,比如手机的震动,网络访问,蓝牙等权限,这些权限会在应用被安装的时候默认授予,用户不能拒绝,也不能取消。
普通权限列表:

ACCESS_LOCATION_EXTRA_COMMANDSACCESS_NETWORK_STATEACCESS_NOTIFICATION_POLICYACCESS_WIFI_STATEBLUETOOTHBLUETOOTH_ADMINBROADCAST_STICKYCHANGE_NETWORK_STATECHANGE_WIFI_MULTICAST_STATECHANGE_WIFI_STATEDISABLE_KEYGUARDEXPAND_STATUS_BARGET_PACKAGE_SIZEINTERNETKILL_BACKGROUND_PROCESSESMODIFY_AUDIO_SETTINGSNFCREAD_SYNC_SETTINGSREAD_SYNC_STATSRECEIVE_BOOT_COMPLETEDREORDER_TASKSREQUEST_INSTALL_PACKAGESSET_TIME_ZONESET_WALLPAPERSET_WALLPAPER_HINTSTRANSMIT_IRUSE_FINGERPRINTVIBRATEWAKE_LOCKWRITE_SYNC_SETTINGSSET_ALARMINSTALL_SHORTCUTUNINSTALL_SHORTCUT

对于上面这些权限,需要和Android6.0(M)之前的系统,在AndroidManifest.xml声明即可。

危险权限
其实就是运行中需要处理的权限,也是我们最需要注意的权限,这些权限会关系到用户的隐私或影响到其他应用的运行,这些危险权限,谷歌还做了一个权限组,以分组的形式来呈现:

这些就是危险权限

比如举个例子说明

  1. 权限

    在AndroidManifest中, 添加两个权限, 录音和修改音量.

 <!--危险权限-->    <uses-permission android:name="android.permission.RECORD_AUDIO"/>    <!--一般权限-->    <uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS"/>

检测权限类

public class PermissionsChecker {    private final Context mContext;    public PermissionsChecker(Context context) {        mContext = context.getApplicationContext();    }    // 判断权限集合    public boolean lacksPermissions(String... permissions) {        for (String permission : permissions) {            if (lacksPermission(permission)) {                return true;            }        }        return false;    }    // 判断是否缺少权限    private boolean lacksPermission(String permission) {        return ContextCompat.checkSelfPermission(mContext, permission) ==                PackageManager.PERMISSION_DENIED;    }}
  1. 首页

    假设首页需要使用权限, 在页面显示前, 即onResume时, 检测权限,
    如果缺少, 则进入权限获取页面; 接收返回值, 拒绝权限时, 直接关闭.

public class MainActivity extends AppCompatActivity {    private static final int REQUEST_CODE = 0; // 请求码    // 所需的全部权限    private static final String[] PERMISSIONS = new String[]{Manifest.permission.READ_PHONE_STATE, Manifest.permission.CAMERA, Manifest.permission.WRITE_EXTERNAL_STORAGE,            Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.ACCESS_FINE_LOCATION, Manifest.permission.ACCESS_COARSE_LOCATION, Manifest.permission.CALL_PHONE,            Manifest.permission.READ_CONTACTS, Manifest.permission.RECORD_AUDIO, Manifest.permission.CALL_PHONE};    Toolbar mTToolbar;    private PermissionsChecker mPermissionsChecker; // 权限检测器    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        mTToolbar = (Toolbar) findViewById(R.id.toolbar);        setSupportActionBar(mTToolbar);        mPermissionsChecker = new PermissionsChecker(this);    }    @Override    protected void onResume() {        super.onResume();        // 缺少权限时, 进入权限配置页面        if (mPermissionsChecker.lacksPermissions(PERMISSIONS)) {            startPermissionsActivity();        }    }    private void startPermissionsActivity() {        PermissionsActivity.startActivityForResult(this, REQUEST_CODE, PERMISSIONS);    }    @Override    protected void onActivityResult(int requestCode, int resultCode, Intent data) {        super.onActivityResult(requestCode, resultCode, data);        // 拒绝时, 关闭页面, 缺少主要权限, 无法运行        if (requestCode == REQUEST_CODE && resultCode == PermissionsActivity.PERMISSIONS_DENIED) {            finish();        }    }}
  1. 授权页

    授权页, 首先使用系统默认的授权页, 当用户拒绝时, 指导用户手动设置, 当用户再次操作失败后, 返回继续提示. 用户手动退出授权页时, 给使用页发送授权失败的通知.

/** * 权限获取页面 * <p/> * Created by wangchenlong on 16/1/26. */public class PermissionsActivity extends AppCompatActivity {    public static final int PERMISSIONS_GRANTED = 0; // 权限授权    public static final int PERMISSIONS_DENIED = 1; // 权限拒绝    private static final int PERMISSION_REQUEST_CODE = 0; // 系统权限管理页面的参数    private static final String EXTRA_PERMISSIONS =            "xiangzi.ceshi.permission.extra_permission"; // 权限参数    private static final String PACKAGE_URL_SCHEME = "package:"; // 方案    private PermissionsChecker mChecker; // 权限检测器    private boolean isRequireCheck; // 是否需要系统权限检测    // 启动当前权限页面的公开接口    public static void startActivityForResult(Activity activity, int requestCode, String... permissions) {        Intent intent = new Intent(activity, PermissionsActivity.class);        intent.putExtra(EXTRA_PERMISSIONS, permissions);        ActivityCompat.startActivityForResult(activity, intent, requestCode, null);    }    @Override protected void onCreate(@Nullable Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        if (getIntent() == null || !getIntent().hasExtra(EXTRA_PERMISSIONS)) {            throw new RuntimeException("PermissionsActivity需要使用静态startActivityForResult方法启动!");        }        setContentView(R.layout.activity_permissions);        mChecker = new PermissionsChecker(this);        isRequireCheck = true;    }    @Override protected void onResume() {        super.onResume();        if (isRequireCheck) {            String[] permissions = getPermissions();            if (mChecker.lacksPermissions(permissions)) {                requestPermissions(permissions); // 请求权限            } else {                allPermissionsGranted(); // 全部权限都已获取            }        } else {            isRequireCheck = true;        }    }    // 返回传递的权限参数    private String[] getPermissions() {        return getIntent().getStringArrayExtra(EXTRA_PERMISSIONS);    }    // 请求权限兼容低版本    private void requestPermissions(String... permissions) {        ActivityCompat.requestPermissions(this, permissions, PERMISSION_REQUEST_CODE);    }    // 全部权限均已获取    private void allPermissionsGranted() {        setResult(PERMISSIONS_GRANTED);        finish();    }    /**     * 用户权限处理,     * 如果全部获取, 则直接过.     * 如果权限缺失, 则提示Dialog.     *     * @param requestCode  请求码     * @param permissions  权限     * @param grantResults 结果     */    @Override    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {        if (requestCode == PERMISSION_REQUEST_CODE && hasAllPermissionsGranted(grantResults)) {            isRequireCheck = true;            allPermissionsGranted();        } else {            isRequireCheck = false;            showMissingPermissionDialog();        }    }    // 含有全部的权限    private boolean hasAllPermissionsGranted(@NonNull int[] grantResults) {        for (int grantResult : grantResults) {            if (grantResult == PackageManager.PERMISSION_DENIED) {                return false;            }        }        return true;    }    // 显示缺失权限提示    private void showMissingPermissionDialog() {        AlertDialog.Builder builder = new AlertDialog.Builder(PermissionsActivity.this);        builder.setTitle(R.string.help);        builder.setMessage(R.string.string_help_text);        // 拒绝, 退出应用        builder.setNegativeButton(R.string.quit, new DialogInterface.OnClickListener() {            @Override public void onClick(DialogInterface dialog, int which) {                setResult(PERMISSIONS_DENIED);                finish();            }        });        builder.setPositiveButton(R.string.settings, new DialogInterface.OnClickListener() {            @Override public void onClick(DialogInterface dialog, int which) {                startAppSettings();            }        });        builder.show();    }    // 启动应用的设置    private void startAppSettings() {        Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);        intent.setData(Uri.parse(PACKAGE_URL_SCHEME + getPackageName()));        startActivity(intent);    }}

这是我用的布局

<?xml version="1.0" encoding="utf-8"?><View xmlns:android="http://schemas.android.com/apk/res/android"    android:id="@+id/image"    android:layout_width="match_parent"    android:layout_height="match_parent"    android:layout_weight="1"    android:adjustViewBounds="true"    android:background="#88888888"    android:contentDescription="@null" />

进入之后你会看到如下
这里写图片描述

注意如果权限没有全部都给,就会变成一个再次向用户请求权限要求

0 0
原创粉丝点击