AssetFileDescriptor des = getAssets().openFd("GPSResp.dat");报错

以下代码:

public void readAsset(View view){
		String result="";
    	try{
    		
    		AssetFileDescriptor des = getAssets().openFd("GPSResp.dat");
    		FileInputStream in = des.createInputStream();
    		
    		int length = in.available();
        	byte [] buffer = new byte[length];
        	in.read(buffer);
        	result = new String(buffer, "UTF-8");
        	System.out.println("result:" + result);
    	}
    	catch(IOException e){
    		e.printStackTrace();
    	}
    	catch(Exception e){
    		e.printStackTrace();
    	}
	}

会报如下错误:

08-01 18:38:52.785: W/System.err(3361): java.io.FileNotFoundException: This file can not be opened as a file descriptor; it is probably compressed
08-01 18:38:52.786: W/System.err(3361): 	at android.content.res.AssetManager.openAssetFd(Native Method)
08-01 18:38:52.787: W/System.err(3361): 	at android.content.res.AssetManager.openFd(AssetManager.java:331)
08-01 18:38:52.788: W/System.err(3361): 	at com.example.testasset.MainActivity.readAsset(MainActivity.java:29)
08-01 18:38:52.789: W/System.err(3361): 	at java.lang.reflect.Method.invokeNative(Native Method)
08-01 18:38:52.789: W/System.err(3361): 	at java.lang.reflect.Method.invoke(Method.java:511)
08-01 18:38:52.790: W/System.err(3361): 	at android.view.View$1.onClick(View.java:3599)
08-01 18:38:52.791: W/System.err(3361): 	at android.view.View.performClick(View.java:4209)
08-01 18:38:52.792: W/System.err(3361): 	at android.view.View$PerformClick.run(View.java:17431)
08-01 18:38:52.793: W/System.err(3361): 	at android.os.Handler.handleCallback(Handler.java:725)
08-01 18:38:52.793: W/System.err(3361): 	at android.os.Handler.dispatchMessage(Handler.java:92)
08-01 18:38:52.794: W/System.err(3361): 	at android.os.Looper.loop(Looper.java:153)
08-01 18:38:52.795: W/System.err(3361): 	at android.app.ActivityThread.main(ActivityThread.java:5297)
08-01 18:38:52.796: W/System.err(3361): 	at java.lang.reflect.Method.invokeNative(Native Method)
08-01 18:38:52.797: W/System.err(3361): 	at java.lang.reflect.Method.invoke(Method.java:511)
08-01 18:38:52.797: W/System.err(3361): 	at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:833)
08-01 18:38:52.798: W/System.err(3361): 	at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:600)
08-01 18:38:52.799: W/System.err(3361): 	at dalvik.system.NativeStart.main(Native Method)

解决方案:


1.把

AssetFileDescriptor des = getAssets().openFd("GPSResp.dat");
    		FileInputStream in = des.createInputStream();
替换为

InputStream in = getResources().getAssets().open("GPSResp.dat");	

2.把扩展名改为mp3。


原因:参见http://ponystyle.com/blog/2010/03/26/dealing-with-asset-compression-in-android-apps/


担心网站某天不能访问了,文章抄录如下:

Dealing with Asset Compression in Android Apps


When developing an Android app, any data file, image or XML file (that is, any Resource or Asset) you use is bundled into your application package (APK) for distribution. The Android Asset Packaging Tool, or aapt, is responsible for creating this bundle, which you can think of as a ZIP file with a particular layout that the Android OS can understand. When your application is installed, whether in development mode or by an end user, this APK file is simply dropped into a special location on the device’s (or emulator’s) filesystem.

As part of preparing your APK, aapt selectively compresses various assets to save space on the device. The way aapt determines which assets need compression is by their file extension. The following code, taken from Package.cpp in the aaptsource code, sheds some light on which types of files are not compressed by default:

/* these formats are already compressed, or don't compress well */
static const char* kNoCompressExt[] = {
    ".jpg", ".jpeg", ".png", ".gif",
    ".wav", ".mp2", ".mp3", ".ogg", ".aac",
    ".mpg", ".mpeg", ".mid", ".midi", ".smf", ".jet",
    ".rtttl", ".imy", ".xmf", ".mp4", ".m4a",
    ".m4v", ".3gp", ".3gpp", ".3g2", ".3gpp2",
    ".amr", ".awb", ".wma", ".wmv"
};

The only way (that I’ve discovered, as of this writing) to control this behavior is by using the -0 (zero) flag to aapt on the command line. This flag, passed without any accompanying argument, will tell aapt to disable compression for all types of assets. Typically you will not want to use this exact option, because for most assets, compression is a desirable thing. Sometimes, however, you will have a specific type of asset (say, a database), that you do not want to apply compression to. In this case, you have two options.

First, you can give your asset file an extension in the list above. While this does not necessarily make sense, it can be an easy workaround if you don’t want to deal with aapt on the command line. The other option is to pass a specific extension to the -0 flag, such as -0 db, to disable compression for assets with that extension. You can pass the -0 flag multiple times, and each time with a separate extension, if you need more than one type to be uncompressed.

Currently, there is no way to pass these extra flags to aapt when using the ADT within Eclipse, so if you don’t want to sacrifice the ease of use of the GUI tools, you will have to go with the first option, and rename your file’s extension.

So, Why Disable Compression Anyway?

For most types of assets, you shouldn’t need to be concerned with how they are packaged. Given the list of extensions above, Android will do the right thing for a large subset of the files you’ll use. If you have a type of asset that is already compressed by nature of the file type, but doesn’t have one of the sanctioned extensions, you can typically ignore that inconsistency and just allow aapt to try to compress it. It will not be very much, if at all, smaller in the final APK, but the performance hit should be relatively minimal unless you just have tons of these files.

This begs the question, why even bother disabling compression? The not-so-obvious answer is that prior to Android 2.3, any compressed asset file with an uncompressed size of over 1 MB cannot be read from the APK. This could come into play if your asset needed to be copied out of the APK and into the app’s writable files area, for example to provide a pre-populated database in your app. When you try to use the various AssetManager or Resources classes’ methods to get an InputStream to the file, it will throw an exception and display a LogCat message like this:

DEBUG/asset(725): Data exceeds UNCOMPRESS_DATA_MAX (1662976 vs 1048576)

The only way around this type of issue is to disable compression for assets that exceed the 1 MB limit. If your file format supports it, you can choose to split the asset into several smaller files that have an uncompressed size of less than 1 MB each, but this can be a real annoyance.

Hopefully the knowledge in this post can help you avoid the headaches I went through to learn it!

Update March 1, 2011: The limit on the uncompressed size of compressed assets was removed in Android 2.3. So someday in the future, when you don’t have to worry about Android versions lower than 2.3, you can avoid this heartache.

This entry was posted in android. Bookmark the  permalinkPost a comment or leave a trackback:  Trackback URL.
另外一篇文章的连接地址: http://gossipcoder.com/?p=858

疑问:我的asset文件夹中的文件未超过1M,为什么还会报错?但是把扩展名改为.mp3后确实解决了问题。







  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值