Android 资源文件分布

  Apk 文件下的资源文件都存放在以下两个目录下: res/, assets/, res/ 目录下的文件在打包成 apk 的时候被 encode 过,文本文件也变成了二进制的。如果想看文本内容,请参考下 android-apktool 工程。assets/ 目录下的文件会被原封不动的打包到 apk 文件中去。下面以一个例子说明下。

    资源源文件目录结构如下:

[lgao@lgao -]$ tree
.
|-- AndroidManifest.xml
|-- assets
|   |-- default.properties
|   |-- docs
|   |   `-- test.file
|   `-- test.xml
|-- res
|   |-- drawable
|   |   |-- taobao.ico
|   |   `-- taobao_tabs.xml
|   |-- drawable-hdpi
|   |   `-- icon.png
|   |-- drawable-ldpi
|   |   `-- icon.png
|   |-- drawable-mdpi
|   |   `-- icon.png
|   |-- layout
|   |   |-- main.xml
|   `-- values
|       `-- strings.xml

    当 apk 安装到模拟器中后, apk 被安装在 /data/app/test.apk,我们来测试下文件内容:

try {
	ZipInputStream zipIn = new ZipInputStream(new FileInputStream("/data/app/test.apk"));
	BufferedReader zipReader = new BufferedReader(new InputStreamReader(zipIn));
	ZipEntry entry = null;
	while((entry = zipIn.getNextEntry()) != null)
	{
		Log.i("TEST[APK-ENTRY]", entry.getName());
		if(entry.getName().indexOf("assets") != -1){
			String sl = null;
			while((sl = zipReader.readLine()) != null){
				Log.i("TEST[xml reader assets]", sl);
			}
		}else if (entry.getName().endsWith(".xml")){
			String sl = null;
			while((sl = zipReader.readLine()) != null){
				Log.i("TEST[xml reader xml]", sl);
			}
		}
	}
	} catch (FileNotFoundException e) {
		// TODO Auto-generated catch block
	} catch (IOException e) {
		// TODO Auto-generated catch block
		e.printStackTrace();
	}

  我们看得到的 Logcat 消息:

12-12 13:43:28.234: INFO/TEST[APK-ENTRY](740): assets/docs/test.file
12-12 13:43:28.254: INFO/TEST[xml reader assets](740): # test string line
12-12 13:43:28.295: INFO/TEST[APK-ENTRY](740): assets/default.properties
12-12 13:43:28.314: INFO/TEST[xml reader assets](740): # This file is automatically generated by Android Tools.
12-12 13:43:28.354: INFO/TEST[APK-ENTRY](740): assets/test.xml
12-12 13:43:28.364: INFO/TEST[xml reader assets](740): <? xml encoding="utf-8" ?>
12-12 13:43:28.364: INFO/TEST[xml reader assets](740): <product>
12-12 13:43:28.364: INFO/TEST[xml reader assets](740): Test Name
12-12 13:43:28.384: INFO/TEST[xml reader assets](740): </product>
12-12 14:47:02.745: INFO/TEST[APK-ENTRY](740): res/drawable/taobao.ico
12-12 14:47:02.775: INFO/TEST[APK-ENTRY](740): res/drawable/taobao_tabs.xml
12-12 14:47:02.795: INFO/TEST[xml reader xml](740): ����4������
12-12 14:47:02.795: INFO/TEST[APK-ENTRY](740): res/layout/main.xml
12-12 14:47:02.815: INFO/TEST[xml reader xml](740): ����L������
12-12 14:47:02.925: INFO/TEST[APK-ENTRY](740): AndroidManifest.xml
12-12 14:47:02.956: INFO/TEST[xml reader xml](740): �������������������
12-12 14:47:03.185: INFO/TEST[APK-ENTRY](740): resources.arsc
12-12 14:47:03.265: INFO/TEST[APK-ENTRY](740): res/drawable-hdpi/icon.png
12-12 14:47:03.285: INFO/TEST[APK-ENTRY](740): res/drawable-ldpi/icon.png
12-12 14:47:03.304: INFO/TEST[APK-ENTRY](740): res/drawable-mdpi/icon.png
12-12 14:47:03.324: INFO/TEST[APK-ENTRY](740): classes.dex
12-12 14:47:03.374: INFO/TEST[APK-ENTRY](740): META-INF/MANIFEST.MF
12-12 14:47:03.394: INFO/TEST[APK-ENTRY](740): META-INF/CERT.SF
12-12 14:47:03.404: INFO/TEST[APK-ENTRY](740): META-INF/CERT.RSA

    从 LogCat 消息我们看到, 所有在 assets/ 目录下的文件都以原始文件格式存放, 因此我们能正确的显示它的内容, 但是在 res/ 目录下的文件,即便是 xml 的文本文件, 也只能显示出为乱码。当然在 Android app 下读取 assets/ 目录下文件不能使用这种方式, 而是 Android 提供的 AssetManager 来实现。

    Activity 下提供了许多获取一个 File 的方法, 我们需要知道这些 File 具体指向 Android 的哪个位置。 我们也来测试一下:

