Android - 文件读写操作 总结

简介:
复制代码
在Android开发中,有两种处理资源文件的方式。其一,是将所有资源文件以及JNI程序放置于一个单独的资源包。使用到他们时,使用文件方式读取。或者直接使用C++层代码读取。 其二,则是将资源文件加入到APK内部。使用各种不同的办法去得到其内容。
方法一:适合于移植较大的C++程序时使用,因为C++代码数量众多,不太可能修改为JAVA代码。所以将其与资源文件以一定方式存放,并让他们自称体系是个好办法。但这造成软件的发布必须以APK+资源包的方式发布。
方法二:则比较适合代码量不是非常大,且资源数量不是特别多的情况。此时,用户安装APK后,不用再费力copy资源包。方便发布。
这次主要介绍的是第二种方式,资源加入APK方式。
0.Android资源介绍:
Android应用程序开发时,大家通常都会用到以下资源:
res/drawable: 通常用来存放图片资源。如logo等。
res/layout:布局文件。
res/values:存放String,如程序名等。
但Android其实还可以使用其它类型资源。今天介绍3种如下:
res/xml: 存放xml文件,与之前所说的资源类似,存放在其中的资源文件会被编译为二进制数据而存入安装包内。通过R类读取xml文件。
res/raw: 存放文件。此目录下文件与之前的资源不同,他们不会被编译为二进制文件.而是以文件形式存放起来。通过R类读取。
assets: 可以在此创建子目录并存放不同文件。不会被编译入二进制,而是以目录/文件存放。通过文件名读取。
复制代码

Android中的文件放在不同位置,它们的读取方式也有一些不同。

本文对android中对资源文件的读取、数据区文件的读取、SD卡文件的读取及RandomAccessFile的方式和方法进行了整理。供参考。


一、资源文件的读取:

      1) 从resource的raw中读取文件数据:

[java]  view plain  copy
  1. String res = "";   
  2. try{   
  3.    
  4.     //得到资源中的Raw数据流  
  5.     InputStream in = getResources().openRawResource(R.raw.test);   
  6.   
  7.     //得到数据的大小  
  8.     int length = in.available();         
  9.   
  10.     byte [] buffer = new byte[length];          
  11.   
  12.     //读取数据  
  13.     in.read(buffer);           
  14.   
  15.     //依test.txt的编码类型选择合适的编码,如果不调整会乱码   
  16.     res = EncodingUtils.getString(buffer, "BIG5");   
  17.       
  18.     //关闭      
  19.     in.close();              
  20.   
  21.    }catch(Exception e){   
  22.       e.printStackTrace();           
  23.    }   


 2) 从resource的asset中读取文件数据

[java]  view plain  copy
  1. String fileName = "test.txt"; //文件名字   
  2. String res="";   
  3. try{   
  4.   
  5.    //得到资源中的asset数据流  
  6.    InputStream in = getResources().getAssets().open(fileName);   
  7.   
  8.    int length = in.available();           
  9.    byte [] buffer = new byte[length];          
  10.   
  11.    in.read(buffer);              
  12.    in.close();  
  13.    res = EncodingUtils.getString(buffer, "UTF-8");       
  14.   
  15.   }catch(Exception e){   
  16.   
  17.       e.printStackTrace();           
  18.   
  19.    }   


二、读写/data/data/<应用程序名>目录上的文件:

[java]  view plain  copy
  1. //写数据  
  2. public void writeFile(String fileName,String writestr) throws IOException{   
  3.   try{   
  4.   
  5.         FileOutputStream fout =openFileOutput(fileName, MODE_PRIVATE);   
  6.   
  7.         byte [] bytes = writestr.getBytes();   
  8.   
  9.         fout.write(bytes);   
  10.   
  11.         fout.close();   
  12.       }   
  13.   
  14.         catch(Exception e){   
  15.         e.printStackTrace();   
  16.        }   
  17. }   
  18.   
  19. //读数据  
  20. public String readFile(String fileName) throws IOException{   
  21.   String res="";   
  22.   try{   
  23.          FileInputStream fin = openFileInput(fileName);   
  24.          int length = fin.available();   
  25.          byte [] buffer = new byte[length];   
  26.          fin.read(buffer);       
  27.          res = EncodingUtils.getString(buffer, "UTF-8");   
  28.          fin.close();       
  29.      }   
  30.      catch(Exception e){   
  31.          e.printStackTrace();   
  32.      }   
  33.      return res;   
  34.   
  35. }     


