代码源码:http://download.youkuaiyun.com/detail/qq_33599978/9724902
一 写入 的步骤:
1. FileOutputStream 找到写入的目的地 FILE_NAME 定义的文件名
FileOutputStream out = new FileOutputStream(getFilesDir() + "/"
+ FILE_NAME,true);
2. 然后将数据写入到文件中 和关闭流 ,count是字符串数据转型成Bt
/* out.write(count.getBytes());
out.close(); */ 数据会丢失
PrintStream ps = new PrintStream(out); // 自动刷新 ,防止数据丢失
// 向文件中写入count(数据),但count是字符,所以转成字节
ps.write(count.getBytes());
ps.close();
out.close();
二:读取的步骤
1.FileInputStream 作为文件输出流,找到文件
FileInputStream in = new FileInputStream(getFilesDir() + "/"
+ FILE_NAME);
2.
//创建byte 的数组
byte[] by = new byte[1024];
//定义读取的长度
int len = -1;
//定义可变长的字符数串
StringBuffer sb = new StringBuffer();
//循环读取 追加末尾 append是将将指定的字符串添加到该缓冲区sb的结尾.
while ((len = in.read(by))!=-1) {
sb.append(new String(by, 0, len));
//new String(by, 0, len);
}
in.close();
三. 封装流
PrintStream
PrintStream在OutputStream基础之上提供了增强的功能,即可以方便地输出各种类型的数据(而不仅限于byte型)的格式化表示形式。PrintStream的方法从不抛出IOEceptin
PrintWriter
PrintWriter提供了PrintStream的所有打印方法,其方法也从不抛出IOException。
与PrintStream的区别:作为处理流使用时,PrintStream只能封装OutputStream类型的字节流,而PrintWriter既可以封装OutputStream类型的字节流,还能够封装Writer类型的字符输出流并增强其功能。
四:字符流和字节流的主要区别:
1.字节流读取的时候,读到一个字节就返回一个字节; 字符流使用了字节流读到一个或多个字节(中文对应的字节数是两个,在UTF-8码表中是3个字节)时。先去查指定的编码表,将查到的字符返回。
2.字节流可以处理所有类型数据,如:图片,MP3,AVI视频文件,而字符流只能处理字符数据。只要是处理纯文本数据,就要优先考虑使用字符流,除此之外都用字节流。
类型 | 字符流 | 字节流 |
File(文件) | FileReader FileWriter | FileInputStream FileOutputSream |
Memory Array | CharArrayReader CharArrayWriter | ByteArrayInputStream ByteArrayOutputSream |
Memory String | StringReader StringWriter | - |
Pipe(管道) | PipedReader PipedWriter | PipedInputSream PipedOutputSream |
处理流类型
该类型是对一个已存在的流的连接和封装,通过所封装的流的功能调用实现数据读写,处理流的构造方法总是要带一个其他流对象作为参数,一个流对象进过其他流的多次包装,叫做流的链接。主要可以分为以下几种:
1、缓冲流(BufferedInPutStream/BufferedOutPutStream和BufferedWriter/BufferedReader)他可以提高对流的操作效率。
写入缓冲区对象:
读取缓冲区对象:
该类型的流有一个特有的方法:readLine();一次读一行,到行标记时,将行标记之前的字符数据作为字符串返回,当读到末尾时,返回null,其原理还是与缓冲区关联的流对象的read方法,只不过每一次读取到一个字符,先不进行具体操作,先进行临时储存,当读取到回车标记时,将临时容器中储存的数据一次性返回。
2、转换流(InputStreamReader/OutputStreamWriter)
该类型时字节流和字符流之间的桥梁,该流对象中可以对读取到的字节数据进行指定编码的编码转换。
构造函数主要有:
注意:在使用FileReader操作文本数据时,该对象使用的时默认的编码表,即
FileReader fr=new FileReader(“a.txt”); 与 InputStreamReader isr=new InputStreamReader(new FileInputStream("a.txt")); 的意义相同。如果要使用指定表编码表时,必须使用转换流,即如果a.txt中的文件中的字符数据是通过utf-8的形式编码,那么在读取时,就必须指定编码表,那么转换流时必须的。即:
InputStreamReader isr=new InputStreamReader(new FileInputStream("a.txt"),utf-8);
3、数据流(DataInputStream/DataOutputStream)
该数据流可以方便地对一些基本类型数据进行直接的存储和读取,不需要再进一步进行转换,通常只要操作基本数据类型的数据,就需要通过DataStream进行包装。
构造方法:
方法举例:
注意:在使用数据流读/存数据的时候,需要有一定的顺序,即某个类型的数据先写入就必须先读出,服从先进先出的原则。
四、打印流(PrintStream/PrintWriter)
PrintStream是一个字节打印流,System.out对应的类型就是PrintStream,它的构造函数可以接受三种数据类型的值:1.字符串路径。2.File对象 3.OutputStream
PrintStream是一个字符打印流,它的构造函数可以接受四种类型的值:1.字符串路径。2.File对象 3.OutputStream 4.Writer 对于1、2类型的数据,可以指定编码表,也就是字符集,对于3、4类型的数据,可以指定自动刷新,当该自动刷新为True时,只有3个方法可以用:println,printf,format。
五、对象流(ObjectInputStream/ObjectOutputStream)
该类型的流可以把类作为一个整体进行存取,主要方法有:
Object readObject();该方法抛出异常:ClassNotFountException。
void writeObject(Object):被写入的对象必须实现一个接口:Serializable,否则就会抛出:NotSerializableException
代码
package com.example.androiddatesave;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.PrintStream;
import com.gress.utils.ToastUtil;
import android.app.Activity;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;
public class MainActivity extends Activity {
// 创建文件名 常量 文件在window-Show View-other-Android-File Exploere
private final String FILE_NAME = "gress.txt";
private EditText mEt_writeDate, mEt_readDate;
// 写入按钮的监听
public void write(View v) {
String count = mEt_writeDate.getText().toString();//一般这里是一个输入流来获取文件,但这里数据已经存在内存所以不需要在使用输入流直接调用就行
try {
/**
* getFileDir():获取应用程序数据文件的路径 "/": 表示在某个路径名下的目录下 防止IO输入为空 抛出异常
* FileOutputStream()是带一个参数是文件绝对地址,如果文件不存在就创建这个文件
* 参数ture 表示在后面追加数据
*/
FileOutputStream out = new FileOutputStream(getFilesDir() + "/"
+ FILE_NAME,true);
ToastUtil.showToast(this,getFilesDir().getPath() );
/*out.write(count.getBytes());
out.close();*/
// 将 文件包装成PrintStream流
PrintStream ps = new PrintStream(out); // 自动刷新 ,防止数据丢失
// 向文件中写入count(数据),但count是字符,所以转成字节
ps.write(count.getBytes());
ps.close();
out.close();
ToastUtil.showToast(this, "写入成功");
// Toast.makeText(this, "写入成功", Toast.LENGTH_LONG).show();
mEt_writeDate.setText("");
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
ToastUtil.showToast(this, "写入失败");
}
}
// 读取数据的监听
/**
* @param 创建文件输入流
*/
public void read(View v) {
try {
// 创建文件输入流 抛出IO输入异常 放大异常
FileInputStream in = new FileInputStream(getFilesDir() + "/"
+ FILE_NAME);
//创建byte 的数组
byte[] by = new byte[1024];
//定义读取的长度
int len = -1;
//定义可变长的字符数串
StringBuffer sb = new StringBuffer();
//循环读取 追加末尾 append是将将指定的字符串添加到该缓冲区sb的结尾.
while ((len = in.read(by))!=-1) {
sb.append(new String(by, 0, len));
//new String(by, 0, len);
}
in.close();
ToastUtil.showToast(this, "读取成功");// 封装的一个工具类
//拿到读到的字节流强转字符
String content = sb.toString();
//显示在界面上
mEt_readDate.setText(content);
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
ToastUtil.showToast(this, "读取失败");
}
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mEt_readDate = (EditText) findViewById(R.id.main_et_read);
mEt_writeDate = (EditText) findViewById(R.id.main_et_writedate);
}
}