Android 版本、權限適配相關總結
首先需要在 AndroidManifest.xml 文件中聲明權限:
<uses-permission android:name='android.permission.READ_EXTERNAL_STORAGE' />
在代碼中請求用戶權限:
// 權限請求碼 private static final int PERMISSION_REQ_ID = 0; // 請求權限 private static final String[] REQUESTED_PERMISSIONS = { Manifest.permission.READ_EXTERNAL_STORAGE }; ... // 判斷有沒有存儲權限 if (checkSelfPermission(REQUESTED_PERMISSIONS[0],PERMISSION_REQ_ID)){//YSE }else {//NO } private boolean checkSelfPermission(String permissions,int requestCode){if (ContextCompat.checkSelfPermission(this,permissions) != PackageManager.PERMISSION_GRANTED){ ActivityCompat.requestPermissions(this, REQUESTED_PERMISSIONS, requestCode); return false;}return true; } // 重寫此方法,接收用戶授權回調 @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {Log.i(TAG, 'onRequestPermissionsResult: requestCode =' + requestCode+'n,permissions =' + Arrays.toString(permissions)+'n,grantResults =' + Arrays.toString(grantResults));if (requestCode == PERMISSION_REQ_ID){ if (grantResults[0] == PackageManager.PERMISSION_GRANTED){//用戶同意權限 }else {//用戶拒絕權限 }} 版本適配
從 Android 6.0 到 Android 10 存儲/訪問文件功能,有發生了很多變化。
Android 7.0 前在Android 7.0 以前我們訪問內存中的文件可以通過 Uri.fromFile,將 File 轉換成 Uri 對象,這個 Uri 對象表示這本地真實路徑。 訪問一個圖片:
String fileName = 'default_Image.jpg';File file = new File('file_path', fileName);Uri uri = Uri.fromFile(file);Android 7.0 后
在 7.0 后,這種通過真實路徑來獲取的 Uri 被認為是不安全的,所以提供了一種新的解決方案,就是通過 FileProvide 來實現文件的訪問,FileProvider 是一種比較特殊的內容提供器,他使用了類似于內容提供器的機制來對數據進行保護。 訪問一個圖片:
File file = new File(CACHE_IMG, 'file_name'); Uri imageUri = FileProvider.getUriForFile(activity,'com.sandan.fileprovider', file); //這里進行替換uri的獲得方式
然而上面這種真的好嗎,對用開發者而且這算是好處吧,但是對用用戶而言,上述的無疑一些流氓作用,因為開發者完全可以訪問的內存中的所有位置,并作出一些改變,導致 SD 卡中的空間變得非常亂,即使卸載了 app,但是一些垃圾文件卻還在內存中。
Android 10.0在 Android 10.0 ,為了解決上述問題,Google 在 Android 10.0 中加入了 作用域功能。
什么是作用域就是 Android 系統對 SD 卡做了很大的限制,從 Android 10.0 開始,每個程序只能有權在自己的外置存儲空間關聯的目錄下讀取和創建相應的文件,也稱作沙箱。獲取該目錄的代碼是:getExternalFilesDir() ,關聯的目錄路徑大致如下:
Html CSS JavaScript Vb vbs Asp PHP Perl Python Ruby C# C++ SQL Delphi Diff Groovy Java JavaFX ActionScript3 Bash/shell powershell Plain Text Scala XML顯示語言名稱 顯示行號 允許折疊
將數據放在這個目錄下,你可以使用之前的方法對文件進行讀寫,不需要作出任何變更和適配。但是這個文件夾中的文件會隨著應用卸載而被隨之刪除。 那如果需要訪問其他目錄怎么辦呢?比如獲取相冊中的圖片,向相冊中添加一張圖片。為此,Android 系統針對系統文件類型進行了分類:圖片,音頻,視頻 這三類文件可以通過 MediaStore API 來進行訪問,這種稱為共享空間,其他的系統文件需要使用 系統的文件選擇器來進行訪問,另外,如果程序向媒體庫寫入圖片,視頻,音頻,將會自動用于讀寫權限,不需要額外申請權限,如果你要讀取其他程序向媒體貢獻的圖片,視頻,音頻,則必須要申請 READ_EXTERNAL_STORAGE 權限,WRITE_EXTERNAL_STORAGE 權限會在未來的版本中被廢棄。
舉個栗子舉例說明:有一張本地圖片,向這張圖片添加水印,并保存到相冊。
直接上代碼:
/** * 保存圖片到相冊 * * @param context 上下文 * @param text 水印文字 */ private void savePhotoAlbum(final Context context, final String text) {//這里開啟子線程,防止堵塞。new Thread(new Runnable() { @Override public void run() {try { //從本地獲取一張圖片,轉成Bitmap Bitmap bitmap = BitmapFactory.decodeResource(context.getResources(), R.drawable.icon_info); //在沙箱中創建文件,名稱:info.jpg File file = new File(context.getExternalFilesDir(Environment.DIRECTORY_PICTURES), 'info.jpg'); //判斷文件是否存在,不存在創建文件。 if (!file.exists()) {file.createNewFile(); } // 向圖片添加水印 Bitmap newBitmap = addInfoWatermark(context, bitmap, text); // 更新相冊 updatePhotoAlbum(context, newBitmap, file);} catch (Exception e) { e.printStackTrace();} }}).start(); //開始線程 } /** * 保存到相冊 * * @param context 上下文 * @param src 源圖片 * @param text 水印文字 */ private Bitmap addInfoWatermark(final Context context, Bitmap src, String text) {//判斷圖片/水印文字 是否為空if (isEmptyBitmap(src) || text == null ) { return null;}// 從源圖片復制一份Bitmap ret = src.copy(src.getConfig(), true);// 初始化畫筆Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG); // 初始化畫布Canvas canvas = new Canvas(ret);// 水印文字:黑色paint.setColor(Color.BLACK);// 文字大小:19dppaint.setTextSize(dip2px(context, 19)); // 開始繪畫canvas.drawText(text, 10, 10 , paint);// 循環利用資源if (!src.isRecycled()) { src.recycle(); }return ret; } /** * 保存到相冊 * * @param context 上下文 * @param src 源圖片 * @param file 要保存到的文件 */ private void savePhotoAlbum(final Context context, Bitmap src, final File file) {//判斷圖片 是否為空if (isEmptyBitmap(src)) { return;}// 保存文件OutputStream outputStream;try { //輸出這個文件 outputStream = new BufferedOutputStream(new FileOutputStream(file)); // 壓縮 src.compress(Bitmap.CompressFormat.JPEG, 100, outputStream); // 循環利用資源 if (!src.isRecycled()) {src.recycle(); }} catch (FileNotFoundException e) { e.printStackTrace();}// 更新圖庫,這個在 Android 6.0 和 Android 10.0 更新圖庫,存在差異。if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) { // Android 10.0 及以上 // 創建 ContentValues 對象,準備插入數據 ContentValues values = new ContentValues(); values.put(MediaStore.MediaColumns.DISPLAY_NAME, file.getName()); values.put(MediaStore.MediaColumns.MIME_TYPE, getMimeType(file)); values.put(MediaStore.MediaColumns.RELATIVE_PATH, Environment.DIRECTORY_DCIM); ContentResolver contentResolver = context.getContentResolver(); // 插入數據,返回所插入數據對應的Uri Uri uri = contentResolver.insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values); if (uri == null) {return; } try {// 獲取剛插入的數據的Uri對應的輸出流outputStream = contentResolver.openOutputStream(uri);FileInputStream fileInputStream = new FileInputStream(file);// 從一個流復制到另一個流上FileUtils.copy(fileInputStream, outputStream);//關閉流fileInputStream.close();outputStream.close(); } catch (IOException e) {e.printStackTrace(); }} else { // android 6.0 - 10.0 // 掃描文件 MediaScannerConnection.scanFile( context.getApplicationContext(), new String[]{file.getAbsolutePath()}, new String[]{'image/jpeg'}, new MediaScannerConnection.OnScanCompletedListener() {@Overridepublic void onScanCompleted(String path, Uri uri) { //通知相冊更新 // 插入圖片 MediaStore.Images.Media.insertImage( context.getContentResolver(), BitmapFactory.decodeFile(path), file.getName(), null); Intent intent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE); Uri u = Uri.fromFile(file); intent.setData(u); context.sendBroadcast(intent); // 發廣播通知,更新相冊} });} } /** * Bitmap對象是否為空。 */ private static boolean isEmptyBitmap(Bitmap src) {return src == null || src.getWidth() == 0 || src.getHeight() == 0; } /** * 獲取 Mime 類型 * * @param file 文件 * @return Mime 類型 */ private static String getMimeType(File file) {FileNameMap fileNameMap = URLConnection.getFileNameMap();String type = fileNameMap.getContentTypeFor(file.getName());return type; } /** * 根據手機的分辨率從 px(像素) 的單位 轉成為 dp */ public int dip2px(Context context, float dpValue) {final float scale = context.getResources().getDisplayMetrics().density;return (int) (dpValue * scale + 0.5f); }
以上就是Android 版本、權限適配相關總結的詳細內容,更多關于Android 版本、權限適配的資料請關注好吧啦網其它相關文章!
相關文章:
