解析Android资源文件及他们的读取方法详解

Sam在Android开发中,有两种处理资源文件的方式。其一,是将所有资源文件以及JNI程序放置于一个单独的资源包。使用到他们时,使用文件方式读取。或者直接使用C++层代码读取。 其二,则是将资源文件加入到APK内部。使用各种不同的办法去得到其内容。
方法一:适合于移植较大的C++程序时使用,因为C++代码数量众多,不太可能修改为JAVA代码。所以将其与资源文件以一定方式存放,并让他们自称体系是个好办法。但这造成软件的发布必须以APK+资源包的方式发布。
方法二:则比较适合代码量不是非常大,且资源数量不是特别多的情况。此时,用户安装APK后,不用再费力copy资源包。方便发布。
这次主要介绍的是第二种方式,资源加入APK方式。
0.Android资源介绍:
Android应用程序开发时,大家通常都会用到以下资源:
res/drawable: 通常用来存放图片资源。如logo等。
res/layout:布局文件。
res/values:存放String,如程序名等。
但Android其实还可以使用其它类型资源。今天介绍3种如下:
res/xml: 存放xml文件,与之前所说的资源类似,存放在其中的资源文件会被编译为二进制数据而存入安装包内。通过R类读取xml文件。
res/raw: 存放文件。此目录下文件与之前的资源不同,他们不会被编译为二进制文件.而是以文件形式存放起来。通过R类读取。
assets: 可以在此创建子目录并存放不同文件。不会被编译入二进制,而是以目录/文件存放。通过文件名读取。
1. 各类文件读取:
1.1:res/raw:
android.app.Activity有一个间接父类: android.content.Context
它有一个方法与应用程序资源包有很大关系:
public abstract Resources getResources ()
它返回本应用程序的资源包实例。此实例是android.content.res.Resources类对象。
Sam首先添加raw目录。光标选中res. 菜单中:File->New->Folder. 输入目录名:raw.
并将一个wav---tennis_room.wav文件copy到此目录中并Refresh工程。
此时察看gen中R class.
发现已经添加进入:


代码如下:

public static final class raw {
        public static final int tennis_room=0x7f040000;
    }

例子:


代码如下:

int byteread = 0;
        byte[] buf = new byte[4096];
        FileInputStream inStream = null;

res = getResources();
        AssetFileDescriptor fd = res.openRawResourceFd(R.raw.tennis_room);

try {
inStream = fd.createInputStream();
} catch (IOException e) {
// TODO Auto-generated catch block
Log.e("3DiJoy", "createInputStream error");
e.printStackTrace();
}

