Android Q文件存儲機(jī)制修改成了沙盒模式
應(yīng)用只能訪問自己沙盒下的文件和公共媒體文件
存儲(也就是write)私有目錄和公共媒體文件都不需要WRITE_EXTERNAL_STORAGE權(quán)限
讀取 (也就是read)私有目錄不需要READ_EXTERNAL_STORAGE權(quán)限,讀取公共媒體文件需要READ_EXTERNAL_STORAGE權(quán)限。
也就是說只能訪問
1.自己沙盒
getExternalFilesDir(這是APP自身目錄下的文件夾 (Android/data/包名/fils))
(1)APP 卸載在這里插入代碼片后愁拭,數(shù)據(jù)會清除而柑。
(2)APP 訪問自己的 App-specific 目錄時無需任何權(quán)限吓蘑。
(3)可以使用FileProvider分享使用自己私有目錄的文件锡垄。
所以在沙盒化的Q系統(tǒng)下要糊,私有目錄下的文件會跟隨APP卸載而刪除盼砍。在其目錄內(nèi)部的文件操作和Q之前的版本一樣,可以隨意處理枕屉。
2.公共媒體文件
公有目錄:Downloads常柄、Documents、Pictures 搀擂、DCIM、Movies卷玉、Music哨颂、Ringtones等
地址:/storage/emulated/0/Downloads(Pictures)等
(1)公共目錄下的文件在 APP 卸載后,不會刪除相种。
(2)APP 可以通過 SAF框架(System Access Framework)威恼、MediaStore 接口訪問其中的文件。
(3)無法直接使用路徑訪問公共目錄文件寝并。
由于公共目錄沒有辦法直接訪問和處理文件箫措,所以我們需要按照Android Q的新規(guī)則來進(jìn)行文件的處理,要使用到ContentResolver 和MediaStore數(shù)據(jù)庫和Cursor 來進(jìn)行查詢等
也就是說如果你的項(xiàng)目中照片存儲的路徑就是APP的私有目錄那么就沒必要去適配AndroidQ
在AndroidQ中適配存儲圖片
1.保存到APP的私有目錄PICTURES下面
private void saveAppPrivateFils(Bitmap bitmap) {
File file = new File(getExternalFilesDir(Environment.DIRECTORY_PICTURES), "劉亦菲2.jpg");
BufferedOutputStream bos = null;
try {
bos = new BufferedOutputStream(new FileOutputStream(file));
bitmap.compress(Bitmap.CompressFormat.JPEG, 80, bos);
bos.flush();
bos.close();
Toast.makeText(MainActivity.this, "保存成功", Toast.LENGTH_SHORT).show();
} catch (IOException e) {
e.printStackTrace();
Toast.makeText(MainActivity.this, "保存失敗" + e.toString(), Toast.LENGTH_SHORT).show();
}
}
2.保存到共享媒體文件夾中DCIM(相冊)
private void saveAppDCIMFils(Bitmap bitmap) {
Uri uri =null;
ContentResolver contentResolver = getContentResolver();
ContentValues contentValues =new ContentValues();
contentValues.put(MediaStore.Images.Media.DISPLAY_NAME, "劉亦菲.jpg");
contentValues.put(MediaStore.Images.Media.DESCRIPTION, "劉亦菲.jpg");
//兼容Android Q和以下版本
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
//android Q中不再使用DATA字段衬潦,而用RELATIVE_PATH代替
//RELATIVE_PATH是相對路徑不是絕對路徑
//DCIM是系統(tǒng)文件夾斤蔓,關(guān)于系統(tǒng)文件夾可以到系統(tǒng)自帶的文件管理器中查看,不可以寫沒存在的名字
contentValues.put(MediaStore.Images.Media.RELATIVE_PATH, "DCIM/MNMZ");
//contentValues.put(MediaStore.Images.Media.RELATIVE_PATH, "Music/signImage");
} else {
contentValues.put(MediaStore.Images.Media.DATA, Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES).getPath());
}
contentValues.put(MediaStore.Images.Media.MIME_TYPE, "image/jpeg");
uri = contentResolver.insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, contentValues);
OutputStream outputStream =null;
try {
outputStream = getContentResolver().openOutputStream(uri);
bitmap.compress(Bitmap.CompressFormat.JPEG, 80, outputStream);
outputStream.flush();
outputStream.close();
Toast.makeText(MainActivity.this,"保存成功",Toast.LENGTH_SHORT).show();
}catch (IOException e) {
e.printStackTrace();
Toast.makeText(MainActivity.this,"保存失敗"+e.toString(),Toast.LENGTH_SHORT).show();
}
}
在AndroidQ中適配讀取圖片
1.在指定的path下獲取圖片
私有目錄:可以直接
Bitmap bitmap = BitmapFactory.decodeFile(path);獲取圖片
有時候filePath下的文件會很大所以我們通過寬高的比例來縮放圖片
/**
* 從文件路徑中獲取bitmap,根據(jù)比例inSampleSize镀岛,來縮放圖片
*/
public static Bitmap getBitmapFromPath(String pathName, int newWidth, int newHeight) {
BitmapFactory.Options opts = new BitmapFactory.Options();
opts.inJustDecodeBounds = true;//設(shè)置為ture只獲取圖片大小
BitmapFactory.decodeFile(pathName, opts);
opts.inSampleSize = getInSampleSize(opts, newWidth, newHeight);//計(jì)算縮放率弦牡,縮放圖片
opts.inJustDecodeBounds = false;//至為false
return BitmapFactory.decodeFile(pathName, opts);
}
/**
* 計(jì)算InSampleSize,大于1的整數(shù)時是縮小原圖
*/
private static int getInSampleSize(BitmapFactory.Options opts, int newW, int newH) {
int outWidth = opts.outWidth;
int outHeight = opts.outHeight;
if (outWidth > newW || outHeight > newH)
return (int) Math.ceil(Math.max(outWidth * 1d / newW, outHeight * 1d / newH));
return 1;
}
公共媒體:
在公共媒體不能直接通過path來獲取文件只能操作文件的uri來操作;
所以我們可以根據(jù)path轉(zhuǎn)換成uri漂羊。
public static Uri getImageContentUri(Context context, String path) {
Cursor cursor = context.getContentResolver().query(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
new String[] { MediaStore.Images.Media._ID }, MediaStore.Images.Media.DATA + "=? ",
new String[] { path }, null);
if (cursor != null && cursor.moveToFirst()) {
int id = cursor.getInt(cursor.getColumnIndex(MediaStore.MediaColumns._ID));
Uri baseUri = Uri.parse("content://media/external/images/media");
return Uri.withAppendedPath(baseUri, "" + id);
} else {
// 如果圖片不在手機(jī)的共享圖片數(shù)據(jù)庫驾锰,就先把它插入。
if (new File(path).exists()) {
ContentValues values = new ContentValues();
values.put(MediaStore.Images.Media.DATA, path);
return context.getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);
} else {
return null;
}
}
}
可以直接操作文件的uri走越,也可以轉(zhuǎn)換成bitmap(通過getContentResolver().openFileDescriptor(uri,"r")) "r"表示讀椭豫,"w"表示寫
public static Bitmap getBitmapFromUri(Context context, Uri uri) {
try {
ParcelFileDescriptor parcelFileDescriptor =
context.getContentResolver().openFileDescriptor(uri, "r");
FileDescriptor fileDescriptor = parcelFileDescriptor.getFileDescriptor();
Bitmap image = BitmapFactory.decodeFileDescriptor(fileDescriptor);
parcelFileDescriptor.close();
return image;
} catch (Exception e) {
e.printStackTrace();
}
return null;
}