Android 14 、15動態申請讀寫權限實現 (Java)

在 Android 14、15 中,Google 進一步優化了存儲權限系統,特別是寫權限的管理。以下是完整的 Java 實現方案:

1. AndroidManifest.xml 聲明權限

<!-- Android 14 存儲權限 -->
<uses-permission android:name="android.permission.READ_MEDIA_IMAGES" />
<uses-permission android:name="android.permission.READ_MEDIA_VIDEO" />
<uses-permission android:name="android.permission.READ_MEDIA_AUDIO" />
<!-- 新增的視覺媒體選擇權限 -->
<uses-permission android:name="android.permission.READ_MEDIA_VISUAL_USER_SELECTED" />
<!-- 兼容舊版本 (可選) -->
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" android:maxSdkVersion="32" />
<!-- 寫入權限 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" android:maxSdkVersion="32" /> <!-- 僅用于兼容舊版本 -->

2. Java 權限請求實現

import android.Manifest;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.os.Build;
import android.provider.Settings;
import android.widget.Toast;

import androidx.activity.result.ActivityResultLauncher;
import androidx.activity.result.contract.ActivityResultContracts;
import androidx.appcompat.app.AlertDialog;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.content.ContextCompat;

import java.util.ArrayList;
import java.util.List;
import java.util.Map;

public class StoragePermissionHelper extends AppCompatActivity {

? ? // 多權限請求啟動器
? ? private final ActivityResultLauncher<String[]> requestPermissionsLauncher =
? ? ? ? ? ? registerForActivityResult(new ActivityResultContracts.RequestMultiplePermissions(),
? ? ? ? ? ? ? ? ? ? this::handlePermissionResult);

? ? // 檢查并請求存儲權限
? ? public void checkAndRequestStoragePermissions() {
? ? ? ? List<String> permissionsToRequest = new ArrayList<>();
? ? ? ??
? ? ? ? // Android 14+ 需要的權限(讀、寫權限)
? ? ? ? if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.UPSIDE_DOWN_CAKE) {
? ? ? ? ? ? if (ContextCompat.checkSelfPermission(this, Manifest.permission.READ_MEDIA_IMAGES)?
? ? ? ? ? ? ? ? ? ? != PackageManager.PERMISSION_GRANTED) {
? ? ? ? ? ? ? ? permissionsToRequest.add(Manifest.permission.READ_MEDIA_IMAGES);
? ? ? ? ? ? }
? ? ? ? ? ? if (ContextCompat.checkSelfPermission(this, Manifest.permission.READ_MEDIA_VIDEO)?
? ? ? ? ? ? ? ? ? ? != PackageManager.PERMISSION_GRANTED) {
? ? ? ? ? ? ? ? permissionsToRequest.add(Manifest.permission.READ_MEDIA_VIDEO);
? ? ? ? ? ? }
? ? ? ? ? ? if (ContextCompat.checkSelfPermission(this, Manifest.permission.READ_MEDIA_VISUAL_USER_SELECTED)?
? ? ? ? ? ? ? ? ? ? != PackageManager.PERMISSION_GRANTED) {
? ? ? ? ? ? ? ? permissionsToRequest.add(Manifest.permission.READ_MEDIA_VISUAL_USER_SELECTED);
? ? ? ? ? ? }
? ? ? ? }?
? ? ? ? // Android 13(讀、寫權限)
? ? ? ? else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
? ? ? ? ? ? if (ContextCompat.checkSelfPermission(this, Manifest.permission.READ_MEDIA_IMAGES)?
? ? ? ? ? ? ? ? ? ? != PackageManager.PERMISSION_GRANTED) {
? ? ? ? ? ? ? ? permissionsToRequest.add(Manifest.permission.READ_MEDIA_IMAGES);
? ? ? ? ? ? }
? ? ? ? ? ? if (ContextCompat.checkSelfPermission(this, Manifest.permission.READ_MEDIA_VIDEO)?
? ? ? ? ? ? ? ? ? ? != PackageManager.PERMISSION_GRANTED) {
? ? ? ? ? ? ? ? permissionsToRequest.add(Manifest.permission.READ_MEDIA_VIDEO);
? ? ? ? ? ? }
? ? ? ? }
? ? ? ? // Android 10-12(讀、寫權限)
? ? ? ? else if (ContextCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE)?
? ? ? ? ? ? ? ? != PackageManager.PERMISSION_GRANTED) {
? ? ? ? ? ? permissionsToRequest.add(Manifest.permission.READ_EXTERNAL_STORAGE);

if (ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {permissionsToRequest.add(Manifest.permission.WRITE_EXTERNAL_STORAGE);}
}

? ? ? ? if (permissionsToRequest.isEmpty()) {
? ? ? ? ? ? onStoragePermissionGranted();
? ? ? ? } else {
? ? ? ? ? ? requestPermissionsLauncher.launch(permissionsToRequest.toArray(new String[0]));
? ? ? ? }
? ? }

