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) 方法。