http://blog.zol.com.cn/4689/article_4688688.html
*res/raw和assets的相同點:
1.兩者目錄下的文件在打包后會原封不動的保存在apk包中,不會被編譯成二進制。
*res/raw和assets的不同點:
1.res/raw中的文件會被映射到R.java文件中,訪問的時候直接使用資源ID即R.id.filename;assets文件夾下的文件不會被映射到R.java中,訪問的時候需要AssetManager類。
2.res/raw不可以有目錄結構,而assets則可以有目錄結構,也就是assets目錄下可以再建立文件夾
*讀取文件資源:
1.讀取res/raw下的文件資源,通過以下方式獲取輸入流來進行寫操作
view plain copy to clipboard print ?
InputStream is = getResources().openRawResource(R.id.filename);
2.讀取assets下的文件資源,通過以下方式獲取輸入流來進行寫操作
view plain copy to clipboard print ?
AssetManager am = null ;
am = getAssets();
InputStream is = am.open("filename" );
補充一下:在未知目錄下有哪些文件,該去和獲取這些文件的名稱并把文件拷貝到目標目錄中呢? (用于內置文件但不知道文件名稱,需要篩選出想要的文件然后拷貝到目標目錄中,推薦內置在assets文件夾中)
1.res/raw目錄:
通過反射的方式得到R.java里面raw內部類里面所有的資源ID的名稱,然后通過名稱獲取資源ID的值來讀取我們想要的文件。(這個方法我沒試過,有用過的同學麻煩發一段代碼看看)。
2.assets目錄:
getAssets().list("");來獲取assets目錄下所有文件夾和文件的名稱,再通過這些名稱再讀取我們想要的文件http://www.myexception.cn/。
在讀取這兩個資源文件夾中的文件時會有一定的限制,即單個文件大小不能超過1M ,如果讀取超過1M的文件會報 "Data exceeds UNCOMPRESS_DATA_MAX (1314625 vs 1048576)" 的IOException。
解決方法如下(假設我們現在要把一個超過1M的文件在程序啟動時拷貝到sdcard中)
1.先把需要拷貝的大文件分割成若干個大小小于1M的小文件(事先寫個程序來分隔或者使用一些工具,我這里直接寫了個程序),把這些 小文件放在assets文件夾中; 2.在程序啟動時我們獲取這些小文件的文件名,當然我們得事先規定小文件的命名方式方便我們來獲取這些文件名;
3.通過獲得的小文件名分別建立輸入流來合并成一個大文件,并拷貝到sdcard中。
下面是解決方法中需要用到的一些代碼,僅供參考,不妨自己寫。 分割大文件的方法:
Java代碼
/**
* 指定大小分割文件
*
* @author zuolongsnail
*/
public static void main(String[] args) throws Exception {
// 大文件放置的路徑
String path = "D:/";
// 大文件的文件名稱
String base = "demo";
String ext = ".db";
// 以每個小文件1024*1024字節即1M的標準來分割
int split = 1024 * 1024;
byte[] buf = new byte[1024];
int num = 1;
// 建立輸入流
File inFile = new File(path + base + ext);
FileInputStream fis = new FileInputStream(inFile);
while (true) {
// 以"demo"+num+".db"方式來命名小文件即分割后為demo1.db,org.htmlparser.util.ParserExceptionSQLServer 2000 Driver for JDBCdemo2.db,java.lang.IllegalArgumentException: bad date header。找不到類異常。。。。。
FileOutputStream fos = new FileOutputStream(new File(path + base
+ num + ext));
for (int i = 0; i < split / buf.length; i++) {
int read = fis.read(buf);
fos.write(buf, 0, read);
// 判斷大文件讀取是否結束
if (read < buf.length) {
fis.close();
fos.close();
return;
}
}
fos.close();
num++;
}
}
獲取輸入流來合并文件,java異常處理機制我們這里以assets文件夾下的文件為例,raw文件夾下如何獲取輸入流請參考之前的那篇博文.
Java代碼
/**
* 合并文件
*
* @param c
* @param partFileList 小文件名集合
* @param dst 目標文件路徑
* @throws IOException
*
* @author zuolongsnail
*/
private void mergeApkFile(Context c, ArrayList partFileList, String dst) throws IOException {
if (!new File(dst).exists()) {
OutputStream out = new FileOutputStream(dst);
byte[] buffer = new byte[1024];
InputStream in;
int readLen = 0;
for(int i=0;i // 獲得輸入流
in = c.getAssets().open(partFileList.get(i));
while((readLen = in.read(buffer)) != -1){
out.write(buffer, 0, readLen);
}
out.flush();
in.close();
}
// 把所有小文件都進行寫操作后才關閉輸出流,這樣就會合并為一個文件了
out.close();
}
}