try {
while((byteread = inStream.read(buf)) != -1)
{
//do something
}
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

1.2:res/xml:
与raw类似,但与xml有关。下节再说。
getXml(int id)
1.3:assets:
同样, android.app.Activity的间接父类:android.content.Context
有个方法:public abstract AssetManager getAssets ()
返回应用程序包的 AssetManager实例。
使用 InputStream open (String fileName);
返回一个InputStream.
则可以读取文件了。
注意,文件是以assets为根目录的。


代码如下:

AssetManager am = getAssets();

try {
am.open("a.txt");
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

assets文件有最大限制:
UNCOMPRESS_DATA_MAX: 1048567字节
assets文件目录分析:
使用getAssets()得到AssetsManager 实例后。可以打开文件,列出所有文件和目录。但它的路径和目录是怎样的呢?我们做如下测试:
首先:我们做程序列出给定目录下所有文件和目录:


代码如下:

public void ListAssetsFile(String AssetsPath)
    {
    AssetManager am = getAssets();
    try {
String[] FileOrDirName = am.list(AssetsPath);
Log.e("3DiJoy", String.format("In Assets Path: [%s]. There is:[%d] file or Dir", AssetsPath, FileOrDirName.length));
for(int i = 0; i < FileOrDirName.length; i++)
{
Log.e("3DiJoy", String.format("File Or Dir:[%s]", FileOrDirName[i]));
}
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

return;
    }

关注点1:
如何判断Assets中某个节点是文件还是目录:
Sam看到网络上一些朋友的做法是判断文件名中是否有 "."  .觉得这个办法不是特别有效,所以作了另一个尝试。


代码如下:

// true: Dir.  false:file
    public boolean isAssetsDirs(String fileOrDirName)
    {
    AssetManager am = getAssets();

try {
am.open(fileOrDirName);
return false;
}
    catch (FileNotFoundException e)
    {
    return true;
    }
    catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
return true;
}

//return !(fileName.startsWith(".") || (fileName.lastIndexOf(".") != -1));
    }

当使用 am.open()时,如果指定的是个目录,则会抛出 FileNotFoundException 异常。 Sam就是利用这一点判断是否为目录。
关注点2:
如何将Assets下某个目录copy到本地:
即做到类似:


代码如下:

#cp DIR_A/* -rf /data/data/.../
public boolean CopyAssetsPath(String AssetsPath, String ObjectPath)
    {
    File ObjPath = new File(ObjectPath);
    if(!ObjPath.exists() || !ObjPath.isDirectory())
    {
    Log.e("3DiJoy", "Object Path not found or not Dir:"+ ObjectPath);
    return false;
    }

AssetManager am = getAssets();

try {
String[] FileOrDirName = am.list(AssetsPath);
//Log.e("3DiJoy", String.format("In Assets Path: [%s]. There is:[%d] file or Dir", AssetsPath, FileOrDirName.length));
for(int i = 0; i < FileOrDirName.length; i++)
{
// if this is a DIR
if(isAssetsDirs(AssetsPath+ "/" + FileOrDirName[i]))
{
File N_DIR = new File(ObjectPath + "/" + FileOrDirName[i]);
    if(!N_DIR.exists())
    {
    Log.e("3DiJoy", String.format("Will Create Dir:[%s]", ObjectPath + "/" + FileOrDirName[i]));
    N_DIR.mkdir();
    CopyAssetsPath(AssetsPath + "/" +FileOrDirName[i], ObjectPath + "/" + FileOrDirName[i]);
    }
}
else  // if this is file. Then copy it
{
Log.e("3DiJoy", String.format("Will Create file:[%s]", ObjectPath + "/" + FileOrDirName[i]));
CopyAssets(AssetsPath + "/" + FileOrDirName[i], ObjectPath + "/" + FileOrDirName[i]);
}
//Log.e("3DiJoy", String.format("File Or Dir:[%s]", FileOrDirName[i]));
}
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

return true;
    }

程序很简单:
使用list列出所有文件和目录。
如果是目录:则在目标区域建立一个同名目录。
如果为文件,则copy it。
关注点3:
如何访问和copy一个超过1M的文件:
上面的程序,如果有文件超过1M,则会报异常。
抛出java.io.IOException的异常如下    
DEBUG/asset(1123): Data exceeds UNCOMPRESS_DATA_MAX (xxxxxxxx vs 1048576);
但请注意:以下文件不受1M大小限制


代码如下:

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"

可以将超过大小的文件,添加以下文件名即可。
测试1:
测试根目录位置:
ListAssetsFile("/");
得到信息是:
更目录其实就是APK解压缩后的根目录:
内容包括:
AndroidManifest.xm.
assets
META-INFO
lib
res
classes.dex
resources.arsc
测试2:
测试相对路径位置:
ListAssetsFile("");
列出的内容是Assets目录中的内容。但不知为何,添加了三项内容:
image, sound, webkit.
测试3:测试当前路径位置:
ListAssetsFile("./");
理论上,./目录应该和当前目录一样,不知为何,此处却无法得到任何文件。不太理解。
因为测试3,所以对Android Assets目录与我们Linux下概念是否相同有了怀疑,所以再次测试:
测试4:
看绝对路径是否可用:
ListAssetsFile("/assets");
呵呵,果然证实,它无法得到其中任何文件。
测试4:
看能否用绝对路径访问assets之外的文件:
ListAssetsFile("/lib");
果然返回0个文件。呵呵。
结论:
想要访问assets文件,只能使用相对路径,且前面不能加 ./

(0)

相关推荐

  • Android 读取资源文件实例详解

    Android 读取资源文件实例详解 本文主要介绍 Android 读取资源文件,直接从 assets 读取,从 Raw 文件中读取,InputStream 转 String. 以下为直接从assets读取: /** * 得到Assets里面相应的文件流 * * @param fileName * @return */ private InputStream getAssetsStream(String fileName) { InputStream is = null; try { is =

  • android读取Assets图片资源保存到SD卡实例

    复制代码 代码如下: public class ReadBitmap { public void readByte(Context c, String name, int indexInt) { byte[] b = null; int[] intArrat = c.getResources().getIntArray(indexInt); try { AssetManager am = null; am = c.getAssets(); InputStream is = am.open(nam

  • Android使用URL读取网络资源的方法

    URL(Uniform Resource Locator)是统一资源定位器,它是对可以从互联网上得到的资源的位置和访问方法的一种简洁的表示,是互联网上标准资源的地址.互联网上的每个文件都有一个唯一的URL,它包含的信息指出文件的位置以及浏览器应该怎么处理它.就通常情况而言,URL可以由协议名.主机.端口和资源组成. URL类提供了多个构造器用于创建URL对象,一旦获得了URL对象之后,就可以调用如下常用方法来访问该URL对应的资源了. ->String getFile():获取此URL的资源名:

  • Android读取资源文件的方法

    Android读取资源文件的方法 1.放入到资源文件夹里面,也就是所创建android工程的res下面. 我们可以在里面建立文件夹,放置我们要读取的文件. 例如:res里面建立目录raw,将要读取的文件abc.bin放入到该文件夹下,此时通过 openRawResource()方法即可读取.代码为 InputStream is = getResources().openRawResource(R.raw.abc); 只要获取InputStream,一切就好办了! 注意:res下面的目录名字是有讲

  • android从资源文件中读取文件流并显示的方法

    本文实例讲述了android从资源文件中读取文件流并显示的方法.分享给大家供大家参考.具体如下: 在android中,假如有的文本文件,比如TXT放在raw下,要直接读取出来,放到屏幕中显示,可以这样: private void doRaw(){ InputStream is = this.getResources().openRawResource(R.raw.ziliao); try{ doRead(is); }catch(IOException e){ e.printStackTrace(

  • Android中实现下载URL地址的网络资源的实例分享

    通过URL来获取网络资源并下载资源简单实例: package com.android.xiong.urltest; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.net.MalformedURLException; import java.net.URL; import android.app.Activity; import android.gra

  • 解析Android资源文件及他们的读取方法详解

    Sam在Android开发中,有两种处理资源文件的方式.其一,是将所有资源文件以及JNI程序放置于一个单独的资源包.使用到他们时,使用文件方式读取.或者直接使用C++层代码读取. 其二,则是将资源文件加入到APK内部.使用各种不同的办法去得到其内容.方法一:适合于移植较大的C++程序时使用,因为C++代码数量众多,不太可能修改为JAVA代码.所以将其与资源文件以一定方式存放,并让他们自称体系是个好办法.但这造成软件的发布必须以APK+资源包的方式发布.方法二:则比较适合代码量不是非常大,且资源数

  • Java实现读取项目中文件(.json或.properties)的方法详解

    目录 1. 读取json file 1.1 Json dependency 1.2 字节流 1.3 buffer reader 2. 读取properties file 3. 好看的css样式 1. 读取json file 1.1 Json dependency <dependency> <groupId>com.alibaba</groupId> <artifactId>fastjson</artifactId> <version>

  • jQ处理xml文件和xml字符串的方法(详解)

    1.xml文件 <?xml version="1.0" encoding="utf-8" ?> <root> <book id="1"> <name>锋利的jQuery1</name> <author>XXX1</author> <price>250</price> </book> <book id="2&quo

  • 对python中大文件的导入与导出方法详解

    1.csv文件的导入和导出 通过一个矩阵导出为csv文件,将csv文件导入为矩阵 将csv文件导入到一个矩阵中 import numpy my_matrix = numpy.loadtxt(open("c:\\1.csv","rb"),delimiter=",",skiprows=0) 将矩阵导出到本地csv中 numpy.savetxt('new.csv', my_matrix, delimiter = ',') 未完待续... 也可以使用pi

  • 对numpy中二进制格式的数据存储与读取方法详解

    使用save可以实现对numpy数据的磁盘存储,存储的方式是二进制.查看使用说明,说明专门提到了是未经压缩的二进制形式.存储后的数据可以进行加载或者读取,通过使用load方法. In [81]:np.save('demo',data1) 通过以上操作,数据data1被存储到了demo文件中,numpy会自动加上npy的文件后缀名. In [82]: a =np.load('demo.npy') In [83]: a Out[83]: array([0,1, 2, 3, 4, 5, 6, 7, 8

  • Android中实现ping功能的多种方法详解

    使用java来实现ping功能. 并写入文件.为了使用java来实现ping的功能,有人推荐使用java的 Runtime.exec()方法来直接调用系统的Ping命令,也有人完成了纯Java实现Ping的程序,使用的是Java的NIO包(native io, 高效IO包).但是设备检测只是想测试一个远程主机是否可用.所以,可以使用以下三种方式来实现: 1. Jdk1.5的InetAddresss方式 自从Java 1.5,java.net包中就实现了ICMP ping的功能. 使用时应注意,如

  • Linux查看日志文件写入速度的4种方法详解

    目录 简介 使用dd 使用pv 使用cv命令 编写小脚本 举一反三 简介 有时,我们需要查看某个文件的增长速度,如日志文件,以此来感受系统的负载情况,因为一般情况下,日志写入越快,说明系统负载越重. 本文就来介绍下Linux中查看日志增长速度的几种方法,如下: 使用dd 首先要介绍的是dd,因为dd命令几乎所有主流发行版都自带,无需额外安装,如下: $ tail -F app.log | dd of=/dev/null status=progress 3875840 bytes (3.9 MB)

  • SpringCloud实现文件上传功能的方法详解

    目录 图片上传 搭建项目 编写上传功能 图片上传 刚才的新增实现中,我们并没有上传图片,接下来我们一起完成图片上传逻辑. 文件的上传并不只是在品牌管理中有需求,以后的其它服务也可能需要,因此我们创建一个独立的微服务,专门处理各种上传. 搭建项目 创建SpringCloud项目 添加依赖 我们需要EurekaClient和web依赖: <?xml version="1.0" encoding="UTF-8"?> <project xmlns=&quo

  • SpringBoot yml配置文件读取方法详解

    目录 yaml介绍 yaml语法规则 yaml数据读取 Environment读取yaml全部属性数据 自定义对象封装指定数据 yaml介绍 YAML(YAML Ain't Markup Language),一种数据序列化格式 优点: 容易阅读 容易与脚本语言交互 以数据为核心,重数据轻格式 YANL文件扩展名 .yml(主流) .yaml 几种数据格式比较 yaml语法规则 大小写敏感 属性层级关系使用多行描述,每行结尾使用冒号结束 使用缩进表示层级关系,同层级左侧对齐,只允许使用空格(不允许

  • Android ListView监听滑动事件的方法(详解)

    ListView的主要有两种滑动事件监听方法,OnTouchListener和OnScrollListener 1.OnTouchListener OnTouchListener方法来自View中的监听事件,可以在监听三个Action事件发生时通过MotionEvent的getX()方法或getY()方法获取到当前触摸的坐标值,来对用户的滑动方向进行判断,并可在不同的Action状态中做出相应的处理 mListView.setOnTouchListener(new View.OnTouchLis

随机推荐