Java IO 基本概念java
Java IO:即 Java 输入 / 输出系统。web
区分 Java 的输入和输出:把本身当成程序, 当你从外边读数据到本身这里就用输入(InputStream/Reader), 向外边写数据就用输出(OutputStream/Writer)。面试
Stream:Java 中将数据的输入输出抽象为流,流是一组有顺序的,单向的,有起点和终点的数据集合,就像水流。按照流中的最小数据单元又分为字节流和字符流。算法
1,字节流:以 8 位(即 1 byte,8 bit)做为一个数据单元,数据流中最小的数据单元是字节。数组
2,字符流:以 16 位(即 1 char,2 byte,16 bit)做为一个数据单元,数据流中最小的数据单元是字符, Java 中的字符是 Unicode 编码,一个字符占用两个字节。tomcat
IO 的分类网络
流式部分和非流式部分dom
Java 的 IO 主要包含两个部分:分布式
1.流式部分:是 IO 的主体部分,也是本文介绍的重点, 流式部分根据流向分为输入流(InputStream/Reader)和输出流(OutputStream/Writer), 根据数据不一样的操做单元,分为字节流(InputStream/OutputStream)和字符流(Reader/Writer),依据字节流和字符流,Java 定义了用来操做数据的抽象基类InputStream/OutputStream 和 Reader/Writer,再根据不一样应用场景(或功能),在这两种抽象基类上基于数据载体或功能派上出不少子类,用来知足文件,网络,管道等不一样场景的 IO 需求,从而造成了 Java 的基本 IO 体系。ide
下面是 Java IO 体系中经常使用的流类:
2.非流式部分:主要包含一些辅助流式部分的类,如: SerializablePermission 类、File 类、RandomAccessFile 类和 FileDescriptor 等;
节点流和处理流
Java io 分类方式有不少,根据是否直接处理数据,Java io又分为节点流和处理流,节点流是真正直接处理数据的;处理流是装饰加工节点流的。
节点流
处理流
处理流是对一个已存在的流的链接和封装,经过所封装的流的功能调用实现数据读写。如 BufferedReader。
处理流的构造方法老是要带一个其余的流对象作参数。
经常使用处理流(经过关闭处理流里面的节点流来关闭处理流)
字节流
字节输入流
下面是 IO 中输入字节流的继承关系。
InputStream
总结:
InputStream 中的三个基本的读方法
字节输出流
下面是 IO 中输出字节流的继承关系。
OutputStream
总结:
outputStream中的三个基本的写方法
其它重要方法:
字节流的输入与输出的对应
java io 的输入和输出是高度对应的,下图表示字节流的输入与输出的对应关系。
上图中蓝色的为主要的对应部分,红色的部分是不对应部分。紫色的虚线部分表明这些流通常要搭配使用。
咱们主要看看这些字节流中不对称的几个类:
搭配使用的三对类: ObjectInputStream / ObjectOutputStream 和 DataInputStream / DataOutputStream 主要是要求写对象/数据和读对象 / 数据的次序要保持一致,不然可能不能获得正确的数据,甚至抛出异常(通常会如此);PipedInputStream / PipedOutputStream 在建立时通常就一块儿建立,调用它们的读写方法时会检查对方是否存在,或者关闭!
字符流
字符输入流 Reader
下面是 IO 中输入字符流的继承关系。
Reader
总结:
Reader 基本的三个读方法(和字节流对应):
(1) public int read() throws IOException; 读取一个字符,返回值为读取的字符。
(2) public int read(char cbuf[]) throws IOException; 读取一系列字符到数组 cbuf[]中,返回值为实际读取的字符的数量。
(3) public abstract int read(char cbuf[],int off,int len) throws IOException; 读取 len 个字符,从数组 cbuf[] 的下标 off 处开始存放,返回值为实际读取的字符数量,该方法必须由子类实现。
字符输出流 Writer
下面是 IO 中输出字符流的继承关系。
Writer
总结(和字节输出流对应):
writer 的主要写方法:
字符流的输入与输出的对应
可参照(字节流的输入与输出的对应)记忆
Java IO 常见用法
一、读取键盘输入,打印到控制台 在刷题网站刷算法题的时候,在程序开头都须要和键盘进行交互,经常用到行夺取器 BufferedReader 和转换流 InputStreamReader。
public static void keyInAndPrintConsole() throws IOException {
PrintWriter out = null;
BufferedReader br = null;
try{
System.out.println("请输入:");
out = new PrintWriter(System.out, true);
br = new BufferedReader(new InputStreamReader(System.in));
String line = null;
while ((line = br.readLine()) != null) {
if (line.equals("exit")) {
System.exit(1);
}
out.println(line);
}
} catch (IOException e) {
e.printStackTrace();
}finally{
out.close();
br.close();
}
}
运行结果:
2 、用字节流读写文件
由于是用字节流来读媒介,因此对应的流是 InputStream 和 OutputStream,而且媒介对象是文件,因此用到子类是 FileInputStream 和 FileOutputStream,这里还能够经过 BufferedInputStream 用缓冲流来读取文件。
public static void readAndWriteByteToFile() throws IOException {
InputStream is =null;
OutputStream os = null;
try {
// 在try()中打开文件会在结尾自动关闭
is = new FileInputStream("D:/FileInputStreamTest.txt");
os = new FileOutputStream("D:/FileOutputStreamTest.txt");
byte[] buf = new byte[4];
int hasRead = 0;
while ((hasRead = is.read(buf)) > 0) {
os.write(buf, 0, hasRead);
}
System.out.println("write success");
} catch (Exception e) {
e.printStackTrace();
}finally{
os.close();
is.close();
}
}
运行结果:
三、用字符流进行读写操做
(1)FileReader 和 FileWriter
// 在try() 中打开的文件, JVM会自动关闭
public static void readAndWriteCharToFile() throws IOException{
Reader reader = null;
Writer writer =null;
try {
File readFile = new File("d:/FileInputStreamTest.txt");
reader = new FileReader(readFile);
File writeFile = new File("d:/FileOutputStreamTest.txt");
writer = new FileWriter(writeFile);
char[] byteArray = new char[(int) readFile.length()];
int size = reader.read(byteArray);
System.out.println("大小:" + size + "个字符;内容:" + new String(byteArray));
writer.write(byteArray);
} catch (Exception e) {
e.printStackTrace();
}finally{
reader.close();
writer.close();
}
}
运行结果:
(2)StringReader 和 StringWriter
public static void stringNode() throws IOException {
StringReader sr =null;
StringWriter sw =null;
try {
String str = "学习不刻苦 " + "不如卖红薯; ";
char[] buf = new char[32];
int hasRead = 0;
// StringReader将以String字符串为节点读取数据
sr = new StringReader(str);
while ((hasRead = sr.read(buf)) > 0) {
System.out.print(new String(buf, 0, hasRead));
}
// 因为String是一个不可变类,所以建立StringWriter时,其实是以一个StringBuffer做为输出节点
sw = new StringWriter();
sw.write("黑夜给了我黑色的眼睛 ");
sw.write("我却用它寻找光明 ");
// toString()返回sw节点内的数据
System.out.println(sw.toString());
} catch (Exception e) {
e.printStackTrace();
}finally{
sw.close();
sr.close();
}
}
运行结果:
四、字节流转换为字符流
在例 3 中用字符流读文件时,打印到控制台的中文会乱码,使用转换流能够解决这一问题。
public static void convertByteToChar() throws IOException {
InputStream is =null;
Reader reader = null;
try {
File file = new File("d:/FileInputStreamTest.txt");
is = new FileInputStream(file);
reader = new InputStreamReader(is,"gbk");
char[] byteArray = new char[(int) file.length()];
int size = reader.read(byteArray);
System.out.println("大小:" + size + ";内容:" + new String(byteArray));
} catch (Exception e) {
e.printStackTrace();
}finally{
reader.close();
is.close();
}
}
运行结果:
五、随机读写文件 使用 RandomAccessFile 能够实现对文件的随机读取,主要是经过 seek() 方法实现指针偏移。
public static void randomAccessFileReadAndWrite() throws IOException {
RandomAccessFile randomAccessFile =null;
try {
// 建立一个RandomAccessFile对象
randomAccessFile = new RandomAccessFile("d:/File.txt", "rw");
// 经过seek方法来移动指针
randomAccessFile.seek(10);
// 获取当前指针
long pointerBegin = randomAccessFile.getFilePointer();
// 从当前指针开始读
byte[] contents = new byte[10];
randomAccessFile.read(contents);
long pointerEnd = randomAccessFile.getFilePointer();
System.out.println("pointerBegin:" + pointerBegin + " " + "pointerEnd:" + pointerEnd + " " + new String(contents));
randomAccessFile.seek(20);
// 获取当前指针
long begin = randomAccessFile.getFilePointer();
randomAccessFile.write(contents);
long end = randomAccessFile.getFilePointer();
System.out.println("begin:" + begin + " " + "end:" + end + " ");
} catch (Exception e) {
e.printStackTrace();
}finally{
randomAccessFile.close();
}
}
运行结果:
操做前文件内容以下图:
操做后控制台打印信息:
操做后的文件内容:
六、读写管道
管道流要成对使用
public static void piped() throws IOException {
final PipedOutputStream output = new PipedOutputStream();
final PipedInputStream input = new PipedInputStream(output);
Thread thread1 = new Thread(new Runnable() {
@Override
public void run() {
try {
output.write("Hello world, pipe!".getBytes());
} catch (IOException e) {
}
}
});
Thread thread2 = new Thread(new Runnable() {
@Override
public void run() {
try {
int data = input.read();
while (data != -1) {
System.out.print((char) data);
data = input.read();
}
} catch (IOException e) {
} finally {
try {
input.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
});
thread1.start();
thread2.start();
}
运行结果:
七、将多个输入流当成一个输入流依次读取
public static void sequeue() throws IOException {
FileInputStream fistream1 =null;
FileInputStream fistream2 =null;
SequenceInputStream sistream =null;
FileOutputStream fostream =null;
try {
fistream1 = new FileInputStream("d:/A.txt");
fistream2 = new FileInputStream("d:/B.txt");
sistream = new SequenceInputStream(fistream1, fistream2);
fostream = new FileOutputStream("d:/C.txt");
int temp;
while( ( temp = sistream.read() ) != -1) {
System.out.print( (char) temp );
fostream.write(temp);
}
} catch (Exception e) {
e.printStackTrace();
}finally{
fostream.close();
sistream.close();
fistream1.close();
fistream2.close();
}
}
运行结果:
八、推回输入流使用实例
public static void pushback() throws FileNotFoundException, IOException {
try (PushbackReader pr = new PushbackReader(new FileReader("D:/A.txt"), 64)) {
char[] buf = new char[32];
String lastContent = "";
int hasRead = 0;
while ((hasRead = pr.read(buf)) > 0) {
String content = new String(buf, 0, hasRead);
int targetIndex = 0;
if ((targetIndex = (lastContent + content).indexOf("A")) > 0) {
System.out.println(targetIndex);
pr.unread((lastContent + content).toCharArray());
if (targetIndex > 32) {
buf = new char[targetIndex];
}
pr.read(buf, 0, targetIndex);
System.out.println(new String(buf, 0, targetIndex));
System.out.println(new String(buf, targetIndex, buf.length-targetIndex));
System.exit(0);
} else {
System.out.println(lastContent);
lastContent = content;
}
}
} catch (IOException e) {
e.printStackTrace();
}
}
运行结果:
Java IO 常见面试题
一、字节流和字符流的区别?
(1)读写单位不一样:字节流以字节(8 bit)为单位,字符流以字符为单位,根据码表映射字符,一次可能读多个字节。
(2)处理对象不一样:字节流能处理全部类型的数据(如图片、avi 等),而字符流只能处理字符类型的数据。
(3)字节流没有缓冲区,是直接输出的,而字符流是输出到缓冲区的。所以在输出时,字节流不调用 colse() 方法时,信息已经输出了,而字符流只有在调用 close() 方法关闭缓冲区时,信息才输出。要想字符流在未关闭时输出信息,则须要手动调用 flush() 方法。
二、什么是节点流,什么是处理流,它们各有什么用处,处理流的建立有什么特征?
见上文:节点流和处理流;
注意:处理流的构造器必需要 传入节点流的子类
三、什么叫对象序列化,什么是反序列化,实现对象序列化须要作哪些工做?
例如,在 web 开发中,若是对象被保存在了 Session 中,tomcat 在重启时要把 Session 对象序列化到硬盘,这个对象就必须实现 Serializable 接口。若是对象要通过分布式系统进行网络传输,被传输的对象就必须实现 Serializable 接口。
四、什么是 Filter 流有哪些?
FilterStream 是一种 IO 流,主要做用是用来对存在的流增长一些额外的功能,像给目标文件增长源文件中不存在的行数,或者增长拷贝的性能等。在 java.io 包中主要由 4 个可用的 filter Stream。两个字节 filter stream,两个字符 filter stream.
分别是:FilterInputStream,FilterOutputStream,FilterReader and FilterWriter. 这些类是抽象类,不能被实例化的。
FilterInputStream 流的子类:
六、说说 RandomAccessFile?
它在 java.io 包中是一个特殊的类,既不是输入流也不是输出流,它二者均可以作到。他是 Object 的直接子类。一般来讲,一个流只有一个功能,要么读,要么写。可是 RandomAccessFile 既能够读文件,也能够写文件。
并且 RandomAccessFile 支持对文件的随机访问,实例可见上文:例 5,随机读写文件。
总结
不少初学者刚刚学习 java 的 IO 时会比较茫然,确实 IO 类不少,不容易记忆,不过咱们能够尝试对其进行总结记忆,把流式部分归纳为:两对应一桥梁一随机。
两个对应指:
总结记忆会使 Java io 清晰好记许多,你们也能够尝试按照本身的方式进行总结,印象会更加深