? ? // 處理權限請求結果
? ? private void handlePermissionResult(Map<String, Boolean> permissions) {
? ? ? ? List<String> deniedPermissions = new ArrayList<>();
? ? ? ??
? ? ? ? for (Map.Entry<String, Boolean> entry : permissions.entrySet()) {
? ? ? ? ? ? if (!entry.getValue()) {
? ? ? ? ? ? ? ? deniedPermissions.add(entry.getKey());
? ? ? ? ? ? }
? ? ? ? }

? ? ? ? if (deniedPermissions.isEmpty()) {
? ? ? ? ? ? onStoragePermissionGranted();
? ? ? ? } else {
? ? ? ? ? ? handleDeniedPermissions(deniedPermissions);
? ? ? ? }
? ? }

? ? // 權限全部授予
? ? private void onStoragePermissionGranted() {
? ? ? ? Toast.makeText(this, "存儲權限已授予", Toast.LENGTH_SHORT).show();
? ? ? ? // 這里可以執行需要權限的操作
? ? }

? ? // 處理被拒絕的權限
? ? private void handleDeniedPermissions(List<String> deniedPermissions) {
? ? ? ? for (String permission : deniedPermissions) {
? ? ? ? ? ? if (shouldShowRequestPermissionRationale(permission)) {
? ? ? ? ? ? ? ? showRationaleDialog(permission);
? ? ? ? ? ? } else {
? ? ? ? ? ? ? ? showGoToSettingsDialog(permission);
? ? ? ? ? ? }
? ? ? ? }
? ? }

? ? // 顯示權限解釋對話框
? ? private void showRationaleDialog(String permission) {
? ? ? ? new AlertDialog.Builder(this)
? ? ? ? ? ? ? ? .setTitle("需要權限")
? ? ? ? ? ? ? ? .setMessage(getPermissionMessage(permission))
? ? ? ? ? ? ? ? .setPositiveButton("確定", (dialog, which) ->?
? ? ? ? ? ? ? ? ? ? checkAndRequestStoragePermissions())
? ? ? ? ? ? ? ? .setNegativeButton("取消", null)
? ? ? ? ? ? ? ? .show();
? ? }

? ? // 顯示前往設置對話框
? ? private void showGoToSettingsDialog(String permission) {
? ? ? ? new AlertDialog.Builder(this)
? ? ? ? ? ? ? ? .setTitle("權限被永久拒絕")
? ? ? ? ? ? ? ? .setMessage("請在應用設置中手動授予" + getPermissionName(permission) + "權限")
? ? ? ? ? ? ? ? .setPositiveButton("去設置", (dialog, which) -> {
? ? ? ? ? ? ? ? ? ? Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
? ? ? ? ? ? ? ? ? ? intent.setData(Uri.fromParts("package", getPackageName(), null));
? ? ? ? ? ? ? ? ? ? startActivity(intent);
? ? ? ? ? ? ? ? })
? ? ? ? ? ? ? ? .setNegativeButton("取消", null)
? ? ? ? ? ? ? ? .show();
? ? }

? ? // 獲取權限說明信息
? ? private String getPermissionMessage(String permission) {
? ? ? ? switch (permission) {
? ? ? ? ? ? case Manifest.permission.READ_MEDIA_IMAGES:
? ? ? ? ? ? ? ? return "需要訪問您的照片以提供完整功能";
? ? ? ? ? ? case Manifest.permission.READ_MEDIA_VIDEO:
? ? ? ? ? ? ? ? return "需要訪問您的視頻以提供完整功能";
? ? ? ? ? ? case Manifest.permission.READ_MEDIA_VISUAL_USER_SELECTED:
? ? ? ? ? ? ? ? return "需要訪問您選擇的媒體文件";
? ? ? ? ? ? case Manifest.permission.READ_EXTERNAL_STORAGE:
? ? ? ? ? ? ? ? return "需要訪問您的文件以提供完整功能";
? ? ? ? ? ? default:
? ? ? ? ? ? ? ? return "需要此權限以提供完整功能";
? ? ? ? }
? ? }