三、读写SD卡中的文件。也就是/mnt/sdcard/目录下面的文件 :

[java]  view plain  copy
  1. //写数据到SD中的文件  
  2. public void writeFileSdcardFile(String fileName,String write_str) throws IOException{   
  3.  try{   
  4.   
  5.        FileOutputStream fout = new FileOutputStream(fileName);   
  6.        byte [] bytes = write_str.getBytes();   
  7.   
  8.        fout.write(bytes);   
  9.        fout.close();   
  10.      }  
  11.   
  12.       catch(Exception e){   
  13.         e.printStackTrace();   
  14.        }   
  15.    }   
  16.   
  17.     
  18. //读SD中的文件  
  19. public String readFileSdcardFile(String fileName) throws IOException{   
  20.   String res="";   
  21.   try{   
  22.          FileInputStream fin = new FileInputStream(fileName);   
  23.   
  24.          int length = fin.available();   
  25.   
  26.          byte [] buffer = new byte[length];   
  27.          fin.read(buffer);       
  28.   
  29.          res = EncodingUtils.getString(buffer, "UTF-8");   
  30.   
  31.          fin.close();       
  32.         }   
  33.   
  34.         catch(Exception e){   
  35.          e.printStackTrace();   
  36.         }   
  37.         return res;   
  38. }   


四、使用File类进行文件的读写:

[java]  view plain  copy
  1. //读文件  
  2. public String readSDFile(String fileName) throws IOException {    
  3.   
  4.         File file = new File(fileName);    
  5.   
  6.         FileInputStream fis = new FileInputStream(file);    
  7.   
  8.         int length = fis.available();   
  9.   
  10.          byte [] buffer = new byte[length];   
  11.          fis.read(buffer);       
  12.   
  13.          res = EncodingUtils.getString(buffer, "UTF-8");   
  14.   
  15.          fis.close();       
  16.          return res;    
  17. }    
  18.   
  19. //写文件  
  20. public void writeSDFile(String fileName, String write_str) throws IOException{    
  21.   
  22.         File file = new File(fileName);    
  23.   
  24.         FileOutputStream fos = new FileOutputStream(file);    
  25.   
  26.         byte [] bytes = write_str.getBytes();   
  27.   
  28.         fos.write(bytes);   
  29.   
  30.         fos.close();   
  31. }   


五、另外,File类还有下面一些常用的操作:

[java]  view plain  copy
  1. String Name = File.getName();  //获得文件或文件夹的名称:  
  2. String parentPath = File.getParent();  //获得文件或文件夹的父目录  
  3. String path = File.getAbsoultePath();//绝对路经  
  4. String path = File.getPath();//相对路经   
  5. File.createNewFile();//建立文件    
  6. File.mkDir(); //建立文件夹    
  7. File.isDirectory(); //判断是文件或文件夹  
  8. File[] files = File.listFiles();  //列出文件夹下的所有文件和文件夹名  
  9. File.renameTo(dest);  //修改文件夹和文件名  
  10. File.delete();  //删除文件夹或文件  


六、使用RandomAccessFile进行文件的读写:

RandomAccessFile的使用方法比较灵活,功能也比较多,可以使用类似seek的方式可以跳转到文件的任意位置,从文件指示器当前位置开始读写。
它有两种构造方法
new RandomAccessFile(f,"rw");//读写方式
new RandomAccessFile(f,"r");//只读方式
使用事例:

