在Android sdcard上创建文件时发生IO异常,并授予

时间:2018-05-07 11:08:22

标签: android file android-permissions android-sdcard sd-card

我正在尝试将文件从内部手机内存移动到SD卡。

val oldFile = File(oldPath) //oldPath is string
val newFile = File(newPath) //newPath is string

if(newFile.exists()) {
    newFile.delete()
}

newFile.parentFile.mkdirs()
newFile.createNewFile() //crashes here

// copy file code

路径是绝对的。

旧文件路径为:

/storage/emulated/0/Android/data/com.package/files/2ec37ce3-ca72-4a35-a6e6-2d7f8e864c6c

新文件路径为:

/storage/48EE-C144/chosenDirectory/2ec37ce3-ca72-4a35-a6e6-2d7f8e864c6c

其中2ec37ce3-ca72-4a35-a6e6-2d7f8e864c6c是文件。

路径正确,用户可以使用外部目录选择器库手动选择新文件路径,该工作正常。

当然,app会询问用户的权限,并且在未授予权限之前不会将用户传递给应用程序。

int perm = this.checkSelfPermission(permission.WRITE_EXTERNAL_STORAGE);
int perm2 = this.checkSelfPermission(permission.READ_EXTERNAL_STORAGE);

if (perm != PackageManager.PERMISSION_GRANTED || perm2 != PackageManager.PERMISSION_GRANTED) {
    /*ask for permissions and show explain dialog*/
}

清单中的分类也被添加。

使用异常日志调用newFile.createNewFile()时应用程序崩溃:

05-07 13:02:14.995 14004-14004/com.packagename E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.packagename, PID: 14004
    java.lang.RuntimeException: java.lang.reflect.InvocationTargetException
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1451)
     Caused by: java.lang.reflect.InvocationTargetException
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1563)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1451) 
     Caused by: java.io.IOException: Permission denied
        at java.io.UnixFileSystem.createFileExclusively0(Native Method)
        at java.io.UnixFileSystem.createFileExclusively(UnixFileSystem.java:280)
        at java.io.File.createNewFile(File.java:948)
        at com.packagename.backend.localApi.changePath.PathChanger.changePath(PathChanger.kt:29)
        at com.packagename.backend.localApi.changePath.PathChangeService$changePath$1.onSelect(PathChangeService.kt:49)
        at com.codekidlabs.storagechooser.fragments.ChooserDialogFragment$1.onItemClick(ChooserDialogFragment.java:169)
        at android.widget.AdapterView.performItemClick(AdapterView.java:343)
        at android.widget.AbsListView.performItemClick(AbsListView.java:1665)
        at android.widget.AbsListView$PerformClick.run(AbsListView.java:4075)
        at android.widget.AbsListView$10.run(AbsListView.java:6552)
        at android.os.Handler.handleCallback(Handler.java:751)
        at android.os.Handler.dispatchMessage(Handler.java:95)
        at android.os.Looper.loop(Looper.java:154)
        at android.app.ActivityThread.main(ActivityThread.java:6816)
        at java.lang.reflect.Method.invoke(Native Method) 
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1563) 
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1451) 

2 个答案:

答案 0 :(得分:5)

需要在 AndroidManifest.xml 文件中添加权限。

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

答案 1 :(得分:0)

对我来说,解决方案是停止使用Java类,然后迁移到Storage Access Framework