Android 版本、權(quán)限適配相關(guān)總結(jié)
首先需要在 AndroidManifest.xml 文件中聲明權(quán)限:
<uses-permission android:name='android.permission.READ_EXTERNAL_STORAGE' />
在代碼中請求用戶權(quán)限:
// 權(quán)限請求碼 private static final int PERMISSION_REQ_ID = 0; // 請求權(quán)限 private static final String[] REQUESTED_PERMISSIONS = { Manifest.permission.READ_EXTERNAL_STORAGE }; ... // 判斷有沒有存儲權(quán)限 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; } // 重寫此方法,接收用戶授權(quán)回調(diào) @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){//用戶同意權(quán)限 }else {//用戶拒絕權(quán)限 }} 版本適配
從 Android 6.0 到 Android 10 存儲/訪問文件功能,有發(fā)生了很多變化。
Android 7.0 前在Android 7.0 以前我們訪問內(nèi)存中的文件可以通過 Uri.fromFile,將 File 轉(zhuǎn)換成 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 來實現(xiàn)文件的訪問,F(xiàn)ileProvider 是一種比較特殊的內(nèi)容提供器,他使用了類似于內(nèi)容提供器的機制來對數(shù)據(jù)進行保護。 訪問一個圖片:
File file = new File(CACHE_IMG, 'file_name'); Uri imageUri = FileProvider.getUriForFile(activity,'com.sandan.fileprovider', file); //這里進行替換uri的獲得方式
然而上面這種真的好嗎,對用開發(fā)者而且這算是好處吧,但是對用用戶而言,上述的無疑一些流氓作用,因為開發(fā)者完全可以訪問的內(nèi)存中的所有位置,并作出一些改變,導(dǎo)致 SD 卡中的空間變得非常亂,即使卸載了 app,但是一些垃圾文件卻還在內(nèi)存中。
Android 10.0在 Android 10.0 ,為了解決上述問題,Google 在 Android 10.0 中加入了 作用域功能。
什么是作用域就是 Android 系統(tǒng)對 SD 卡做了很大的限制,從 Android 10.0 開始,每個程序只能有權(quán)在自己的外置存儲空間關(guān)聯(lián)的目錄下讀取和創(chuàng)建相應(yīng)的文件,也稱作沙箱。獲取該目錄的代碼是:getExternalFilesDir() ,關(guān)聯(lián)的目錄路徑大致如下:
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顯示語言名稱 顯示行號 允許折疊
將數(shù)據(jù)放在這個目錄下,你可以使用之前的方法對文件進行讀寫,不需要作出任何變更和適配。但是這個文件夾中的文件會隨著應(yīng)用卸載而被隨之刪除。 那如果需要訪問其他目錄怎么辦呢?比如獲取相冊中的圖片,向相冊中添加一張圖片。為此,Android 系統(tǒng)針對系統(tǒng)文件類型進行了分類:圖片,音頻,視頻 這三類文件可以通過 MediaStore API 來進行訪問,這種稱為共享空間,其他的系統(tǒng)文件需要使用 系統(tǒng)的文件選擇器來進行訪問,另外,如果程序向媒體庫寫入圖片,視頻,音頻,將會自動用于讀寫權(quán)限,不需要額外申請權(quán)限,如果你要讀取其他程序向媒體貢獻的圖片,視頻,音頻,則必須要申請 READ_EXTERNAL_STORAGE 權(quán)限,WRITE_EXTERNAL_STORAGE 權(quán)限會在未來的版本中被廢棄。
舉個栗子舉例說明:有一張本地圖片,向這張圖片添加水印,并保存到相冊。
直接上代碼:
/** * 保存圖片到相冊 * * @param context 上下文 * @param text 水印文字 */ private void savePhotoAlbum(final Context context, final String text) {//這里開啟子線程,防止堵塞。new Thread(new Runnable() { @Override public void run() {try { //從本地獲取一張圖片,轉(zhuǎn)成Bitmap Bitmap bitmap = BitmapFactory.decodeResource(context.getResources(), R.drawable.icon_info); //在沙箱中創(chuàng)建文件,名稱:info.jpg File file = new File(context.getExternalFilesDir(Environment.DIRECTORY_PICTURES), 'info.jpg'); //判斷文件是否存在,不存在創(chuàng)建文件。 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;}// 從源圖片復(fù)制一份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);// 循環(huán)利用資源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); // 循環(huán)利用資源 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 及以上 // 創(chuàng)建 ContentValues 對象,準備插入數(shù)據(jù) 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(); // 插入數(shù)據(jù),返回所插入數(shù)據(jù)對應(yīng)的Uri Uri uri = contentResolver.insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values); if (uri == null) {return; } try {// 獲取剛插入的數(shù)據(jù)的Uri對應(yīng)的輸出流outputStream = contentResolver.openOutputStream(uri);FileInputStream fileInputStream = new FileInputStream(file);// 從一個流復(fù)制到另一個流上FileUtils.copy(fileInputStream, outputStream);//關(guān)閉流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); // 發(fā)廣播通知,更新相冊} });} } /** * 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; } /** * 根據(jù)手機的分辨率從 px(像素) 的單位 轉(zhuǎn)成為 dp */ public int dip2px(Context context, float dpValue) {final float scale = context.getResources().getDisplayMetrics().density;return (int) (dpValue * scale + 0.5f); }
以上就是Android 版本、權(quán)限適配相關(guān)總結(jié)的詳細內(nèi)容,更多關(guān)于Android 版本、權(quán)限適配的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. Properties 持久的屬性集的實例詳解2. 詳解python 內(nèi)存優(yōu)化3. Python使用xlrd實現(xiàn)讀取合并單元格4. SpringBoot整合Redis的步驟5. IBM發(fā)布AUS: 消除你Java程序中脆弱的API6. ASP.NET MVC實現(xiàn)區(qū)域或城市選擇7. python 網(wǎng)頁解析器掌握第三方 lxml 擴展庫與 xpath 的使用方法8. ASP.NET MVC使用jQuery ui的progressbar實現(xiàn)進度條9. 基于python調(diào)用jenkins-cli實現(xiàn)快速發(fā)布10. JS中6個對象數(shù)組去重的方法

網(wǎng)公網(wǎng)安備