[java]  view plain  copy
  1. /*  
  2.  * 程序功能:演示了RandomAccessFile类的操作,同时实现了一个文件复制操作。  
  3.  */    
  4.     
  5. import java.io.*;    
  6.     
  7. public class RandomAccessFileDemo {    
  8.  public static void main(String[] args) throws Exception {    
  9.   RandomAccessFile file = new RandomAccessFile("file", "rw");    
  10.   // 以下向file文件中写数据    
  11.   file.writeInt(20);// 占4个字节    
  12.   file.writeDouble(8.236598);// 占8个字节    
  13.   file.writeUTF("这是一个UTF字符串");// 这个长度写在当前文件指针的前两个字节处,可用readShort()读取    
  14.   file.writeBoolean(true);// 占1个字节    
  15.   file.writeShort(395);// 占2个字节    
  16.   file.writeLong(2325451l);// 占8个字节    
  17.   file.writeUTF("又是一个UTF字符串");    
  18.   file.writeFloat(35.5f);// 占4个字节    
  19.   file.writeChar('a');// 占2个字节    
  20.     
  21.   file.seek(0);// 把文件指针位置设置到文件起始处    
  22.     
  23.   // 以下从file文件中读数据,要注意文件指针的位置    
  24.   System.out.println("——————从file文件指定位置读数据——————");    
  25.   System.out.println(file.readInt());    
  26.   System.out.println(file.readDouble());    
  27.   System.out.println(file.readUTF());    
  28.     
  29.   file.skipBytes(3);// 将文件指针跳过3个字节,本例中即跳过了一个boolean值和short值。    
  30.   System.out.println(file.readLong());    
  31.     
  32.   file.skipBytes(file.readShort()); // 跳过文件中“又是一个UTF字符串”所占字节,注意readShort()方法会移动文件指针,所以不用加2。    
  33.   System.out.println(file.readFloat());    
  34.       
  35.   //以下演示文件复制操作    
  36.   System.out.println("——————文件复制(从file到fileCopy)——————");    
  37.   file.seek(0);    
  38.   RandomAccessFile fileCopy=new RandomAccessFile("fileCopy","rw");    
  39.   int len=(int)file.length();//取得文件长度(字节数)    
  40.   byte[] b=new byte[len];    
  41.   file.readFully(b);    
  42.   fileCopy.write(b);    
  43.   System.out.println("复制完成!");    
  44.  }    
  45. }    


七、读取资源文件时能否实现类似于seek的方式可以跳转到文件的任意位置,从指定的位置开始读取指定的字节数呢?

答案是可以的。

在FileInputStream和InputStream中都有下面的函数:

[java]  view plain  copy
  1. public long skip (long byteCount); //从数据流中跳过n个字节  
  2. public int read (byte[] buffer, int offset, int length); //从数据流中读取length数据存在buffer的offset开始的位置。offset是相对于buffer的开始位置的,不是数据流。  


可以使用这两个函数来实现类似于seek的操作,请看下面的测试代码:

[java]  view plain  copy
  1. //其中read_raw是一个txt文件,存放在raw目录下。  
  2. //read_raw.txt文件的内容是:"ABCDEFGHIJKLMNOPQRST"  
  3. public String getRawString() throws IOException {  
  4.       
  5.     String str = null;  
  6.       
  7.     InputStream in = getResources().openRawResource(R.raw.read_raw);  
  8.       
  9.     int length = in.available();  
  10.     byte[] buffer = new byte[length];  
  11.       
  12.     in.skip(2); //跳过两个字节  
  13.     in.read(buffer,0,3); //读三个字节  
  14.       
  15.     in.skip(3); //跳过三个字节  
  16.     in.read(buffer,0,3); //读三个字节  
  17.       
  18.     //最后str="IJK"  
  19.     str = EncodingUtils.getString(buffer, "BIG5");  
  20.       
  21.       
  22.     in.close();  
  23.       
  24.     return str;  
  25. }  


从上面的实例可以看出skip函数有点类似于C语言中的seek操作,但它们之间有些不同。

需要注意的是:

1、skip函数始终是从当前位置开始跳的。在实际应用当中还要再判断一下该函数的返回值。

2、read函数也始终是当前位置开始读的。

3、另外,还可以使用reset函数将文件的当前位置重置为0,也就是文件的开始位置。


如何得到文件的当前位置?

我没有找到相关的函数和方法,不知道怎么样才能得到文件的当前位置,貌似它也并不是太重要。


八、如何从FileInputStream中得到InputStream?

[java]  view plain  copy
  1. public String readFileData(String fileName) throws IOException{   
  2.   String res="";   
  3.   try{   
  4.          FileInputStream fin = new FileInputStream(fileName);   
  5.      InputStream in = new BufferedInputStream(fin);  
  6.   
  7.          ...  
  8.   
  9.       }  
  10.       catch(Exception e){   
  11.          e.printStackTrace();   
  12.       }  
  13.   
  14. }  

 

