Android檔案操作 —— 6.0之後檔案操作
阿新 • • 發佈:2019-01-24
今天做一個需求,需要在SD卡建立公共資料夾,以前也做過,建立SD卡資料夾的程式碼很簡單。但是卻出現了問題,在6.0版本之後都不能建立,最後查了文件發現問題所在,就記錄一下(很久沒有做檔案操作相關的功能,現在都7.0了,out太多!)。
1、許可權宣告
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
2、程式碼砸上
if (Environment.getExternalStorageState()
.equals(Environment.MEDIA_MOUNTED)){
try {
File dir = new File(filePath);
if (!dir.exists()) {//判斷檔案目錄是否存在
dir.mkdir();
}
file = new File(filePath + File.separator + fileName);
if (!file.exists()){
file.createNewFile();
}
bos = new BufferedOutputStream(new FileOutputStream(file));
bos.write(bfile);
} catch (Exception e) {
e.printStackTrace();
} finally {
IOUtitl.close(bos);
}
但是很遺憾,我這次卻建立失敗:
ava.io.IOException: open failed: ENOENT (No such file or directory)
xxxx xxxx xxxx W/System.err: at java.io.File.createNewFile(File.java:942)
解決方案:
然後我就懵了,差點懷疑自己。然後我就一步一步除錯!最後確信程式碼沒問題,就去google了一下,發現問題出在系統版本上。原來在Android 6.0之後,有些許可權是需要使用者同意才能生效的,否則並沒有卵用。如果你的程式碼執行在6.0之後的系統,那麼恭喜你,加上下面的程式碼就搞定了:
// Storage Permissions
private static final int REQUEST_EXTERNAL_STORAGE = 1;
private static String[] PERMISSIONS_STORAGE = {
Manifest.permission.READ_EXTERNAL_STORAGE,
Manifest.permission.WRITE_EXTERNAL_STORAGE };
/**
* Checks if the app has permission to write to device storage
* If the app does not has permission then the user will be prompted to
* grant permissions
* @param activity
*/
public static void verifyStoragePermissions(Activity activity) {
// Check if we have write permission
int permission = ActivityCompat.checkSelfPermission(activity,
Manifest.permission.WRITE_EXTERNAL_STORAGE);
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
return ;
}
if (permission != PackageManager.PERMISSION_GRANTED) {
// We don't have permission so prompt the user
ActivityCompat.requestPermissions(activity, PERMISSIONS_STORAGE,
REQUEST_EXTERNAL_STORAGE);
}
}
//使用者處理許可權反饋,在這裡判斷使用者是否授予相應的許可權
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
boolean writeAccepted = false;
switch (requestCode) {
case REQUEST_EXTERNAL_STORAGE:
if (grantResults.length == 1){
writeAccepted = grantResults[0] == PackageManager.PERMISSION_GRANTED;
}
break;
default:
break;
}
if (writeAccepted){//在6.0以上的系統,使用者需要自己確認是否授權,授權過後重新初始化伺服器配置檔案
MessageCenter.postMsgNotify(MessageConstant.OBSERVER_SERVER_CHANGE,new MessageCenter.Notification<Object>());
}
}
預設的檔案操作
預設情況下,我們通過context.openOutputFile()得到的是程式的私有檔案,存放在/data/data/Package name/下,不在root模式下面我們是無法看到檔案的,同時也無法操作/data資料夾,我們得到的許可權僅僅是程式所在的packageName資料夾下的讀寫許可權。