Android权限机制
在Android6.0系统之前,如果安装程序,会在程序提示时改程序需要申请的权限,从而让用于决定是否安装该程序;也可以随时在应用程序管理界面中查看一个程序的权限申请情况。
而在Android6.0系统中加入了运行时权限功能,从而更好的保护用户的安全和隐私。用于不需要在安装软件的时候一次性授权所有的申请权限,而是可以在软件使用的过程中再对某一项权限申请进行授权。
也不是所有的权限都需要在运行时授权。
对于不会直接威胁到用户安全和隐私的权限申请,系统会自动帮用户授权,称为普通权限。在申请时只需要在AndroidManifest.xml
中添加一下权限声明即可。
而一些可能会触及用户隐私和安全的权限,就需要用户自己授权,称为危险权限。在申请时就需要进行运行时权限处理。同时,当用户同意某个危险权限时,同组的其他权限也会被系统自动授权。
在程序运行时申请权限
以CALL_PHONE
权限为例,这个权限是在编写拨打电话功能的时候需要声明的。
Android 6.0 之前
首先在activity_main.xml
文件中添加一个id
为makeCallBtn
的按钮。
然后在MainActivity
中添加他的点击事件出发拨打电话的逻辑
class MainActivity : AppCompatActivity() {
private lateinit var mainBinding: ActivityMainBinding
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
mainBinding = ActivityMainBinding.inflate(layoutInflater)
setContentView(mainBinding.root)
mainBinding.makeCallBtn.setOnClickListener {
try {
val intent = Intent(Intent.ACTION_CALL) //系统内置的打电话动作
intent.data = Uri.parse("tel:10086") //协议是tel 电话是10086
startActivity(intent)
}catch (e: SecurityException){
e.printStackTrace()
}
}
}
}
最后记得在AndroidManifest.xml
文件中申请权限
这样在Android6.0系统
以下的手机上是可以正常运转的,而在6.0以上的系统手机中就会报错。
Android 6.0 之后
只需修改MainActivity
中的代码
class MainActivity : AppCompatActivity() {
private lateinit var mainBinding: ActivityMainBinding
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
mainBinding = ActivityMainBinding.inflate(layoutInflater)
setContentView(mainBinding.root)
mainBinding.makeCallBtn.setOnClickListener {
//首先通过checkSelfPermission()方法判断用户是否授权
//将其返回值于PackageManager.PERMISSION_GRANTED作比较,相同即为授权
if(ContextCompat.checkSelfPermission(this,
Manifest.permission.CALL_PHONE) != PackageManager.PERMISSION_GRANTED){
//没有授权则向用户申请
//requestPermissions()方法调用后会弹出向用户请求权限的对话框
//对话框的结果会返会在下面的方法中onRequestPermissionsResult()
ActivityCompat.requestPermissions(this,
arrayOf(Manifest.permission.CALL_PHONE), 1)
} else {
//授权则直接拨打电话
call()
}
}
}
override fun onRequestPermissionsResult(requestCode: Int, permissions: Array<out String>, grantResults: IntArray) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults)
when(requestCode){
1 -> {
if (grantResults.isNotEmpty() && grantResults[0] == PackageManager.PERMISSION_GRANTED){
//同意则拨打电话
call()
} else {
//不同意则弹窗提示
Toast.makeText(this,"用户拒绝了申请",Toast.LENGTH_SHORT).show()
}
}
}
}
private fun call() {
try {
val intent = Intent(Intent.ACTION_CALL) //系统内置的打电话动作
intent.data = Uri.parse("tel:10086") //协议是tel 电话是10086
startActivity(intent)
}catch (e: SecurityException){
e.printStackTrace()
}
}
}
点击按钮后弹出对话框询问
拒绝后:
同意后: