IO流引入
将数据保存到变量中,集合等内存区域时存在一个弊端,即程序一旦结束,数据就会立刻消失,无法做到数据的持久化存储。
如何做到数据的持久化呢?
将数据从内存写入文件系统做到持久化。
如何将数据写入呢?
Java提供了IO流(输入输出流)。
流:观察生活中的水流、物流等,描述一下数据流的概念。
IO流概述
在程序中所有的数据都是以流的方式进行传输或保存的,程序通过输入流读取数据;当程序需要将一些数据长期保存起来的时候使用输出流完成。
例如:本地文件拷贝,上传文件和下载文件等等。
注意:
1、但凡是对数据的操作,Java都是通过流的方式来操作的。
2、程序中的输入输出都是以流的形式保存的,流中保存的实际上全都是字节文件。
3、IO流可以做到数据的持久化,但是IO流本质上是用来处理本地文件系统以及不同设备之间的数据传输。
IO流分类
按照数据流向
输入流:从外界(键盘、网络、文件…)读取数据到内存
输出流:用于将程序中的数据写出到外界(显示器、文件…)
数据源 目的地 交通工具
按照数据类型
字节流:主要用来处理字节或二进制对象。
字节输入流(InputStream)
字节输出流 (OutputStream)
字符流:主要用来处理字符、字符数组或字符串。
字符输入流(Reader)
字符输出流(Writer)
字节流和字符流的区别:
读写单位不同:字节流以字节(8bit)为单位,字符流以字符为单位,根据码表映射字符,一次可能读多个字节。
处理对象不同:字节流能处理所有类型的数据(如图片、avi等),而字符流只能处理字符类型的数据。
选取:只要是处理纯文本数据,就优先考虑使用字符流,除此之外都使用字节流。
IO流体系结构图
字节流
字节输出流
OutputStream(抽象类)
FileOutputStream
构造方法
FileOutputStream(String name) 创建一个向name文件中写数据的输出文件流
FileOutputStream(File file) 创建一个向file文件中写数据的输出文件流
FileOutputStream(File file, boolean append) 创建一个向file文件中追加数据的输出文件流
成员方法
public void write(int b) 写一个字节
public void write(byte[] b) 写一个字节数组
public void write(byte[] b,int off,int len) 写一个字节数组的一部分
void close() 关闭此文件输出流并释放与此流有关的所有系统资源
关于字节流的一些注意事项:
1、数据写入完成后记得调用close()方法关闭流对象,如果没有关闭流对象并且还在继续使用的话,会抛出异常,显示Stream Closed
2、数据追加写入要使用如下构造方法
FileOutputStream(File file, boolean append)
3、不同的系统针对不同的换行符号识别是不一样的
windows \r\n
Linux \n
max \r
常见的一些高级记事本,是可以识别任意换行符号的。
4、数据写入中存在两个异常需要处理FileNotFoundException,IOException。
Map<String, Object> map = new HashMap<String, Object>();
map.put("title", "静夜思");
map.put("author", "李白");
List<String> list = new ArrayList<>();
list.add("床前明月光");
list.add("疑似地上霜");
list.add("举头望明月");
list.add("低头思故乡");
map.put("poem", list);
FileOutputStream fos = null;
try {
fos = new FileOutputStream("aaa\\bbb\\ccc\\ddd\\eee.txt");
fos.write(((String)map.get("title")).getBytes());
fos.write("\r\n".getBytes());
fos.flush();
fos.write(((String)map.get("author")).getBytes());
fos.write("\r\n".getBytes());
fos.flush();
@SuppressWarnings("unchecked")
List<String> poemList = (List<String>) map.get("poem");
for (String poem : poemList) {
fos.write((poem + "\r\n").getBytes());
fos.flush();
}
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
if (fos != null) {
try {
fos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
//静夜思
//李白
//床前明月光
//疑似地上霜
//举头望明月
//低头思故乡
需求:将文本文件的数据读取出来显示在控制台
InputStream
FileInputStream
FileInputStream的构造方法
FileInputStream(File file) 创建一个从file读取数据的输入流
FileInputStream(String name) 创建一个从name文件读取数据的输入流
FileInputStream的成员方法
int read() 一次读取一个字节
1.读取一个字节并且返回给调用者,返回的是读取的内容
2.将指针移动到下一个位置
3.读取到文件末尾返回-1
int b = 0;
while((b = is.read()) != -1) {
System.out.print((char)b);
}
int read(byte[] b) 一次读取一个字节数组
1.读取一个字节数组到定义好的数组中,返回的是实际读取的长度
2.将指针移动到下一个位置
3.读取到文件末尾返回-1
int len = 0;
byte[] bys = new byte[5];
while((len = is.read(bys)) != -1) {
System.out.print(new String(bys, 0, len));
}
int read(byte[] b, int off, int len) 一次读取一个字节数组的一部分
void close() 关闭此文件输入流并释放与此流有关的所有系统资源
注意:
关于输入流的一些注意事项
1.输出流文件不一定要存在,会自动创建 ,输入流文件一定要存在,否则会抛出异常 抛出FileNotFindException
2.计算机如何识别中文? 中文在GBK编码表中使用两个字节表示,两个字节的第一个字节是负数,
计算机它首先读取一个字节,发现该字节是负数,它会自动等待下一个字节来组合
InputStream is = new FileInputStream("a.txt");
int len = 0;
byte[] bys = new byte[1024*8];
while ((len = is.read(bys)) != -1) {
System.out.println(new String(bys, 0, len));
}