Android開發時,到6.0系統上之后,有的權限就得申請才能用了。
Android將權限分為正常權限 和 危險權限
Android系統權限分為幾個保護級別。需要了解的兩個最重要保護級別是 正常權限 和 危險權限:
(1)正常權限:
涵蓋應用需要訪問其沙盒外部數據或資源,但對用戶隱私或其他應用操作風險很小的區域。
這些權限在應用安裝時授予,運行時不再詢問用戶。例如: 網絡訪問、WIFI狀態、音量設置等。
(2)危險權限:
涵蓋應用需要涉及用戶隱私信息的數據或資源,或者可能對用戶存儲的數據或其他應用的操作產生影響的區域。
例如: 讀取通訊錄、讀寫存儲器數據、獲取用戶位置等。如果應用聲明需要這些危險權限,則必須在運行時明確告訴用戶,讓用戶手動授予。
權限相關知識,權限表請看博客: Android6.0------權限管理
前提:APP運行在Android 6.0 (API level 23)或者更高級別的設備中,而且targetSdkVersion>=23時,系統將會自動采用動態權限管理策略,
先來看看效果圖:(注:如果未授權就點擊打電話或拍照就會直接閃退,由此6.0必須手動授權,開發時如果未授權,可以判斷並提示用戶從新授權)

上圖:
1:單個授權,電話授權。
2:有電話,SD卡,拍照授權三個一起授權
案例代碼:
public class MainActivity extends AppCompatActivity implements View.OnClickListener{ private static final int MY_PERMISSIONS_REQUEST_CALL_PHONE = 1; private static final int MY_PERMISSIONS_REQUEST_CALL_CAMERA = 2; String[] permissions = new String[]{ Manifest.permission.CAMERA, Manifest.permission.WRITE_EXTERNAL_STORAGE, Manifest.permission.CALL_PHONE }; // 聲明一個集合,在后面的代碼中用來存儲用戶拒絕授權的權 List<String> mPermissionList = new ArrayList<>(); @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); initView(); } private void initView(){ findViewById(R.id.btn1).setOnClickListener(this); findViewById(R.id.btn2).setOnClickListener(this); findViewById(R.id.btn3).setOnClickListener(this); findViewById(R.id.btn4).setOnClickListener(this); } @Override public void onClick(View v) { switch (v.getId()){ case R.id.btn1: //單個授權 //檢查版本是否大於M if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { if (ContextCompat.checkSelfPermission(this, Manifest.permission.CALL_PHONE) != PackageManager.PERMISSION_GRANTED) { ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.CALL_PHONE}, MY_PERMISSIONS_REQUEST_CALL_PHONE); }else { showToast("權限已申請"); } } break; case R.id.btn2://多個授權 mPermissionList.clear(); for (int i = 0; i < permissions.length; i++) { if (ContextCompat.checkSelfPermission(MainActivity.this, permissions[i]) != PackageManager.PERMISSION_GRANTED) { mPermissionList.add(permissions[i]); } } if (mPermissionList.isEmpty()) {//未授予的權限為空,表示都授予了 Toast.makeText(MainActivity.this,"已經授權",Toast.LENGTH_LONG).show(); } else {//請求權限方法 String[] permissions = mPermissionList.toArray(new String[mPermissionList.size()]);//將List轉為數組 ActivityCompat.requestPermissions(MainActivity.this, permissions, MY_PERMISSIONS_REQUEST_CALL_CAMERA); } break; case R.id.btn3: Intent camera = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); startActivityForResult(camera, 1); break; case R.id.btn4: Intent intent = new Intent(Intent.ACTION_CALL); Uri data = Uri.parse("tel:" + "10086"); intent.setData(data); startActivity(intent); break; } } @Override public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) { if (requestCode == MY_PERMISSIONS_REQUEST_CALL_PHONE) { if (grantResults[0] == PackageManager.PERMISSION_GRANTED) { showToast("權限已申請"); } else { showToast("權限已拒絕"); } }else if (requestCode == MY_PERMISSIONS_REQUEST_CALL_CAMERA){ for (int i = 0; i < grantResults.length; i++) { if (grantResults[i] != PackageManager.PERMISSION_GRANTED) { //判斷是否勾選禁止后不再詢問 boolean showRequestPermission = ActivityCompat.shouldShowRequestPermissionRationale(MainActivity.this, permissions[i]); if (showRequestPermission) { showToast("權限未申請"); } } } } super.onRequestPermissionsResult(requestCode, permissions, grantResults); } private void showToast(String string){ Toast.makeText(MainActivity.this,string,Toast.LENGTH_LONG).show(); } }
前提一定要注意:AndroidManifest中:
<uses-permission android:name="android.permission.CALL_PHONE"/> //電話 <uses-permission android:name="android.permission.CAMERA"/> //拍照 <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> //sd卡
此案例是自己全部用Java代碼寫的,項目危險權限少則推薦使用,多的話就自己封裝或者借助第三方了。
權限申請有很多第三方封裝好的庫(工具類)可以實現,Github上一大把,后續博客將找幾個好一點的來講解一下。
