内容提供程序/解析程序API提供了一种使用URI和openInputStream()
和openOutputStream()
方法在进程之间传输数据的复杂但强大的方法。自定义内容提供商可以使用自定义代码覆盖openFile()
方法,以有效地将URI解析为Stream
;但是,openFile()
的方法签名具有ParcelFileDescriptor
返回类型,并且不清楚如何为此方法返回动态生成的内容生成正确的表示。
Returning a memory mapped InputStream from a content provider?
是否有针对现有代码库中的动态内容实施ContentProvider.openFile()
方法的示例?如果没有,你可以建议这样做的源代码或过程吗?
答案 0 :(得分:24)
从常用的CommonsWare中查看这个很棒的示例项目。它允许您创建一个ParcelFileDescriptor管道,其中包含您想要的任何一个InputStream,以及另一侧的接收应用程序:
https://github.com/commonsguy/cw-omnibus/tree/master/ContentProvider/Pipe
关键部分是在openFile
:
public ParcelFileDescriptor openFile(Uri uri, String mode)
throws FileNotFoundException {
ParcelFileDescriptor[] pipe=null;
try {
pipe=ParcelFileDescriptor.createPipe();
AssetManager assets=getContext().getResources().getAssets();
new TransferThread(assets.open(uri.getLastPathSegment()),
new AutoCloseOutputStream(pipe[1])).start();
}
catch (IOException e) {
Log.e(getClass().getSimpleName(), "Exception opening pipe", e);
throw new FileNotFoundException("Could not open pipe for: "
+ uri.toString());
}
return(pipe[0]);
}
然后创建一个保持管道满的线程:
static class TransferThread extends Thread {
InputStream in;
OutputStream out;
TransferThread(InputStream in, OutputStream out) {
this.in = in;
this.out = out;
}
@Override
public void run() {
byte[] buf = new byte[8192];
int len;
try {
while ((len = in.read(buf)) > 0) {
out.write(buf, 0, len);
}
in.close();
out.flush();
out.close();
} catch (IOException e) {
Log.e(getClass().getSimpleName(),
"Exception transferring file", e);
}
}
}
答案 1 :(得分:1)
MemoryFile支持此功能,但公共API尚未最终确定。