九、APK资源文件的大小不能超过1M,如果超过了怎么办?我们可以将这个数据再复制到data目录下,然后再使用。复制数据的代码如下:

[java]  view plain  copy
  1. public boolean assetsCopyData(String strAssetsFilePath, String strDesFilePath){  
  2.        boolean bIsSuc = true;  
  3.        InputStream inputStream = null;  
  4.        OutputStream outputStream = null;  
  5.          
  6.        File file = new File(strDesFilePath);  
  7.        if (!file.exists()){  
  8.            try {  
  9.               file.createNewFile();  
  10.               Runtime.getRuntime().exec("chmod 766 " + file);  
  11.            } catch (IOException e) {  
  12.               bIsSuc = false;  
  13.            }  
  14.              
  15.        }else{//存在  
  16.            return true;  
  17.        }  
  18.          
  19.        try {  
  20.            inputStream = getAssets().open(strAssetsFilePath);  
  21.            outputStream = new FileOutputStream(file);  
  22.              
  23.            int nLen = 0 ;  
  24.              
  25.            byte[] buff = new byte[1024*1];  
  26.            while((nLen = inputStream.read(buff)) > 0){  
  27.               outputStream.write(buff, 0, nLen);  
  28.            }  
  29.              
  30.            //完成  
  31.        } catch (IOException e) {  
  32.            bIsSuc = false;  
  33.        }finally{  
  34.            try {  
  35.               if (outputStream != null){  
  36.                   outputStream.close();  
  37.               }  
  38.                 
  39.               if (inputStream != null){  
  40.                   inputStream.close();  
  41.               }  
  42.            } catch (IOException e) {  
  43.               bIsSuc = false;  
  44.            }  
  45.              
  46.        }  
  47.          
  48.        return bIsSuc;  
  49.     }     




本文转自农夫山泉别墅博客园博客,原文链接:http://www.cnblogs.com/yaowen/p/5443408.html,如需转载请自行联系原作者

相关文章
|
2月前
|
ARouter Android开发
Android不同module布局文件重名被覆盖
Android不同module布局文件重名被覆盖
|
4月前
|
Java Android开发 C++
Android Studio JNI 使用模板:c/cpp源文件的集成编译,快速上手
本文提供了一个Android Studio中JNI使用的模板,包括创建C/C++源文件、编辑CMakeLists.txt、编写JNI接口代码、配置build.gradle以及编译生成.so库的详细步骤,以帮助开发者快速上手Android平台的JNI开发和编译过程。
290 1
|
2月前
|
ARouter Android开发
Android不同module布局文件重名被覆盖
Android不同module布局文件重名被覆盖
127 0
|
4月前
|
开发工具 git 索引
repo sync 更新源码 android-12.0.0_r34, fatal: 不能重置索引文件至版本 ‘v2.27^0‘。
本文描述了在更新AOSP 12源码时遇到的repo同步错误,并提供了通过手动git pull更新repo工具来解决这一问题的方法。
135 1
|
4月前
|
存储 监控 数据库
Android经典实战之OkDownload的文件分段下载及合成原理
本文介绍了 OkDownload,一个高效的 Android 下载引擎,支持多线程下载、断点续传等功能。文章详细描述了文件分段下载及合成原理,包括任务创建、断点续传、并行下载等步骤,并展示了如何通过多种机制保证下载的稳定性和完整性。
123 0
|
6月前
|
Java 开发工具 Android开发
详细解读Android开发DNK开发将.c文件打包成os
详细解读Android开发DNK开发将.c文件打包成os
33 0
|
6月前
|
Android开发
Android Gradle开发—脚本实现自动打包后复制一份APK文件,并修改APK名称,到指定目录作备份
Android Gradle开发—脚本实现自动打包后复制一份APK文件,并修改APK名称,到指定目录作备份
304 0
|
10天前
|
开发框架 前端开发 Android开发
安卓与iOS开发中的跨平台策略
在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
|
15天前
|
搜索推荐 Android开发 开发者
探索安卓开发中的自定义视图:打造个性化UI组件
【10月更文挑战第39天】在安卓开发的世界中,自定义视图是实现独特界面设计的关键。本文将引导你理解自定义视图的概念、创建流程,以及如何通过它们增强应用的用户体验。我们将从基础出发,逐步深入,最终让你能够自信地设计和实现专属的UI组件。
|
1天前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。