? ? // 獲取權限名稱
? ? private String getPermissionName(String permission) {
? ? ? ? switch (permission) {
? ? ? ? ? ? case Manifest.permission.READ_MEDIA_IMAGES:
? ? ? ? ? ? ? ? return "照片訪問";
? ? ? ? ? ? case Manifest.permission.READ_MEDIA_VIDEO:
? ? ? ? ? ? ? ? return "視頻訪問";
? ? ? ? ? ? case Manifest.permission.READ_MEDIA_VISUAL_USER_SELECTED:
? ? ? ? ? ? ? ? return "選擇的媒體文件訪問";
? ? ? ? ? ? case Manifest.permission.READ_EXTERNAL_STORAGE:
? ? ? ? ? ? ? ? return "文件訪問";
? ? ? ? ? ? default:
? ? ? ? ? ? ? ? return "存儲";
? ? ? ? }
? ? }
}

3. 使用示例

public class MainActivity extends StoragePermissionHelper {

? ? @Override
? ? protected void onCreate(Bundle savedInstanceState) {
? ? ? ? super.onCreate(savedInstanceState);
? ? ? ? setContentView(R.layout.activity_main);
? ? ? ??
? ? ? ? findViewById(R.id.btn_request_storage).setOnClickListener(v -> {
? ? ? ? ? ? checkAndRequestStoragePermissions();
? ? ? ? });
? ? }
? ??
? ? @Override
? ? protected void onResume() {
? ? ? ? super.onResume();
? ? ? ? // 從設置返回后檢查權限狀態
? ? ? ? verifyStoragePermissions();
? ? }
? ??
? ? private void verifyStoragePermissions() {
? ? ? ? boolean hasPermissions;
? ? ? ??
? ? ? ? if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.UPSIDE_DOWN_CAKE) {
? ? ? ? ? ? hasPermissions = ContextCompat.checkSelfPermission(this,?
? ? ? ? ? ? ? ? ? ? Manifest.permission.READ_MEDIA_IMAGES) == PackageManager.PERMISSION_GRANTED
? ? ? ? ? ? ? ? ? ? && ContextCompat.checkSelfPermission(this,?
? ? ? ? ? ? ? ? ? ? Manifest.permission.READ_MEDIA_VIDEO) == PackageManager.PERMISSION_GRANTED;
? ? ? ? } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
? ? ? ? ? ? hasPermissions = ContextCompat.checkSelfPermission(this,?
? ? ? ? ? ? ? ? ? ? Manifest.permission.READ_MEDIA_IMAGES) == PackageManager.PERMISSION_GRANTED;
? ? ? ? } else {
? ? ? ? ? ? hasPermissions = ContextCompat.checkSelfPermission(this,?
? ? ? ? ? ? ? ? ? ? Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED;
? ? ? ? }
? ? ? ??
? ? ? ? if (hasPermissions) {
? ? ? ? ? ? onStoragePermissionGranted();
? ? ? ? }
? ? }
}

Android 14 存儲權限關鍵點

  1. 新增權限:

    • READ_MEDIA_VISUAL_USER_SELECTED: 允許用戶選擇特定媒體文件

    • 其他媒體權限與 Android 13 相同

  2. 權限策略變化:

    • 完全移除了?READ_EXTERNAL_STORAGE?對媒體文件的控制

    • 應用默認只能訪問自己創建的文件

    • 訪問其他媒體文件必須請求權限

  3. 最佳實踐:

    • 優先使用系統照片選擇器 (Photo Picker)

    • 只在真正需要時才請求權限

    • 提供清晰的權限解釋

    • 正確處理所有可能的拒絕情況

  4. 兼容性考慮:

    • 使用?Build.VERSION.SDK_INT?檢查系統版本

    • 為不同版本提供不同的權限請求策略

    • 測試在各種場景下的權限行為

這套實現方案完全符合 Android 14 的存儲權限要求,同時保持了良好的向后兼容性。

Android 14 寫權限關鍵點

  1. 主要變化:

    • Android 14 繼續限制?WRITE_EXTERNAL_STORAGE?的使用

    • 需要?MANAGE_EXTERNAL_STORAGE?權限才能管理所有文件

    • 必須引導用戶到系統設置手動開啟"管理所有文件"權限

  2. 權限策略:

    • 應用默認只能寫入自己的專屬目錄

    • 寫入共享存儲需要相應權限

    • 管理所有文件需要用戶顯式授權

  3. 最佳實踐:

    • 優先使用 MediaStore API 來寫入共享媒體文件

    • 使用 SAF (Storage Access Framework) 讓用戶選擇保存位置

    • 只在絕對必要時請求管理所有文件權限

    • 提供清晰的權限解釋和引導

  4. 兼容性處理:

    • 為不同 Android 版本提供不同的權限請求策略

    • 使用?Environment.isExternalStorageManager()?檢查管理權限

    • 測試在各種情況下的權限行為

