- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我的目的是能够在 Android 21+ 上编辑可移动外部存储上的文件。为此,必须使用存储访问框架。
首先我授予了我感兴趣的文件夹的读写权限:
startActivityForResult(new Intent(Intent.ACTION_OPEN_DOCUMENT_TREE), 123);
@Override
public void onActivityResult(int requestCode,int resultCode,Intent resultData) {
if (resultCode != RESULT_OK)
return;
Uri treeUri = resultData.getData();
PreferenceUtils.setPersistedUri(this, treeUri);
grantUriPermission(getPackageName(), treeUri, Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
getContentResolver().takePersistableUriPermission(treeUri, Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
}
我确信这是正确完成的,因为我仔细检查了它:
for(UriPermission uri : getContentResolver().getPersistedUriPermissions())
if(uri.isReadPermission() && uri.isWritePermission())
// This prints the correct folder
如果相关,我还在 list 中声明并请求了 READ_EXTERNAL_STORAGE
和 WRITE_EXTERNAL_STORAGE
的运行时权限。
DocumentFile parentDocumentFile = DocumentFile.fromTreeUri(getActivity(), PreferenceUtils.getPersistedUri());
if(parentDocumentFile == null) return;
for(DocumentFile file : parentDocumentFile.listFiles()) {
if(file.canRead() && file.canWrite() && file.exists()) {
try
{
OutputStream outStream = getActivity().getContentResolver().openOutputStream(file.getUri());
// I/O operations
} (catch FileNotFoundException e) {}
}
}
在代码的后面,我获得了对我感兴趣的文件的引用,该文件的内容 URI 由 DocumentFile
包装。我想强调 DocumentFile.exists()
、DocumentFile.canRead()
、DocumentFile.canWrite()
返回肯定结果,也是检查使用系统 File.exists()
已经完成。例如,我通过 DocumentFile.getUri()
获得的 URI 如下所示:
content://com.android.externalstorage.documents/tree/486E-2542%3A/document/486E-2542%3ADownload%2FFildoDownloads%2FLed%20Zeppelin%2FLed%20Zeppelin%20-%20Kashmir.mp3
好的。但是如果我想打开它进行写入,我不能,这是唯一可用的写入文件的方法,该文件的权限是通过存储访问框架授予的:
OutputStream outStream = getContentResolver().openOutputStream(DocumentFile.getUri());
一些设备(我无法重现)抛出一个 FileNotFoundExcpetion
并带有以下堆栈跟踪:
java.io.FileNotFoundException: Failed opening content provider: content://com.android.externalstorage.documents/tree/486E-2542%3A/document/486E-2542%3ADownload%2FFildoDownloads%2FLed%20Zeppelin%2FLed%20Zeppelin%20-%20Kashmir.mp3
at android.content.ContentResolver.openAssetFileDescriptor(ContentResolver.java:1032)
at android.content.ContentResolver.openOutputStream(ContentResolver.java:718)
at android.content.ContentResolver.openOutputStream(ContentResolver.java:694)
at helpers.c.b(SourceFile:405)
一些其他的:
java.io.FileNotFoundException: Failed to open for writing: java.io.FileNotFoundException: Read-only file system
at android.database.DatabaseUtils.readExceptionWithFileNotFoundExceptionFromParcel(DatabaseUtils.java:144)
at android.content.ContentProviderProxy.openAssetFile(ContentProviderNative.java:621)
at android.content.ContentResolver.openAssetFileDescriptor(ContentResolver.java:1011)
at android.content.ContentResolver.openOutputStream(ContentResolver.java:753)
at android.content.ContentResolver.openOutputStream(ContentResolver.java:729)
at helpers.c.b(SourceFile:405)
这种情况不经常发生,比方说 200 台设备中大约有 1 台设备,但它仍然是一个相关问题,因为它破坏了应用程序的主要用途,因此用户可以将其卸载。 p>
StackOverflow 上所有过去的问题看起来都与此类似,但那里的解决方案是不写入存储的根目录,这显然不是这里的问题。在这种特定情况下,我真的不知道这些异常,所以我可能会问是否有人知道可能无法获得具有文件写访问权限的文件描述符的情况。
最佳答案
异常中的消息显示“只读文件系统”。该消息通常与 native 错误号 "EROFS" 相关联,当 Linux 进程试图打开只读存储上的文件进行写入时,就会发生这种情况。
存储可以以只读方式挂载的原因有多种。例如,在严重数据损坏的情况下,大多数文件系统会被内核自动重新挂载为只读(Android 系统甚至可能没有意识到,设备 SD 卡已切换到只读模式)。
也有可能是因为Android系统代码的bug。在这种情况下,了解设备供应商和型号可能有助于调试。
最简单的解决方法:如果写入失败,尝试以只读模式打开文件。如果只读打开成功,问题可能可以通过弹出并重新插入 SD 卡来解决(这应该会导致 Android 自动在其上运行 fsck)。
Uri uri = file.getUri();
try (OutputStream o = context.getContentResolver().openOutputStream(uri))
{
// I/O operations
} (catch FileNotFoundException e) {
try (OutputStream s = context.getContentResolver().openInputStream(uri)) {
// the file can be opened for reading, but not for writing
// ask user to reinsert SD card and/or disable
// write protection toggle on it
Toast.makeText(context, "Please reinsert SD card", LENGTH_SHORT);
} catch (FileNotFoundException err) {
// This is hopeless, just report error to user
Toast.makeText(context, err.toString(), LENGTH_SHORT);
}
}
如果你想确定一个文件系统是否真的以只读方式挂载,考虑从特殊文件 /proc/self/mountinfo
中读取。您可以将它的内容发送给 Crashlytics 进行调试,甚至 parse them in your code .您可以从 Linux kernel documentation 了解有关每个进程 mountinfo
文件的更多信息.
关于java - ContentResolver.openOutputStream() 抛出 FileNotFoundException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48560028/
我正在尝试根据显示名称过滤文档。这是代码: val files = ArrayList(0) val uri = DocumentsContract.buildChildDocumentsUriUsi
如果 ContentResolver.addPeriodicSync 没有触发 onPerformSync(...) 方法,可能是什么问题。 但是 ContentResolver.requestSyn
我正在 Udacity 上学习 Android 类(class),并且遇到了一位学习者在标题中提出的问题。 Youtube link to lesson 引用自视频: ContentResolver
我是安卓领域的新手,正处于学习阶段。我有几个疑问: 每个应用程序是否有一个 ContentResolver 对象? 它是一个单例对象吗? 谁管理这个对象生命周期? 如果是单例,它如何处理多个查询 Co
我使用此代码通过 documentFile.createFile() 复制图像 private void newcopyFile(File fileInput, String outputParent
我在尝试删除具有特定 ID 的一行数据时遇到问题,而不是通过内容提供程序删除我数据库中的所有数据。 这里是删除内容提供者的方法: @Override public int delete(@No
我想查询用户选择的图片或视频的数据。我的 onActivityResult 查询具有给定 URI 的图像的数据字段。 protected final String[] queryColumns = {
当我使用原始的 Google 音乐播放器时,我的应用程序运行良好。它找到了歌曲和播放列表,没问题。自从我开始使用 Play Music 以来,我的应用程序找不到任何这些。这是我的代码: Cur
我的内容 uri 有一个取决于内容的路径,例如content://controller/ContentItem/0/Favorites/RecentlyPlayed 我观察到,当我用这个 uri 插入
我正在使用指定的代码 here存储 Bitmap 图像并将其显示到图库中。 不幸的是,该代码让我可以将图像存储到“图片”文件夹中的图库中。 是否可以使用我决定的名称更改文件夹名称? 最佳答案 好的,我
我几乎完成了我的应用程序......但现在我不能像这样使用 ContentResolver: Bitmap imageBitmap = MediaStore.Images.Media.getBitma
我们都知道 ContentResolver 查询不应该在 UI 线程上执行,但是,令人惊讶的是,我找不到关于 ContentResolver 类线程安全的信息官方文档。 我知道如何编写线程安全的 Co
执行insert new data into content provider时报错,看下面代码,PS,我直接把下面代码放到Activity的onCreate()中,然后执行。 ContentValu
我将文件添加到 Documents/MyExcelsFolder通过使用 ContentResolver.insert然后还将新文件添加到 Documents/MyExcelsFolder另一个应用程
我有这个更新代码: public Boolean update() { try { data.put(ContactsContract.Groups.SHOULD_SYNC, true);
根据this文档强制执行至少 60 分钟,但根据 this回答是60秒。正确的最小 pollFrequency 是多少? 最佳答案 最后,在我的应用程序上进行试验后,我能够每分钟同步一次,因此 and
我想删除我手机上的所有短信,但每次对话的最后 500 条短信除外。这是我的代码,但速度很慢(删除一条短信大约需要 10 秒)。我如何加速这段代码: ContentResolver cr = g
我有一个覆盖 bulkInsert() 的 MyContentProvider 类。在此方法中,我使用 SQLite 事务将大约 4,000 行插入数据库,这在三星 Galaxy S4 设备上大约需要
我正在使用 Cursor 从图库中获取图像并从图库中获取所有图像,有什么方法可以只获取最近拍摄的几张图像,例如最近拍摄的 20 张图像。 还有一个问题,我面临的是图像从旧到新的顺序,我想以相反的顺序(
我正在使用以下代码在设备上构建电子邮件地址列表以显示在 ListActivity 上。目前我可以正常检索电子邮件地址,但它们以以下形式出现:null (email@address.com) 而不是 n
我是一名优秀的程序员,十分优秀!