Log.i("TEST[getCacheDir()]", getCacheDir().getAbsolutePath());
		Log.i("TEST[getDatabasePath()]", getDatabasePath("test.db").getAbsolutePath());
		Log.i("TEST[getDir()]", getDir("testDir", Context.MODE_PRIVATE).getAbsolutePath());
		Log.i("TEST[getExternalCacheDir()]", getExternalCacheDir().getAbsolutePath());
		Log.i("TEST[getExternalFilesDir('')]", getExternalFilesDir("").getAbsolutePath());
		Log.i("TEST[getExternalFilesDir('test_data')]", getExternalFilesDir("test_data").getAbsolutePath());
		Log.i("TEST[getExternalFilesDir('music')]", getExternalFilesDir("music").getAbsolutePath());
		Log.i("TEST[getFilesDir()]", getFilesDir().getAbsolutePath());
		Log.i("TEST[getFileStreamPath('test_steam.file')]", getFileStreamPath("test_steam.file").getAbsolutePath());

     我们看下 LogCat 结果:

12-12 15:06:25.715: INFO/TEST[getCacheDir()](1702): /data/data/org.tangao.pettyman/cache
12-12 15:06:25.745: INFO/TEST[getDatabasePath()](1702): /data/data/org.tangao.pettyman/databases/test.db
12-12 15:06:25.755: INFO/TEST[getDir()](1702): /data/data/org.tangao.pettyman/app_testDir
12-12 15:06:25.965: INFO/TEST[getExternalCacheDir()](1702): /mnt/sdcard/Android/data/org.tangao.pettyman/cache
12-12 15:06:26.174: INFO/TEST[getExternalFilesDir('')](1702): /mnt/sdcard/Android/data/org.tangao.pettyman/files
12-12 15:06:26.250: INFO/TEST[getExternalFilesDir('test_data')](1702): /mnt/sdcard/Android/data/org.tangao.pettyman/files/test_data
12-12 15:06:26.367: INFO/TEST[getExternalFilesDir('music')](1702): /mnt/sdcard/Android/data/org.tangao.pettyman/files/music
12-12 15:06:26.384: INFO/TEST[getFilesDir()](1702): /data/data/org.tangao.pettyman/files
12-12 15:06:26.424: INFO/TEST[getFileStreamPath('test_steam.file')](1702): /data/data/org.tangao.pettyman/files/test_steam.file

    除了以上的资源文件外, Android 还有 database 文件的支持, 而 database 文件是在相应 SQLiteOpenHelper 实现的 getReadableDatabase() 或者 getWritableDatabase() 方法在第一次被调用的时候创建的,然后调用 SQLiteOpenHelper 的 onCreate(SQLiteDatabase db) 方法, 如果 Database 的版本有变化, 不管是版本号变高还是变低都会调用 onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) 方法。

相关推荐