A pk 文件下的资源文件都存放在以下两个目录下: 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) 方法。