Android权限管理之Permission权限机制及使用详解
作者:总李写代码
前言:
最近突然喜欢上一句诗:“宠辱不惊,看庭前花开花落;去留无意,望天空云卷云舒。” 哈哈~,这个和今天的主题无关,最近只要不学习总觉得生活中少了点什么,所以想着围绕着最近面试过程中讨论比较多的一个知识点Android 6.0 权限适配问题来进行学习,不过我不想直接进入这个主题,所以选择先去了解一下Android的Permission权限机制及使用
Android权限机制:
权限是一种安全机制。Android权限主要用于限制应用程序内部某些具有限制性特性的功能使用以及应用程序之间的组件访问。
Android权限列表:
访问登记属性 | android.permission.ACCESS_CHECKIN_PROPERTIES ,读取或写入登记check-in数据库属性表的权限 |
获取错略位置 | android.permission.ACCESS_COARSE_LOCATION,通过WiFi或移动基站的方式获取用户错略的经纬度信息,定位精度大概误差在30~1500米 |
获取精确位置 | android.permission.ACCESS_FINE_LOCATION,通过GPS芯片接收卫星的定位信息,定位精度达10米以内 |
访问定位额外命令 | android.permission.ACCESS_LOCATION_EXTRA_COMMANDS,允许程序访问额外的定位提供者指令 |
获取模拟定位信息 | android.permission.ACCESS_MOCK_LOCATION,获取模拟定位信息,一般用于帮助开发者调试应用 |
获取网络状态 | android.permission.ACCESS_NETWORK_STATE,获取网络信息状态,如当前的网络连接是否有效 |
访问Surface Flinger | android.permission.ACCESS_SURFACE_FLINGER,Android平台上底层的图形显示支持,一般用于游戏或照相机预览界面和底层模式的屏幕截图 |
获取WiFi状态 | android.permission.ACCESS_WIFI_STATE,获取当前WiFi接入的状态以及WLAN热点的信息 |
账户管理 | android.permission.ACCOUNT_MANAGER,获取账户验证信息,主要为GMail账户信息,只有系统级进程才能访问的权限 |
验证账户 | android.permission.AUTHENTICATE_ACCOUNTS,允许一个程序通过账户验证方式访问账户管理ACCOUNT_MANAGER相关信息 |
电量统计 | android.permission.BATTERY_STATS,获取电池电量统计信息 |
绑定小插件 | android.permission.BIND_APPWIDGET,允许一个程序告诉appWidget服务需要访问小插件的数据库,只有非常少的应用才用到此权限 |
绑定设备管理 | android.permission.BIND_DEVICE_ADMIN,请求系统管理员接收者receiver,只有系统才能使用 |
绑定输入法 | android.permission.BIND_INPUT_METHOD ,请求InputMethodService服务,只有系统才能使用 |
绑定RemoteView | android.permission.BIND_REMOTEVIEWS,必须通过RemoteViewsService服务来请求,只有系统才能用 |
绑定壁纸 | android.permission.BIND_WALLPAPER,必须通过WallpaperService服务来请求,只有系统才能用 |
使用蓝牙 | android.permission.BLUETOOTH,允许程序连接配对过的蓝牙设备 |
蓝牙管理 | android.permission.BLUETOOTH_ADMIN,允许程序进行发现和配对新的蓝牙设备 |
变成砖头 | android.permission.BRICK,能够禁用手机,非常危险,顾名思义就是让手机变成砖头 |
应用删除时广播 | android.permission.BROADCAST_PACKAGE_REMOVED,当一个应用在删除时触发一个广播 |
收到短信时广播 | android.permission.BROADCAST_SMS,当收到短信时触发一个广播 |
连续广播 | android.permission.BROADCAST_STICKY,允许一个程序收到广播后快速收到下一个广播 |
WAP PUSH广播 | android.permission.BROADCAST_WAP_PUSH,WAP PUSH服务收到后触发一个广播 |
拨打电话 | android.permission.CALL_PHONE,允许程序从非系统拨号器里输入电话号码 |
通话权限 | android.permission.CALL_PRIVILEGED,允许程序拨打电话,替换系统的拨号器界面 |
拍照权限 | android.permission.CAMERA,允许访问摄像头进行拍照 |
改变组件状态 | android.permission.CHANGE_COMPONENT_ENABLED_STATE,改变组件是否启用状态 |
改变配置 | android.permission.CHANGE_CONFIGURATION,允许当前应用改变配置,如定位 |
改变网络状态 | android.permission.CHANGE_NETWORK_STATE,改变网络状态如是否能联网 |
改变WiFi多播状态 | android.permission.CHANGE_WIFI_MULTICAST_STATE,改变WiFi多播状态 |
改变WiFi状态 | android.permission.CHANGE_WIFI_STATE,改变WiFi状态 |
清除应用缓存 | android.permission.CLEAR_APP_CACHE,清除应用缓存 |
清除用户数据 | android.permission.CLEAR_APP_USER_DATA,清除应用的用户数据 |
底层访问权限 | android.permission.CWJ_GROUP,允许CWJ账户组访问底层信息 |
手机优化大师扩展权限 | android.permission.CELL_PHONE_MASTER_EX,手机优化大师扩展权限 |
控制定位更新 | android.permission.CONTROL_LOCATION_UPDATES,允许获得移动网络定位信息改变 |
删除缓存文件 | android.permission.DELETE_CACHE_FILES,允许应用删除缓存文件 |
删除应用 | android.permission.DELETE_PACKAGES,允许程序删除应用 |
电源管理 | android.permission.DEVICE_POWER,允许访问底层电源管理 |
应用诊断 | android.permission.DIAGNOSTIC,允许程序到RW到诊断资源 |
禁用键盘锁 | android.permission.DISABLE_KEYGUARD,允许程序禁用键盘锁 |
转存系统信息 | android.permission.DUMP,允许程序获取系统dump信息从系统服务 |
状态栏控制 | android.permission.EXPAND_STATUS_BAR,允许程序扩展或收缩状态栏 |
工厂测试模式 | android.permission.FACTORY_TEST,允许程序运行工厂测试模式 |
使用闪光灯 | android.permission.FLASHLIGHT,允许访问闪光灯 |
强制后退 | android.permission.FORCE_BACK,允许程序强制使用back后退按键,无论Activity是否在顶层 |
访问账户Gmail列表 | android.permission.GET_ACCOUNTS,访问GMail账户列表 |
获取应用大小 | android.permission.GET_PACKAGE_SIZE,获取应用的文件大小 |
获取任务信息 | android.permission.GET_TASKS,允许程序获取当前或最近运行的应用 |
允许全局搜索 | android.permission.GLOBAL_SEARCH,允许程序使用全局搜索功能 |
硬件测试 | android.permission.HARDWARE_TEST,访问硬件辅助设备,用于硬件测试 |
注射事件 | android.permission.INJECT_EVENTS,允许访问本程序的底层事件,获取按键、轨迹球的事件流 |
安装定位提供 | android.permission.INSTALL_LOCATION_PROVIDER,安装定位提供 |
安装应用程序 | android.permission.INSTALL_PACKAGES,允许程序安装应用 |
内部系统窗口 | android.permission.INTERNAL_SYSTEM_WINDOW,允许程序打开内部窗口,不对第三方应用程序开放此权限 |
访问网络 | android.permission.INTERNET,访问网络连接,可能产生GPRS流量 |
结束后台进程 | android.permission.KILL_BACKGROUND_PROCESSES,允许程序调用killBackgroundProcesses(String).方法结束后台进程 |
管理账户 | android.permission.MANAGE_ACCOUNTS,允许程序管理AccountManager中的账户列表 |
管理程序引用 | android.permission.MANAGE_APP_TOKENS,管理创建、摧毁、Z轴顺序,仅用于系统 |
高级权限 | android.permission.MTWEAK_USER,允许mTweak用户访问高级系统权限 |
社区权限 | android.permission.MTWEAK_FORUM,允许使用mTweak社区权限 |
软格式化 | android.permission.MASTER_CLEAR,允许程序执行软格式化,删除系统配置信息 |
修改声音设置 | android.permission.MODIFY_AUDIO_SETTINGS,修改声音设置信息 |
修改电话状态 | android.permission.MODIFY_PHONE_STATE,修改电话状态,如飞行模式,但不包含替换系统拨号器界面 |
格式化文件系统 | android.permission.MOUNT_FORMAT_FILESYSTEMS,格式化可移动文件系统,比如格式化清空SD卡 |
挂载文件系统 | android.permission.MOUNT_UNMOUNT_FILESYSTEMS,挂载、反挂载外部文件系统 |
允许NFC通讯 | android.permission.NFC,允许程序执行NFC近距离通讯操作,用于移动支持 |
永久Activity | android.permission.PERSISTENT_ACTIVITY,创建一个永久的Activity,该功能标记为将来将被移除 |
处理拨出电话 | android.permission.PROCESS_OUTGOING_CALLS,允许程序监视,修改或放弃播出电话 |
读取日程提醒 | android.permission.READ_CALENDAR,允许程序读取用户的日程信息 |
读取联系人 | android.permission.READ_CONTACTS,允许应用访问联系人通讯录信息 |
屏幕截图 | android.permission.READ_FRAME_BUFFER,读取帧缓存用于屏幕截图 |
读取收藏夹和历史记录 | com.android.browser.permission.READ_HISTORY_BOOKMARKS,读取浏览器收藏夹和历史记录 |
读取输入状态 | android.permission.READ_INPUT_STATE,读取当前键的输入状态,仅用于系统 |
读取系统日志 | android.permission.READ_LOGS,读取系统底层日志 |
读取电话状态 | android.permission.READ_PHONE_STATE,访问电话状态 |
读取短信内容 | android.permission.READ_SMS,读取短信内容 |
读取同步设置 | android.permission.READ_SYNC_SETTINGS,读取同步设置,读取Google在线同步设置 |
读取同步状态 | android.permission.READ_SYNC_STATS,读取同步状态,获得Google在线同步状态 |
重启设备 | android.permission.REBOOT,允许程序重新启动设备 |
开机自动允许 | android.permission.RECEIVE_BOOT_COMPLETED,允许程序开机自动运行 |
接收彩信 | android.permission.RECEIVE_MMS,接收彩信 |
接收短信 | android.permission.RECEIVE_SMS,接收短信 |
接收Wap Push | android.permission.RECEIVE_WAP_PUSH,接收WAP PUSH信息 |
录音 | android.permission.RECORD_AUDIO,录制声音通过手机或耳机的麦克 |
排序系统任务 | android.permission.REORDER_TASKS,重新排序系统Z轴运行中的任务 |
结束系统任务 | android.permission.RESTART_PACKAGES,结束任务通过restartPackage(String)方法,该方式将在外来放弃 |
发送短信 | android.permission.SEND_SMS,发送短信 |
设置Activity观察其 | android.permission.SET_ACTIVITY_WATCHER,设置Activity观察器一般用于monkey测试 |
设置闹铃提醒 | com.android.alarm.permission.SET_ALARM,设置闹铃提醒 |
设置总是退出 | android.permission.SET_ALWAYS_FINISH,设置程序在后台是否总是退出 |
设置动画缩放 | android.permission.SET_ANIMATION_SCALE,设置全局动画缩放 |
设置调试程序 | android.permission.SET_DEBUG_APP,设置调试程序,一般用于开发 |
设置屏幕方向 | android.permission.SET_ORIENTATION,设置屏幕方向为横屏或标准方式显示,不用于普通应用 |
设置应用参数 | android.permission.SET_PREFERRED_APPLICATIONS,设置应用的参数,已不再工作具体查看addPackageToPreferred(String) 介绍 |
设置进程限制 | android.permission.SET_PROCESS_LIMIT,允许程序设置最大的进程数量的限制 |
设置系统时间 | android.permission.SET_TIME,设置系统时间 |
设置系统时区 | android.permission.SET_TIME_ZONE,设置系统时区 |
设置桌面壁纸 | android.permission.SET_WALLPAPER,设置桌面壁纸 |
设置壁纸建议 | android.permission.SET_WALLPAPER_HINTS,设置壁纸建议 |
发送永久进程信号 | android.permission.SIGNAL_PERSISTENT_PROCESSES,发送一个永久的进程信号 |
状态栏控制 | android.permission.STATUS_BAR,允许程序打开、关闭、禁用状态栏 |
访问订阅内容 | android.permission.SUBSCRIBED_FEEDS_READ,访问订阅信息的数据库 |
写入订阅内容 | android.permission.SUBSCRIBED_FEEDS_WRITE,写入或修改订阅内容的数据库 |
显示系统窗口 | android.permission.SYSTEM_ALERT_WINDOW,显示系统窗口 |
更新设备状态 | android.permission.UPDATE_DEVICE_STATS,更新设备状态 |
使用证书 | android.permission.USE_CREDENTIALS,允许程序请求验证从AccountManager |
使用SIP视频 | android.permission.USE_SIP,允许程序使用SIP视频服务 |
使用振动 | android.permission.VIBRATE,允许振动 |
唤醒锁定 | android.permission.WAKE_LOCK,允许程序在手机屏幕关闭后后台进程仍然运行 |
写入GPRS接入点设置 | android.permission.WRITE_APN_SETTINGS,写入网络GPRS接入点设置 |
写入日程提醒 | android.permission.WRITE_CALENDAR,写入日程,但不可读取 |
写入联系人 | android.permission.WRITE_CONTACTS,写入联系人,但不可读取 |
写入外部存储 | android.permission.WRITE_EXTERNAL_STORAGE,允许程序写入外部存储,如SD卡上写文件 |
写入Google地图数据 | android.permission.WRITE_GSERVICES,允许程序写入Google Map服务数据 |
写入收藏夹和历史记录 | com.android.browser.permission.WRITE_HISTORY_BOOKMARKS,写入浏览器历史记录或收藏夹,但不可读取 |
读写系统敏感设置 | android.permission.WRITE_SECURE_SETTINGS,允许程序读写系统安全敏感的设置项 |
读写系统设置 | android.permission.WRITE_SETTINGS,允许读写系统设置项 |
编写短信 | android.permission.WRITE_SMS,允许编写短信 |
写入在线同步设置 | android.permission.WRITE_SYNC_SETTINGS,写入Google在线同步设置 |
Android权限使用:
1.)在 AndroidManifest.xml <manifest>标签内使用<uses-permission>声明使用某一个权限
<uses-permission android:name="string"/>
例如申请使用网络权限
<uses-permission android:name="android.permission.INTERNET"/>
如果特定的权限必须申明使用,如果没有申请使用就会报出Permission Denial错误,例如访问通讯录报出如下错误
Caused by: java.lang.SecurityException: Permission Denial: reading com.androintacts.ContactsProvider2 uri content://contacts/data/phones from pid=23763, uid=10036 requires android.permission.READ_CONTACTS
解决此类错误我们只需根据提示添加对应的权限即可
<uses-permission android:name="android.permission.READ_CONTACTS" />
2.)自定义权限permission
虽然这种使用场景不多见,但是在有些特定的场景下出于安全考虑就需要自定义权限了,比如两个APP之间需要共享数据而采用了ContentProvider,此时我们需要对一个app访问另外一个app的数据时需要添加权限申请。自定义权限通过<permission>标签
<permission android:description=”string resource” android:icon=”drawable resource” android:label=”string resource” android:name=”string” android:permissionGroup=”string” android:protectionLevel=[“normal” | “dangerous” | “signature” | “signatureOrSystem”] />
- android:description:对权限的描述,比lable更加的详细,介绍该权限的相关使用情况,比如当用户被询问是否给其他应用该权限时。注意描述应该使用的是string资源,而不是直接使用string串。
- android:icon:用来标识该权限的一个图标。
- android:label:权限的一个给用户展示的简短描述。方便的来说,这个可以直接使用一个string字串来表示,但是如果要发布的话,还是应该使用string资源来表示。
- android:name:权限的唯一名字,由于独立性,一般都是使用包名加权限名,该属性是必须的,其他的可选,未写的系统会指定默认值。
- android:permissionGroup: 权限所属权限组的名称,并且需要在这个或其他应用中使用<permission-group>标签提前声明该名称,如果没有声明,该权限就不属于该组。
- android:protectionLevel:权限的等级
- 关于android:protectionLevel:权限的等级
- normal 低风险权限,只要申请了就可以使用(在AndroidManifest.xml中添加<uses-permission>标签),安装时不需要用户确认;
- dangerous 高风险权限,安装时需要用户的确认才可使用;
- signature 只有当申请权限的应用程序的数字签名与声明此权限的应用程序的数字签名相同时(如果是申请系统权限,则需要与系统签名相同),才能将权限授给它;
- signatureOrSystem 签名相同,或者申请权限的应用为系统应用(在system image中),与signature类似,只是增加了rom中自带的app的声明 ,尽量不要使用该选项,因为signature已经适合绝大部分的情况。
对于普通和危险级别的权限,我们称之为低级权限,应用申请即授予。其他两级权限,我们称之为高级权限或系统权限。当应用试图在没有权限的情况下做受限操作,应用将被系统杀掉以警示。系统应用可以使用任何权限。权限的声明者可无条件使用该权限。
根据上面的介绍我们实战一下自定义权限如下
<permission android:name="personprovider.permission.read" android:description="@string/personprovider_permission_read_desstring" android:icon="@mipmap/ic_launcher" android:label="@string/personprovider_permission_read_labestring" android:protectionLevel="normal"/> <permission android:name="personprovider.permission.write" android:description="@string/personprovider_permission_write_desstring" android:icon="@mipmap/ic_launcher" android:label="@string/personprovider_permission_write_labestring" android:protectionLevel="normal"/>
3.)自定义权限组permission-group
为了方便管理某一特定功能的权限,所以我会对权限进行分组,这时我们需要通过<permission-group>自定义一个权限组
<permission-group android:description="string resource" android:icon="drawable resource" android:label="string resource" android:name="string"/>
- android:description 这个属性用于给权限组定义一个用户可读的说明性文本。这个文本应该比标签更长、更详细。这个属性必须要引用一个字符串资源,跟label属性不一样,它不能够使用原生的字符串。
- android:icon 这个属性定义了一个代表权限的图标。这个属性要使用包含图片定义的可绘制资源来定义。
- android:label 这个属性给权限组定义了一个用户可读的名称。为了开发方便,在开发时,可以直接使用原生的字符串来设置这个属性。但是,当应用程序正式发布时,应该使用字符串资源来设置,以便能够像用户界面中其他的字符串一样能够被本地化。
- android:name 这个属性定义了权限组的名称,它是能够分配给<permission>元素的permissionGroup属性的名称。
上面的权限组我们可以自定义为下面这种方式:
<permission-group android:name="personprovider.permission-group" android:description="@string/personprovider_permission_group_desstring" android:icon="@mipmap/ic_launcher" android:label="@string/personprovider_permission_group_labelstring"/> <permission android:name="personprovider.permission.read" android:description="@string/personprovider_permission_read_desstring" android:icon="@mipmap/ic_launcher" android:label="@string/personprovider_permission_read_labestring" android:permissionGroup="personprovider.permission-group" android:protectionLevel="normal"/> <permission android:name="personprovider.permission.write" android:description="@string/personprovider_permission_write_desstring" android:icon="@mipmap/ic_launcher" android:label="@string/personprovider_permission_write_labestring" android:permissionGroup="personprovider.permission-group" android:protectionLevel="normal"/>
其他关于permission-tree这里就不介绍了,这个至今没有用到过,<permission-tree>是为一组permissions声明了一个namespace。
4.)使用自定义权限
申请权限
<uses-permission android:name="personprovider.permission.read"/> <uses-permission android:name="personprovider.permission.write"/>
在宿主上声明需要申请访问权限
<provider android:name=".PersonProvider" android:authorities="com.whoislcj.testsqlite.personprovider" android:enabled="true" android:exported="true" android:readPermission="personprovider.permission..read" android:writePermission="personprovider.permission..write"/>
总结:
本篇大致了解了Android的权限permission以及如何使用和自定义权限,下篇将总结学一下如何解决Android 6.0的权限的适配问题。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。