Android SD卡简单的文件读写操作

简介:

最近有这样的需求,把每次统计到的数据,以txt形式保存到手机SD卡或是手机内存中,遇到一些问题,记录下来。


首先如果要在程序中使用sdcard进行存储,我们必须要在AndroidManifset.xml文件进行下面的权限设置:

[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1.   <!-- SDCard中创建与删除文件权限 -->  
  2.   <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>  
  3.  <!-- 向SDCard写入数据权限 -->  
  4.  <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>  

  接着在使用SDcard进行读写的时候 会用到Environment类下面的几个静态方法  : 

    1: getDataDirectory() 获取到Android中的data数据目录(sd卡中的data文件夹)
    2:getDownloadCacheDirectory() 获取到下载的缓存目录(sd卡中的download文件夹)
    3:getExternalStorageDirectory() 获取到外部存储的目录 一般指SDcard(/storage/sdcard0)
    4:getExternalStorageState() 获取外部设置的当前状态 一般指SDcard,比较常用的应该是 MEDIA_MOUNTED(SDcard存在并且可以进行读写)还有其他的一些状态,可以在文档中进行查找。

    5:getRootDirectory()  获取到Android Root路径


好,以下是具体操作,直接看代码:

1,判断SD卡是否存在

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. /** 
  2.  * 判断SDCard是否存在 [当没有外挂SD卡时,内置ROM也被识别为存在sd卡] 
  3.  *  
  4.  * @return 
  5.  */  
  6. public static boolean isSdCardExist() {  
  7.     return Environment.getExternalStorageState().equals(  
  8.             Environment.MEDIA_MOUNTED);  
  9. }  

2,获取SD卡根目录

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. /** 
  2.  * 获取SD卡根目录路径 
  3.  *  
  4.  * @return 
  5.  */  
  6. public static String getSdCardPath() {  
  7.     boolean exist = isSdCardExist();  
  8.     String sdpath = "";  
  9.     if (exist) {  
  10.         sdpath = Environment.getExternalStorageDirectory()  
  11.                 .getAbsolutePath();  
  12.     } else {  
  13.         sdpath = "不适用";  
  14.     }  
  15.     return sdpath;  
  16.   
  17. }  

3,获取默认的文件存放路径

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. /** 
  2.  * 获取默认的文件路径 
  3.  *  
  4.  * @return 
  5.  */  
  6. public static String getDefaultFilePath() {  
  7.     String filepath = "";  
  8.     File file = new File(Environment.getExternalStorageDirectory(),  
  9.             "abc.txt");  
  10.     if (file.exists()) {  
  11.         filepath = file.getAbsolutePath();  
  12.     } else {  
  13.         filepath = "不适用";  
  14.     }  
  15.     return filepath;  
  16. }  

4-1,使用FileInputStream读取文件

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. try {  
  2. le file = new File(Environment.getExternalStorageDirectory(),  
  3. "test.txt");  
  4.     FileInputStream is = new FileInputStream(file);  
  5.     byte[] b = new byte[inputStream.available()];  
  6.     is.read(b);  
  7.     String result = new String(b);  
  8.     System.out.println("读取成功:"+result);  
  9. catch (Exception e) {  
  10.     e.printStackTrace();  
  11. }  

4-2,使用BufferReader读取文件

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. try {  
  2.     File file = new File(Environment.getExternalStorageDirectory(),  
  3.             DEFAULT_FILENAME);  
  4.     BufferedReader br = new BufferedReader(new FileReader(file));  
  5.     String readline = "";  
  6.     StringBuffer sb = new StringBuffer();  
  7.     while ((readline = br.readLine()) != null) {  
  8.         System.out.println("readline:" + readline);  
  9.         sb.append(readline);  
  10.     }  
  11.     br.close();  
  12.     System.out.println("读取成功:" + sb.toString());  
  13. catch (Exception e) {  
  14.     e.printStackTrace();  
  15. }  

httpConnection读取流保存成String数据

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. URL url = new URL(getForwardUrl("/queryUserByUNorIP"));  
  2. HttpURLConnection conn = (HttpURLConnection) url.openConnection();  
  3. InputStream is = conn.getInputStream();  
  4. BufferedReader br = new BufferedReader(new InputStreamReader(is));  
  5. StringBuilder sb = new StringBuilder();  
  6. String readline = null;  
  7. while ((readline = br.readLine()) != null) {  
  8.     sb.append(readline);  
  9. }  
  10. System.out.println("result"+sb.toString());  
等效于使用ByteArrayOutputStream

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. InputStream is = conn.getInputStream();  
  2. ByteArrayOutputStream bos = new ByteArrayOutputStream();  
  3. byte[] buffer = new byte[1024];  
  4. int len =-1 ;  
  5. while ((len=is.read(buffer))!=-1) {  
  6.     bos.write(buffer, 0, len);  
  7. }  
  8. is.close();  
  9. bos.close();  
  10. String result = new String(bos.toByteArray());  
  11. System.out.println("result"+result);  

5-1,使用FileOutputStream写入文件

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. try {  
  2.     File file = new File(Environment.getExternalStorageDirectory(),  
  3.             DEFAULT_FILENAME);  
  4.         FileOutputStream fos = new FileOutputStream(file);  
  5.         String info = "I am a chinanese!";  
  6.            fos.write(info.getBytes());  
  7.            fos.close();  
  8.     System.out.println("写入成功:");  
  9. catch (Exception e) {  
  10.     e.printStackTrace();  
  11. }  

5-2,使用BufferedWriter写入文件

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. try {  
  2.     File file = new File(Environment.getExternalStorageDirectory(),  
  3.             DEFAULT_FILENAME);  
  4.     //第二个参数意义是说是否以append方式添加内容  
  5.     BufferedWriter bw = new BufferedWriter(new FileWriter(file, true));  
  6.     String info = " hey, yoo,bitch";  
  7.     bw.write(info);  
  8.     bw.flush();  
  9.     System.out.println("写入成功");  
  10. catch (Exception e) {  
  11.     e.printStackTrace();  
  12. }  


读取和写入我们都实现了,貌似很简单的样子,但是我们现在想每隔30秒进行一次数据整理,然后把他们写入到我们制定的txt文件中,但是我想每次都能在上一次的结尾处开始写入,这样在电脑上通过文本打开时,就能看到每一行的数据了。

这其实要求我们每一次写入数据时,都要有换行的操作符号,比如:\n,并且IO读写能以追加的方式写入到文件里。

刚开始我很笨的想到,每次写入前,先把文件读取出来并且生成一个StringBuffer,然后再append,然后再写入.....这种方式导致每次都要2次以上的IO操作,读和写。其实系统写入时就给我们自带了append方式,还是要勤看文档啊!

BufferedWriter

使用BufferedWriter,在构造BufferedWriter时,把第二个参数设为true
        BufferedWriter out = new BufferedWriter(new OutputStreamWriter(  
                    new FileOutputStream(file, true)));  
         out.write(conent);  

FileWriter

构造函数中的第二个参数true表示以追加形式写文件  
         FileWriter writer = new FileWriter(fileName, true);  
         writer.write(content);  
         writer.close();

// 打开一个随机访问文件流,按读写方式  
RandomAccessFile randomFile = new RandomAccessFile(fileName, "rw");  
// 文件长度,字节数  
long fileLength = randomFile.length();  
// 将写文件指针移到文件尾。  
randomFile.seek(fileLength);  
randomFile.writeBytes(content);  
randomFile.close(); 



问题:我在file写入时,没一次写完后,明明都添加了换行符(bw.write("\n")),为什么在Window的文本文档中看不到换行呢?而在EditPlus或是notepad++中就能看到换行后的效果?

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. BufferedWriter bw = new BufferedWriter(new FileWriter(file, true));  
  2. String info = " hey, yoo,bitch";  
  3. bw.write(info);  
  4. bw.write("\n");  
  5. bw.flush();  

如上代码所示,可是在windows的文本文档中:


但是在诸如notepad++或EditPlus中看到却是没问题的:



这是为什么呢?

这是windows与linux系统的编码模式不同造成的。android系统是linux内核,与windows不同。windows是采用的是DOS编码方式,所用的换行符是DOS换行符CR/LF,也就是我们俗称的\r\n,(如果不理解可以去百度一下转义字符,一般程序员会用到这些知识),而linux系统的换行符为UNIX换行符LF,也就是\n,苹果的MAC系统用的是MAC换行符CR,也就是\r,现在我想你也差不多理解了。你在android手机里建立的文档肯定用的是UNIX换行符,也就是一个\n,但是这个文档你拿到windows里用记事本打开的话,因为windows记事本是DOS换行符\r\n,所以你少了个\r,所以没法识别成换行,只能给你识别成一个小方块了,解决办法很简单,你可以用EditPlus或者UltraEdit软件打开,UltraEdit也能转换这些编码模式,转换成DOS模式就可以了。


所以,我们只需要添加:\r\n  

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. BufferedWriter bw = new BufferedWriter(new FileWriter(file, true));  
  2. String info = " hey, yoo,bitch";  
  3. bw.write(info);  
  4. bw.write("\r\n");  
  5. bw.flush();  

相关文章
|
4天前
|
Shell Android开发
Android系统 adb shell push/pull 禁止特定文件
Android系统 adb shell push/pull 禁止特定文件
16 1
|
4天前
|
存储 Java API
Android系统 文件访问权限笔记
Android系统 文件访问权限笔记
35 1
|
4天前
|
移动开发 Java Unix
Android系统 自动加载自定义JAR文件
Android系统 自动加载自定义JAR文件
21 1
|
4天前
|
Android开发
Android源代码定制:添加customize.mk文件进行分项目和分客户的定制
Android源代码定制:添加customize.mk文件进行分项目和分客户的定制
2 0
|
2月前
|
JSON Java Go
|
Java Android开发 存储
Android 文件操作心得体会
android 的文件操作说白了就是Java的文件操作的处理。所以如果对Java的io文件操作比较熟悉的话,android的文件操作就是小菜一碟了。好了,话不多说,开始今天的正题吧。
947 0
|
Android开发
android 文件操作
详细介绍:http://wenku.baidu.com/view/fcf6d3f47c1cfad6195fa724.html?from=rec&pos=0&weight=14&lastweight=1&count=5        /**          * 在SD卡上创建文件  ...
718 0
|
5天前
|
Linux 编译器 Android开发
FFmpeg开发笔记(九)Linux交叉编译Android的x265库
在Linux环境下,本文指导如何交叉编译x265的so库以适应Android。首先,需安装cmake和下载android-ndk-r21e。接着,下载x265源码,修改crosscompile.cmake的编译器设置。配置x265源码,使用指定的NDK路径,并在配置界面修改相关选项。随后,修改编译规则,编译并安装x265,调整pc描述文件并更新PKG_CONFIG_PATH。最后,修改FFmpeg配置脚本启用x265支持,编译安装FFmpeg,将生成的so文件导入Android工程,调整gradle配置以确保顺利运行。
24 1
FFmpeg开发笔记(九)Linux交叉编译Android的x265库