`
dfeixtay
  • 浏览: 57534 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

apk携带资源之raw & assets

阅读更多
wo-07 10-14 dfeixtay or

/res/raw & /assets 携带资源

    在android开发中,总有一些资源是想要随着安装包apk携带的,这些资源如数据库文件,文本,mp3等等。最早的做法是,在prebulid过程中,修改mk文件,将指定文件提前拷贝到系统某一文件夹目录下。这样的做法,既不科学也不美观还不安全,处于对代码的洁癖,我终于在不太忙的时候把这些资源文件带在自己的apk下。

    之前说过没,由于提升速度,从文本解析调整为数据库查询。在携带文件时,这两种文件本没有不同,只是这里要讲一点inputstream的时候,二进制文件的db文件,和纯文本文件txt的方式可以不同,原因好像是,纯文本是按照unicode编码的,是16位16位的传的,二进制文件是8位传的。又想到之前ftp传输的时候,也是写的二进制传输。

    言归正传,无论是raw文件夹还是assets文件夹,都是在生成apk的时候不编译而直接携带在apk的压缩包中的,这可以打开apk检验。这想必也是raw的得名。

    提取的方法都是从inputstream转,转成什么形式的,要看对inputstream的操作,下篇也许会讲。具体的过程是:

private void getRaw(){
        File target = new File("/data/data/com.android.");

//        InputStream in = this.getResources().openRawResource(R.raw.weather_db);
//        try {
//        } catch (IOException e1) {
//            // TODO Auto-generated catch block
//            e1.printStackTrace();
//        }
        InputStream in = null;
        OutputStream out = null;    
          
        BufferedInputStream bin = null;
        BufferedOutputStream bout = null;
        try{
            //////////////////////////////////////////////////////////////////////
            int xx = 1;//R.raw.parse_weather_db_aa;
            xx+=1;
            try{
                in = getResources().openRawResource(xx);
            }catch(Exception e){
                e.printStackTrace();
            }

            out = new FileOutputStream(target);
            bin = new BufferedInputStream(in);
            bout = new BufferedOutputStream(out);

            byte[] b = new byte[1024];
            int len = bin.read(b);

            while (len != -1){
                bout.write(b, 0, len);
                len = bin.read(b);
            }
//            exec("chmod 777 "+target.getAbsolutePath());
        }
        catch (FileNotFoundException e){

            e.printStackTrace();
        }
        catch (IOException e)
        {

            e.printStackTrace();
        }
        finally
        {
            try{
                if (bin != null){
                    bin.close();
                }
                if (bout != null){
                    bout.close();
                }
            }
            catch (IOException e){
                e.printStackTrace();
            }
        }
    }
    
    private void getAsset(){
//        if(WeatherWidget.loadRunning(this, "FirstSetup")){
        if(true){

            WeatherWidget.saveRunning(this, false, "FirstSetup");
            
            File databs = new File("/data/data/com.android.");
            if(!databs.exists()){
                
                databs.mkdir();
            }
            
            InputStream in = null;
            OutputStream out = null;    
              
            BufferedInputStream bin = null;
            BufferedOutputStream bout = null;
            
            AssetManager am = getAssets();
            String asName = "parse_weather_db_aa";
            

            while(true){
                
                try{
                    
                    in = am.open(asName);
                    out = new FileOutputStream(databs.getPath()+"/weather_db");
                    asName = nextAsset(asName);
                    
                    
                    bin = new BufferedInputStream(in);
                    bout = new BufferedOutputStream(out);
                    
                    byte[] b = new byte[8192];
                    int len = bin.read();
                    while (len != -1){
                        bout.write(b, 0, len);
                        len = bin.read(b);
                    }  
                    continue;
                }
                catch (FileNotFoundException e){
                    e.printStackTrace();
                    try{
                        if (bin != null){
                            bin.close();
                        }
                        if (bout != null){ 
                            bout.close();
                        }
                        break;
                    }
                    catch (IOException ee){
                        ee.printStackTrace();
                    }
                    break;
                }
                catch (IOException e){
                    e.printStackTrace();
                    try{
                        if (bin != null){
                            bin.close();
                        }
                        if (bout != null){
                            bout.close();
                        }
                        break;
                    }
                    catch (IOException eee){
                        eee.printStackTrace();
                    }
                    break;
                }
            }
        }
    }



    代码比较乱,有些东西讲一遍不如需要的时候直接回来看一眼代码,去伪存真吧。

    最后,讲raw & assets资源比较重要的一点,就是文件的大小限制,单个文件的大小不可以大于1M,有人说这是android的bug,呵呵,解决的方法是:

    将文件split为小于1M的文件,在读取的时候outputstream不要close,而是合并写这些文件,最后就得到原始文件。分割可以用ubuntu命令,

  • split -b[byte] 512k [source] [prefix]


也可以用windows下文件分割器,或者直接在android中制作,代码inputstream的时候限制大小,分割存储,我没有实践过,只是看到有人是这么写的。

以下是原文,需要时搜索关键字,网址没了。

引用

汉语词典开发-assets,raw的InputStream数据流操作(文件分割合并)
文章分类:移动开发
 
文件移动
一. 在应用中由于种种原因需要将一些外部文件放在 assets 或者 raw 文件夹内,以便进一步使用。这两个文件夹有以下的区别和联系:
1. 都是以数据流的形式进行读取,从而导致 Java 中其他的一些读取方式不能很好的作用在这些文件上面,例如 RandomAccessFile 、 FileReader 等之类的类。如果由于需要需要使用基于文件的类,则可以根据数据流创建创建临时文件( File.createTempFile )当做一个折中的办法。这是一种方法,下面将会介绍另一种方法
2. raw 文件夹中的文件不能包含有目录结构并且每个文件会映射到一个 id ,而 assets 文件夹可以有目录结构。对于对文件名敏感的程序则使用 raw 进行外部文件存储较为方便,而对于较依赖目录结构的文件则使用 assets 存储
3. 这两个文件夹中的文件都不能太大,官方数据是小于 1M 。这点需要时刻记住,因为产生的问题十分隐蔽,在程序中可以找到该文件也可以产生 inputStream 但是在读取时会抛出 IOException 异常。这中大文件需要先分割在进行读取
4. 这两个文件夹对文件名称大小写敏感,命名是尽量用小写,并且在分割合并后也要注意文件名称,否则程序会认为它们是不同的文件,但是在创建时会覆盖掉先前的文件(这点太隐蔽了, ~~~~(>_<)~~~~ )
  
以下是分割数据的代码:
 
Java代码
1. public static void CutFilesInSizeParts(InputStream fis,   
2.         String OutputFileName, int MaxPartSize) {   
3.     try {   
4.            
5.         int TotalLength = fis.available();   
6.         byte[] buffer = new byte[TotalLength + 1];   
7.         int len = fis.read(buffer);   
8.   
9.         int nbPart = len / MaxPartSize + 1;   
10.         int CurPos = 0;   
11.   
12.         for (int i = 0; i < nbPart; i++) {   
13.             int PartLen = MaxPartSize;   
14.             if (CurPos + PartLen >= len)   
15.                 PartLen = len - CurPos;   
16.             String outRealFileName = OutputFileName + (i + 1);   
17.             FileOutputStream fos = new FileOutputStream(outRealFileName);   
18.             fos.write(buffer, CurPos, PartLen);   
19.             CurPos += PartLen;   
20.         }   
21.     } catch (IOException e) {   
22.         e.printStackTrace();   
23.     }   
24. }  
public static void CutFilesInSizeParts(InputStream fis,
String OutputFileName, int MaxPartSize) {
try {

int TotalLength = fis.available();
byte[] buffer = new byte[TotalLength + 1];
int len = fis.read(buffer);

int nbPart = len / MaxPartSize + 1;
int CurPos = 0;

for (int i = 0; i < nbPart; i++) {
int PartLen = MaxPartSize;
if (CurPos + PartLen >= len)
PartLen = len - CurPos;
String outRealFileName = OutputFileName + (i + 1);
FileOutputStream fos = new FileOutputStream(outRealFileName);
fos.write(buffer, CurPos, PartLen);
CurPos += PartLen;
}
} catch (IOException e) {
e.printStackTrace();
}
}
 
 
二,正是以上两个文件夹只能产生 InputStream 数据流,当程序有别的需要时会显的无能为力。例如在外面创建了一个 sqlite 的 db 文件,该文件需要内置到 apk 中只能放入这两个文件夹中,可是在使用时可要根据需要放置到 sdcard 中或者 database 中去。因此需要对读取文件并在相应位置生成目的文件,按照自己需要的方式进行读取,这也提供了第二种方法。这一步需要注意一下几点:
 
1. 文件的权限,否则可能会被别的应用使用
2. 若是要对分割后的文件进行合并,则要注意文件顺序
 
这里附上合并数据并拷贝的代码
Java代码
1. //合并并拷贝数据   
2.     public static void CreateFromRawDbFiles(File[] filelist,   
3.             FileOutputStream Fos) {   
4.   
5.         try {   
6.             for (File file : filelist) {   
7.                 InputStream inputFile = new FileInputStream(file);   
8.                 int TotalLength = 0;   
9.                 try {   
10.                     TotalLength = inputFile.available();   
11.                 } catch (IOException e) {   
12.                 }   
13.                 // Reading and writing the file Method 1 :   
14.                 byte[] buffer = new byte[TotalLength];   
15.                 int len = 0;   
16.                 try {   
17.                     len = inputFile.read(buffer);   
18.                 } catch (IOException e) {   
19.                 }   
20.                 Fos.write(buffer,0,len);   
21.                 inputFile.close();   
22.             }   
23.             Fos.close();   
24.         } catch (IOException e) {   
25.         }   
26.     }  

分享到:
评论
2 楼 dfeixtay 2010-12-07  
yankunren 写道
哥们 ,raw目录下的东西有没有办法让它编译啊


raw和asset下东西,android不编译。
编译你带到xml下吧,具体不了解。为啥编译,编译啥
1 楼 yankunren 2010-11-30  
哥们 ,raw目录下的东西有没有办法让它编译啊

相关推荐

    androidgui程序设计复习题.docx

    8. **res/raw目录**:res/raw目录下的文件会被原封不动地打包到APK中,但当应用运行时,它们会被转换为二进制资源,可以通过Resources类的openRawResource()方法访问。 9. **Activity组织方式**:Android中的...

    【大数据课设】p105出租车数据可视化分析-大数据-实训大作业.zip

    项目资源包含:可运行源码+数据集+文档 python + numpy, pandas, matplotlib, pyecharts, wordcloud 适用人群:学习不同技术领域的小白或进阶学习者;可作为课程设计、大作业、工程实训或初期项目立项。 数据来源:数据集taxis.csv从网络下载 数据清洗:异常值与缺失值的处理:有一些数据distance(乘车距离)为零而且上下车地点为空,还有些一些数据的payment(支付方式)为空。 数据预处理:将列名更改成中文 标准化与归一化: 数据分析: 数据可视化:

    TypeScript 入门教程

    TypeScript 入门教程

    人脸识别_课堂考勤_OpenCV_服务端系统_1741777828.zip

    人脸识别项目实战

    历届电赛试题及综合测评(真题+模拟题)

    本资源汇总了 历届全国电子设计竞赛(电赛)真题+模拟题,涵盖 电路设计、嵌入式系统、信号处理、自动控制等核心考点,并提供详细解析及综合测评,帮助参赛者高效备赛、查漏补缺、提升实战能力。 适用人群: 适合 准备参加电子设计竞赛的大学生、电赛爱好者、电子信息类相关专业的学生,以及希望提高电子设计和电路分析能力的工程师。 能学到什么: 电赛考察重点:熟悉往届竞赛的命题方向及考核重点。 电路设计与仿真:提升模拟电路、数字电路、单片机等核心技能。 问题分析与解决能力:通过综合测评找到薄弱点并针对性提升。 实战经验:掌握竞赛策略,提高应试效率和设计能力。 阅读建议: 建议先 通读真题,了解题型与解题思路,然后 结合模拟题实战演练,查找不足并通过测评强化练习,逐步提升竞赛能力。

    2024人工智能如何塑造未来产业:AI对各行业组织带来的的变革研究研究报告.pdf

    2024人工智能如何塑造未来产业:AI对各行业组织带来的的变革研究研究报告.pdf

    人脸识别_Golang_SDK_命令行登录_微信小程序应用_1741772240.zip

    人脸识别项目源码实战

    Vulkan原理与实战课程

    给大家分享一套课程——Vulkan原理与实战课程

    SiriYXR_Sokoban11_1741860914.zip

    c语言学习

    海豚鲸鱼数据集 5435张图 正确识别率可达92.6% 可识别:海豚 虎鲸 蜥蜴 海豹 鲨鱼 龟 支持yolov8格式标注

    海豚鲸鱼数据集 5435张图 正确识别率可达92.6% 可识别:海豚 虎鲸 蜥蜴 海豹 鲨鱼 龟 支持yolov8格式标注

    答谢中书书教学设计.docx

    答谢中书书教学设计.docx

    人脸识别_环境搭建_dlib_face_recognitio_1741771308.zip

    人脸识别项目源码实战

    网络技术_Web服务器_C语言_学习交流版_1741863251.zip

    c语言学习

    安卓开发_Gradle配置_React_Native_Meg_1741777287.zip

    人脸识别项目源码实战

    人工智能_深度学习_图像识别_UI界面_项目展示.zip

    人脸识别项目实战

    基于Springboot框架的美发门店管理系统的设计与实现(Java项目编程实战+完整源码+毕设文档+sql文件+学习练手好项目).zip

    本美发门店管理系统有管理员和用户两个角色。用户功能有项目预定管理,产品购买管理,会员充值管理,余额查询管理。管理员功能有个人中心,用户管理,美容项目管理,项目类型管理,项目预定管理,产品库存管理,产品购买管理,产品入库管理,会员卡管理,会员充值管理,余额查询管理,产品类型管理,系统管理等。因而具有一定的实用性。 本站是一个B/S模式系统,采用SSM框架,MYSQL数据库设计开发,充分保证系统的稳定性。系统具有界面清晰、操作简单,功能齐全的特点,使得美发门店管理系统管理工作系统化、规范化。本系统的使用使管理人员从繁重的工作中解脱出来,实现无纸化办公,能够有效的提高美发门店管理系统管理效率。 关键词:美发门店管理系统;SSM框架;MYSQL数据库;Spring Boot 1系统概述 1 1.1 研究背景 1 1.2研究目的 1 1.3系统设计思想 1 2相关技术 2 2.1 MYSQL数据库 2 2.2 B/S结构 3 2.3 Spring Boot框架简介 4 3系统分析 4 3.1可行性分析 4 3.1.1技术可行性 4 3.1.2经济可行性 5 3.1.3操作可行性 5 3.2系

    Python实现基于SSA-CNN-GRU麻雀算法优化卷积门控循环单元数据分类预测的详细项目实例(含完整的程序,GUI设计和代码详解)

    内容概要:本文档介绍了基于SSA-CNN-GRU麻雀算法优化卷积门控循环单元数据分类预测的详细项目实例,重点讲述了该项目的背景、目标、挑战与解决方案、技术特点、应用领域等方面的内容。文档详细记录了从项目启动、数据预处理、算法设计(SSA优化CNN-GRU模型)、构建与评估模型到实现美观的GUI界面整个过程,并讨论了防止过拟合的技术如正则化、早停和超参数优化。另外还涵盖了项目扩展的可能性、部署和应用策略、需要注意的地方以及未来改进的方向。全文强调了模型的泛化能力和计算效率,展示了该混合算法模型在实际应用中的优越性能。 适合人群:具备一定的Python编程经验及机器学习基础知识的研究人员和技术人员;对深度学习、智能优化算法及实际应用感兴趣的学者和从业者;寻求提升数据分析和预测准确性的金融分析师、数据科学家等相关专业人士。 使用场景及目标:本文档非常适合用作学习和参考资料,以掌握如何将SSA、CNN与GRU三种先进技术结合起来进行复杂的分类和预测问题求解。具体应用场景包括但不限于以下几个方面:金融领域——股票价格预测;医疗保健领域——辅助诊断;工业制造——预防性维护;智能家居——个性化服务;以及其他涉及到时序数据分析和多模态数据处理的场合。文档既包含了理论知识又提供了完整的源代码示例,可以帮助读者理解算法原理并通过实践中加深对其的认识。 其他说明:该项目不仅仅是关于算法的设计实现,更是有关于系统的整体架构规划以及工程上的考量,比如环境准备(确保环境洁净、必要包的安装等)、数据准备、GPU配置支持等等。同时文中给出了详细的代码片段,方便开发者理解和复现实验成果。值得注意的是,虽然文中提供了一套通用解决方案,但在真实场景下还需要针对性的调整参数或修改网络结构来达到最好的性能效果。此外,对于追求更高的预测精度或解决更大规模的问题,作者建议进一步探索深度强化学习等高级技术和多任务学习策略,并且考虑使用增量学习让模型能够适应新数据而不必重新训练整个模型。最后提到安全性和隐私保护也是项目实施过程中的重要因素,要妥善保管用户的敏感信息并且做到合法合规地收集和使用数据。

    人脸识别_T形分布_Gabor变换_特征提取_增强鲁棒性_1741777397.zip

    人脸识别项目实战

    13005463562_FaceWeb_1741771809.zip

    人脸识别项目实战

    水下垃圾检测数据集,基于voc和yolo标注的两种格式

    水下垃圾检测数据集,基于voc和yolo标注的两种格式,共23,056个文件,已经划分了训练集和验证集、测试集。并且提供了真实水下的视频数据,可以用作视频推理

Global site tag (gtag.js) - Google Analytics