Android 11 動(dòng)態(tài)權(quán)限

前言

  • 首先 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)。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

友情鏈接更多精彩內(nèi)容