Android 5.0与6.0权限的不同
- 5.0以前,只需要manifest.xml中注册声明即可
- 5.0以后,用户可以在安装的时候关闭某些权限
- 6.0以后,对于一些用户隐私权限总是会在第一次提示用户是否授权权限
运行时权限的优势
- 新的权限机制更好的保护了用户的隐私
- 给了程序向用户说明权限的作用
- 可以防止一些恶意程序盗取用户或者手机信息,增强了android 系统的安全性
运行时权限的分类
- Normal Premission
- Dangerous Permission &&Dangerous Premission Group
权限组的概念
如果你申请某个危险的权限,假设你的app早已被用户授权了同一组的某个危险权限,那么系统会立即授权,而不需要用户去点击授权。
使用流程
1.在manifest中添加需要的权限,这里以打电话和写sd卡为例
<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.CALL_PHONE"/>
2.应用中一般不止一个地方会用到权限,所以对权限的处理最好放在父类中
BaseActivity 里有以下几个方法
/**
* 为子类提供权限检查方法
* @param premissions
* @return
*/
public boolean hasPremission(String... premissions){
for(String premission : premissions){
if(ContextCompat.checkSelfPermission(this,premission)
!= PackageManager.PERMISSION_GRANTED){
return false;
}
}
return true;
}
/**
* 为子类提供请求权限的方法
*/
public void requestPremission(int code ,String... premissions){
ActivityCompat.requestPermissions(this,premissions,code);
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
switch (requestCode){
case Constants.WRITE_EXTERNAL_STORAGE_CODE:
doSdPermission();
break;
case Constants.CALL_PHONE_CODE:
doCallPhone();
break;
}
}
/**
* 为子类提供一个打电话的方法
*/
public void doCallPhone() {
}
/**
* 为子类提供一个写sd卡的方法
*/
public void doSdPermission() {
}
在子类中继承BaseActivity
//写sd卡方法
private void doWriteSd() {
if(!hasPremission(Manifest.permission.WRITE_EXTERNAL_STORAGE)){
requestPremission(Constants.WRITE_EXTERNAL_STORAGE_CODE,Manifest.permission.WRITE_EXTERNAL_STORAGE);
}else {
Toast.makeText(this,"此处处理操作sd卡的动作如下载文件",Toast.LENGTH_SHORT).show();
}
}
//拨打电话
private void callPhone() {
if(!hasPremission(Manifest.permission.CALL_PHONE)){
requestPremission(Constants.CALL_PHONE_CODE,Manifest.permission.CALL_PHONE);
}else {
doCallPhone();
}
}
@Override
public void doCallPhone(){
//用intent启动拨打电话
Intent intent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:"+"10010"));
this.startActivity(intent);
}
此处贴出完整代码
public class BaseActivity extends AppCompatActivity {
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
/**
* 为子类提供权限检查方法
* @param premissions
* @return
*/
public boolean hasPremission(String... premissions){
for(String premission : premissions){
if(ContextCompat.checkSelfPermission(this,premission)
!= PackageManager.PERMISSION_GRANTED){
return false;
}
}
return true;
}
/**
* 为子类提供请求权限的方法
*/
public void requestPremission(int code ,String... premissions){
ActivityCompat.requestPermissions(this,premissions,code);
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
switch (requestCode){
case Constants.WRITE_EXTERNAL_STORAGE_CODE:
doSdPermission();
break;
case Constants.CALL_PHONE_CODE:
doCallPhone();
break;
}
}
/**
* 为子类提供一个打电话的方法
*/
public void doCallPhone() {
}
/**
* 为子类提供一个写sd卡的方法
*/
public void doSdPermission() {
}
}
public class MainActivity extends BaseActivity implements View.OnClickListener{
private Button sd;
private Button call;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
}
private void initView() {
sd = (Button)findViewById(R.id.btn_sd);
call = (Button)findViewById(R.id.btn_call);
sd.setOnClickListener(this);
call.setOnClickListener(this);
}
@Override
public void onClick(View v) {
switch (v.getId()){
case R.id.btn_sd:
doWriteSd();
break;
case R.id.btn_call:
callPhone();
break;
}
}
private void callPhone() {
if(!hasPremission(Manifest.permission.CALL_PHONE)){
requestPremission(Constants.CALL_PHONE_CODE,Manifest.permission.CALL_PHONE);
}else {
doCallPhone();
}
}
@Override
public void doCallPhone(){
//用intent启动拨打电话
Intent intent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:"+"10010"));
this.startActivity(intent);
}
private void doWriteSd() {
if(!hasPremission(Manifest.permission.WRITE_EXTERNAL_STORAGE)){
requestPremission(Constants.WRITE_EXTERNAL_STORAGE_CODE,Manifest.permission.WRITE_EXTERNAL_STORAGE);
}else {
Toast.makeText(this,"此处处理操作sd卡的动作如下载文件",Toast.LENGTH_SHORT).show();
}
}
}
mainfest.xml
<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.CALL_PHONE"/>
**
* 整个应用的常量放置
* Created by Lenovo on 2016/12/6.
*/
public class Constants {
public static final int WRITE_EXTERNAL_STORAGE_CODE = 0x01;
public static final int CALL_PHONE_CODE = 0x02;
}
布局
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<Button
android:id="@+id/btn_sd"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="读写SD卡" />
<Button
android:id="@+id/btn_call"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="读写SD卡" />
</LinearLayout>