前言
首先 Android 的權(quán)限大致分為三種:
普通權(quán)限:只需要在清單文件中注冊(cè)即可
危險(xiǎn)權(quán)限:需要在代碼中動(dòng)態(tài)申請(qǐng),以彈系統(tǒng) Dialog 的形式進(jìn)行請(qǐng)求
特殊權(quán)限:需要在代碼中動(dòng)態(tài)申請(qǐng),以跳系統(tǒng) Activity 的形式進(jìn)行請(qǐng)求
而我們今天要講的主題,是關(guān)于存儲(chǔ)權(quán)限,在 Android 6.0 之后就變成了危險(xiǎn)權(quán)限,而到了 Android 11 上面變成了特殊權(quán)限,而最明顯的區(qū)別是一個(gè)是通過(guò) Dialog 展示給用戶看,另外一個(gè)是通過(guò) Activity 展現(xiàn)給用戶看。

image

image
- 從危險(xiǎn)權(quán)限到特殊權(quán)限,這不單單是權(quán)限的屬性發(fā)生了變化,還是申請(qǐng)方式的發(fā)生了不同
Android 10.0 以下存儲(chǔ)權(quán)限適配
- 升級(jí) targetSdkVersion
android
defaultConfig {
targetSdkVersion 23
}
}
- 添加清單權(quán)限
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
- 代碼動(dòng)態(tài)申請(qǐng)
public final class PermissionActivity extends AppCompatActivity {
private static final int REQUEST_CODE = 1024;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestPermission();
}
private void requestPermission() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
// 先判斷有沒(méi)有權(quán)限
if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED &&
ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) {
writeFile();
} else {
ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE}, REQUEST_CODE);
}
} else {
writeFile();
}
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode == REQUEST_CODE) {
if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED &&
ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) {
writeFile();
} else {
ToastUtils.show("存儲(chǔ)權(quán)限獲取失敗");
}
}
}
/**
* 模擬文件寫(xiě)入
*/
private void writeFile() {
ToastUtils.show("寫(xiě)入文件成功");
}
}

image
- 需要注意的是,如果 targetSdkVersion >= 29 上,還需要在清單文件中加上
<application
android:requestLegacyExternalStorage="true">
···
- 否則就算申請(qǐng)了存儲(chǔ)權(quán)限,在安卓 10.0 的設(shè)備上將無(wú)法正常讀寫(xiě)外部存儲(chǔ)上的文件
Android 11 及以上申請(qǐng)存儲(chǔ)權(quán)限
- 升級(jí) targetSdkVersion
android
defaultConfig {
targetSdkVersion 30
}
}
- 添加清單權(quán)限
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE" />
- 代碼動(dòng)態(tài)申請(qǐng)
public final class PermissionActivity extends AppCompatActivity {
private static final int REQUEST_CODE = 1024;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestPermission();
}
private void requestPermission() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
// 先判斷有沒(méi)有權(quán)限
if (Environment.isExternalStorageManager()) {
writeFile();
} else {
startActivityForResult(new Intent(Settings.ACTION_MANAGE_ALL_FILES_ACCESS_PERMISSION), REQUEST_CODE);
}
} else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
// 先判斷有沒(méi)有權(quán)限
if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED &&
ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) {
writeFile();
} else {
ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE}, REQUEST_CODE);
}
} else {
writeFile();
}
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode == REQUEST_CODE) {
if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED &&
ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) {
writeFile();
} else {
ToastUtils.show("存儲(chǔ)權(quán)限獲取失敗");
}
}
}
@Override
protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == REQUEST_CODE && Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
if (Environment.isExternalStorageManager()) {
writeFile();
} else {
ToastUtils.show("存儲(chǔ)權(quán)限獲取失敗");
}
}
}
/**
* 模擬文件寫(xiě)入
*/
private void writeFile() {
ToastUtils.show("寫(xiě)入文件成功");
}
}

image
解決方案
-
看到這里,可能大家心里就會(huì)有一些想法了:
申請(qǐng)個(gè)權(quán)限還要寫(xiě)那么多代碼,感覺(jué)心好累!
為什么那么麻煩?有沒(méi)有更簡(jiǎn)單的寫(xiě)法或方式?
答案當(dāng)然是有了,并且還是一句代碼就能搞定
XXPermissions.with(MainActivity.this)
// 不適配 Android 11 可以這樣寫(xiě)
//.permission(Permission.Group.STORAGE)
// 適配 Android 11 需要這樣寫(xiě),這里無(wú)需再寫(xiě) Permission.Group.STORAGE
.permission(Permission.MANAGE_EXTERNAL_STORAGE)
.request(new OnPermission() {
@Override
public void hasPermission(List<String> granted, boolean all) {
if (all) {
ToastUtils.show("獲取存儲(chǔ)權(quán)限成功");
}
}
@Override
public void noPermission(List<String> denied, boolean never) {
if (never) {
ToastUtils.show("被永久拒絕授權(quán),請(qǐng)手動(dòng)授予存儲(chǔ)權(quán)限");
// 如果是被永久拒絕就跳轉(zhuǎn)到應(yīng)用權(quán)限系統(tǒng)設(shè)置頁(yè)面
XXPermissions.startPermissionActivity(MainActivity.this, denied);
} else {
ToastUtils.show("獲取存儲(chǔ)權(quán)限失敗");
}
}
});
- 有了這款框架,你幾乎可以零成本適配 Android 11 存儲(chǔ)新特性,萬(wàn)萬(wàn)沒(méi)想到版本適配原來(lái)可以這么簡(jiǎn)單。依賴文件適合小白,最好還是使用以上代碼去適配。
-
提示
不要把舊版存儲(chǔ)權(quán)限和新版一起配置上去申請(qǐng)權(quán)限 需要判斷一下是否為Android11 不然會(huì)報(bào)錯(cuò)的
推薦使用帖子代碼 后選依賴文件
點(diǎn)擊此處直達(dá)GitHub
文章很短,路還漫長(zhǎng),大家好,我是i小灰,一個(gè)帥氣與才華并存的男人,我們下期再見(jiàn)。