我需要知道属于资源文件夹的文件的字符串路径,因为我正在使用需要接收字符串路径的地图API,而我的地图必须存储在资产文件夹< / p>
这是我正在尝试的代码:
MapView mapView = new MapView(this);
mapView.setClickable(true);
mapView.setBuiltInZoomControls(true);
mapView.setMapFile("file:///android_asset/m1.map");
setContentView(mapView);
"file:///android_asset/m1.map"
出现问题,因为地图未加载。
哪个是存储在我的资源文件夹中的文件 m1.map 的正确字符串路径文件?
由于
Dimitru的编辑:此代码不起作用,它在is.read(buffer);
上失败并且 IOException
try {
InputStream is = getAssets().open("m1.map");
int size = is.available();
byte[] buffer = new byte[size];
is.read(buffer);
is.close();
text = new String(buffer);
} catch (IOException e) {throw new RuntimeException(e);}
答案 0 :(得分:87)
AFAIK资产目录中的文件无法解压缩。 相反,它们直接从APK(ZIP)文件中读取。
所以,你真的不能制作需要文件接受资产'文件'的东西。
相反,您必须提取资产并将其写入单独的文件,例如Dumitru建议:
File f = new File(getCacheDir()+"/m1.map");
if (!f.exists()) try {
InputStream is = getAssets().open("m1.map");
int size = is.available();
byte[] buffer = new byte[size];
is.read(buffer);
is.close();
FileOutputStream fos = new FileOutputStream(f);
fos.write(buffer);
fos.close();
} catch (Exception e) { throw new RuntimeException(e); }
mapView.setMapFile(f.getPath());
答案 1 :(得分:9)
查看SDK附带的API示例中的ReadAsset.java。
try {
InputStream is = getAssets().open("read_asset.txt");
// We guarantee that the available method returns the total
// size of the asset... of course, this does mean that a single
// asset can't be more than 2 gigs.
int size = is.available();
// Read the entire asset into a local byte buffer.
byte[] buffer = new byte[size];
is.read(buffer);
is.close();
// Convert the buffer into a string.
String text = new String(buffer);
// Finally stick the string into the text view.
TextView tv = (TextView)findViewById(R.id.text);
tv.setText(text);
} catch (IOException e) {
// Should never happen!
throw new RuntimeException(e);
}
答案 2 :(得分:7)
您可以使用此方法。
public static File getRobotCacheFile(Context context) throws IOException {
File cacheFile = new File(context.getCacheDir(), "robot.png");
try {
InputStream inputStream = context.getAssets().open("robot.png");
try {
FileOutputStream outputStream = new FileOutputStream(cacheFile);
try {
byte[] buf = new byte[1024];
int len;
while ((len = inputStream.read(buf)) > 0) {
outputStream.write(buf, 0, len);
}
} finally {
outputStream.close();
}
} finally {
inputStream.close();
}
} catch (IOException e) {
throw new IOException("Could not open robot png", e);
}
return cacheFile;
}
在这种情况下,你应该永远不要使用InputStream.available()。它仅返回缓冲的字节。使用.available()的方法永远不会使用更大的文件,根本无法在某些设备上使用。
在Kotlin(; D):
@Throws(IOException::class)
fun getRobotCacheFile(context: Context): File = File(context.cacheDir, "robot.png")
.also {
it.outputStream().use { cache -> context.assets.open("robot.png").use { it.copyTo(cache) } }
}
答案 3 :(得分:3)
添加Jacek的完美解决方案。如果你想在Kotlin做这件事,它就不会马上工作。相反,你会想要使用它:
@Throws(IOException::class)
fun getSplashVideo(context: Context): File {
val cacheFile = File(context.cacheDir, "splash_video")
try {
val inputStream = context.assets.open("splash_video")
val outputStream = FileOutputStream(cacheFile)
try {
inputStream.copyTo(outputStream)
} finally {
inputStream.close()
outputStream.close()
}
} catch (e: IOException) {
throw IOException("Could not open splash_video", e)
}
return cacheFile
}