這套實現方案完全符合 Android 14 的寫權限要求,同時保持了良好的向后兼容性。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/901325.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/901325.shtml
英文地址,請注明出處:http://en.pswp.cn/news/901325.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

小剛說C語言刷題——第23講 字符數組

前面&#xff0c;我們學習了一維數組和二維數組的概念。今天我們學習一種特殊的數組&#xff0c;字符數組。 1.字符數組的概念 字符數組就是指元素類型為字符的數組。字符數組是用來存放字符序列或者字符串的。 2.字符數組的定義及語法 char ch[5]; 3.字符數組的初始化及賦…

用AI生成系統架構圖

DeepSeek+Drawio+SVG繪制架構圖-找到一種真正可行實用的方法和思路 1、使用DeepSeek生成SVG文件,導入drawio工具的方法 ?? 問題根源分析 錯誤現象: ? 導入時報錯包含 data:image/SVG;base64 和 %20 等 URL 編碼字符 ? 代碼被錯誤轉換為 Base64 格式(適用于網頁嵌入,但…

免費干凈!付費軟件的平替款!

今天給大家分享一款超棒的電腦錄屏軟件&#xff0c;簡直不要太好用&#xff01;它的界面特別干凈&#xff0c;沒有一點兒廣告&#xff0c;看起來特別清爽。 電腦錄屏 無廣告的錄屏軟件 這個軟件超方便&#xff0c;根本不用安裝&#xff0c;打開就能直接用。 它功能也很強大&am…

【XCP實戰】AUTOSAR架構下XCP從0到1開發配置實踐

目錄 前言 正文 1.CAN功能開發 1.1 DBC的制作及導入 1.2 CanTrcv模塊配置 1.3 Can Controller模塊配置 1.4 CanIf模塊配置 2.XCP模塊集成配置配置 2.1.XCP模塊配置 2.2.XCP模塊的Task Mapping 2.3.XCP模塊的初始化 3.在鏈接文件中定義標定段 4.編寫標定相關的測試…

Vitis: 使用自定義IP時 Makefile錯誤 導致編譯報錯

參考文章: 【小梅哥FPGA】 Vitis開發中自定義IP的Makefile路徑問題解決方案 Vitis IDE自定義IP Makefile錯誤&#xff08;arm-xilinx-eabi-gcc.exe: error: *.c: Invalid argument&#xff09;解決方法 Vitis 使用自定義IP時: Makefile 文件里的語句是需要修改的&#xff0c;…

Python中NumPy的統計運算

在數據分析和科學計算領域&#xff0c;Python憑借其豐富的庫生態系統成為首選工具之一&#xff0c;而NumPy作為Python數值計算的核心庫&#xff0c;憑借其高效的數組操作和強大的統計運算功能&#xff0c;廣泛應用于機器學習、信號處理、統計分析等場景。本文將系統介紹NumPy在…

C語言程序環境和預處理詳解

本章重點&#xff1a; 程序的翻譯環境 程序的執行環境 詳解&#xff1a;C語言程序的編譯鏈接 預定義符號介紹 預處理指令 #define 宏和函數的對比 預處理操作符#和##的介紹 命令定義 預處理指令 #include 預處理指令 #undef 條件編譯 程序的翻譯環境和執行環…

智能工廠調度系統設計方案研究報告

一、系統架構設計 1.1 物理部署架構 設備層&#xff1a;部署大量搭載多傳感器陣列的 AGV 智能循跡車&#xff0c;這些傳感器包括激光雷達、視覺相機、超聲波傳感器等&#xff0c;用于感知周圍環境信息&#xff0c;實現自主導航與避障功能&#xff1b;在每個工序節點處設置 RF…

全新突破 | 更全面 · 更安全 · 更靈活

xFile 高可用存儲網關 2.0 重磅推出&#xff0c;新增多空間隔離功能從根源上防止數據沖突&#xff0c;保障各業務數據的安全性與獨立性。同時支持 NFS、CIFS、FTP 等多種主流文件協議&#xff0c;無需繁瑣的數據拷貝轉換&#xff0c;即可與現有系統無縫對接&#xff0c;降低集成…

C# js 判斷table中tr否存在相同的值

html 中如&#xff1a; 實現&#xff1a;table數據表格中&#xff0c;點擊刪除按鈕時&#xff0c;驗證相同子訂單號條數是否大于1&#xff0c;大于允許刪除。保證數據表格中只有唯一的一條子訂單號數據。 <table style"width: 100%; background-color: #fff;" ce…

操作系統基礎:07 我們的任務

課程回顧與后續規劃 上節課我們探討了操作系統的歷史。了解歷史能讓我們明智&#xff0c;從操作系統的發展歷程中&#xff0c;我們總結出兩個核心的里程碑式圖像&#xff1a;多進程&#xff08;多任務切換&#xff09;圖像和文件操作圖像 。Unix和Windows等系統的成功&#xf…

16.【.NET 8 實戰--孢子記賬--從單體到微服務--轉向微服務】--單體轉微服務--微服務的部署與運維

部署與運維是微服務架構成功實施的關鍵環節。一個良好的部署與運維體系能夠保障微服務的高可用性、可擴展性和可靠性。在這一階段&#xff0c;重點包括微服務的容器化與編排、API 網關的實現以及日志與監控體系的建設。 一、容器化與編排 1.1 使用 Docker 容器化微服務 容器…

MCP基礎學習計劃詳細總結

MCP基礎學習計劃詳細總結 1.MCP概述與基礎 ? MCP&#xff08;Model Context Protocol&#xff09;&#xff1a;由Anthropic公司于2024年11月推出&#xff0c;旨在實現大型語言模型&#xff08;LLM&#xff09;與外部數據源和工具的無縫集成。 ? 核心功能&#xff1a; ? 資…

NoSQL入門指南:Redis與MongoDB的Java實戰

一、為什么需要NoSQL&#xff1f; 在傳統SQL數據庫中&#xff0c;數據必須嚴格遵循預定義的表結構&#xff0c;就像把所有物品整齊擺放在固定尺寸的貨架上。而NoSQL&#xff08;Not Only SQL&#xff09;數據庫則像一個靈活的儲物間&#xff0c;允許存儲各種類型的數據&#x…

Java 列表初始化全解析:7種方式詳解與最佳實踐

文章目錄 **引言****1. 傳統逐個添加元素****特點****注意事項** **2. Arrays.asList() 構造函數****特點****注意事項** **3. 雙括號初始化&#xff08;匿名內部類&#xff09;****特點****注意事項** **4. Java 9 List.of()&#xff08;不可變列表&#xff09;****特點****注…

最大公約數和最小倍數 java

在Java中&#xff0c;計算兩個數的最大公約數&#xff08;Greatest Common Divisor, GCD&#xff09;和最小公倍數&#xff08;Least Common Multiple, LCM&#xff09;是常見的編程問題。以下是具體的實現方法和代碼示例。 --- ### **1. 最大公約數 (GCD)** 最大公約數是指…

數據庫——視圖

一、視圖的定義與核心特性 1.基本概念 (1)視圖(View)是基于一個或多個底層表(或視圖)的虛擬表,其本身不存儲數據,僅保存查詢語句的定義。當用戶查詢視圖時,數據庫會動態執行其封裝的SQL語句,生成結果集。 (2)本質:視圖是底層表的邏輯映射,結構與表相同(由行和列…

【Proteus仿真】【32單片機-A008】MPX4115壓力檢測系統設計

目錄 一、主要功能 二、使用步驟 三、硬件資源 四、軟件設計 五、實驗現象 聯系作者 一、主要功能 1、壓力檢測與LCD顯示 2、超過上限&#xff0c;降壓模塊啟動 3、壓力檢測范圍15kpa-115kpa 4、壓力閾值設置 5、超限報警 二、使用步驟 系統運行后&#xff0c;LCD160…

java和c#的相似及區別基礎對比

用過十幾種語言&#xff0c;但是java和c#是最為重要的兩門。c#發明人曾主導開發了pascal和delphi&#xff0c;加入微軟后&#xff0c;參考了c和java完成了c#和net。大家用過java或c#任意一種的&#xff0c;可以通過本篇文章快速掌握另外一門語言。 基礎語法 變量聲明&#xf…

OpenBayes 一周速覽|1分鐘生成完整音樂,DiffRhythm人聲伴奏一鍵搞定; Stable Virtual Camera重塑3D視頻創作

公共資源速遞 5 個數據集&#xff1a; * 302 例罕見病病例數據集 * DRfold2 RNA 結構測試數據集 * NaturalReasoning 自然推理數據集 * VenusMutHub 蛋白質突變小樣本數據集 * Bird Vs Drone 鳥類與無人機圖像分類數據集 2 個模型&#xff1a; * Qwen2.5-0mni * Llama…