博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Android - 文件读写操作 总结
阅读量:5951 次
发布时间:2019-06-19

本文共 8003 字,大约阅读时间需要 26 分钟。

在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中对资源文件的读取、数据区文件的读取、SD卡文件的读取及RandomAccessFile的方式和方法进行了整理。供参考。

一、资源文件的读取:

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

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

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

[java]
  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]
  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]
  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.     }     

转载时请注明出处:

总结:

1、apk中有两种资源文件,使用两种不同的方式进行打开使用。

raw使用InputStream in = getResources().openRawResource(R.raw.test);
asset使用InputStream in = getResources().getAssets().open(fileName);

这些数据只能读取,不能写入。更重要的是该目录下的文件大小不能超过1M。

同时,需要注意的是,在使用InputStream的时候需要在函数名称后加上throws IOException。

2、SD卡中的文件使用FileInputStream和FileOutputStream进行文件的操作。

3、存放在数据区(/data/data/..)的文件只能使用openFileOutput和openFileInput进行操作。
注意不能使用FileInputStream和FileOutputStream进行文件的操作。
4、RandomAccessFile类仅限于文件的操作,不能访问其他IO设备。它可以跳转到文件的任意位置,从当前位置开始读写。

5、InputStream和FileInputStream都可以使用skip和read(buffre,offset,length)函数来实现文件的随机读取。

你可能感兴趣的文章
水滴状的自己定义视图,让您摆脱单调的Dialog
查看>>
名不正则言不顺,言不顺则事不成
查看>>
zookeeper 四字命令的使用
查看>>
IntelliJ中的main函数、for循环、System.out.println()快捷键
查看>>
Android OpenSL ES 开发:Android OpenSL 录制 PCM 音频数据
查看>>
linux每日命令(13):more命令
查看>>
WebSocket学习与使用
查看>>
Logistic Regression
查看>>
防止sql注入:替换危险字符
查看>>
.net 获取 存储过程的输出参数
查看>>
eclipse+webservice开发实例
查看>>
Eclipse导入Android项目的方法(转)
查看>>
真武汤
查看>>
live555直播
查看>>
转载:JS进度条
查看>>
面向切面编程(AOP)的理解
查看>>
【转】远程桌面 剪切板复制文件失效解决方法
查看>>
SQL Server差异备份的备份/还原原理
查看>>
java 内部类
查看>>
leetcode - Search in Rotated Sorted Array II
查看>>