1. 什么是IO
IO:Java对数据的操作是通过流的方式,IO流用来处理设备之间的数据传输,上传文件和下载文件,Java用于操作流的对象都在IO包中。
I:Input输入,O:Output输出
- Input指从外部读入数据到内存,例如,把文件从磁盘读取到内存,从网络读取数据到内存等等。
- Output指把数据从内存输出到外部,例如,把数据从内存写入到文件,把数据从内存输出到网络等等。
字节流和字符流:
io默认都是直接操作字节的,多用于读取或书写二进制数据,这些类的基类为InputStream或OutputStream。
而字符流操作的是为了支持Unicode编码,用于字符国际化,一个字符占用两个字节,这些类的基类为Reader或Writer。
java的io在jdk1.1以后添加了字符流的支持,为我们直接操作字符流提供了方便。
关于字符类库和字节类库的选择
最明智的做法是尽量尝试使用Reader和Writer,一旦程序代码无法成功编译,我们就会发现自己不得不使用面向字节的类库。
根据数据的流向,流分为输入流和输出流,这里指的都是内存,内存输入为读,输出为写,I读O写。
Java提供了针对不同情况的处理流的类,以便于我们直观地进行数据操作,这些类就在java的io包之中。下面介绍java io包的类,整个io包大量应用了装饰模式。
在实际使用过程中,涉及多层套用的构造函数必然都会有自己的一个装饰模式的架构,包括被装饰类(原构建基类),装饰超类,装饰具体类,用的过程中懂得去区分理解会让你更加灵活地使用它们。
- 节点流:文件(File),管道(Piped)和数组(Array)(他们每个类都分别包括输入输出和字节字符四种流)
- 处理流:其余的都是处理类,他们都是属于节点流的装饰类,下面是一个关于处理流的表格。
处理流 | who | 个数 | 装饰功能 |
---|---|---|---|
缓冲区 | Buffered开头的类 | 4((IO/BC四种) | 可将流放在缓冲区内操作 |
转化流 | InputStreamReader/OutputStreamWriter | 2 | 可将字节流转化为字符流 |
基本类型 | DataXXXStream | 2(IO替换XXX) | 可传输基本类型数据 |
对象流 | ObjectXXXStream | 2(IO替换XXX) | 可传输对象类型数据(序列化) |
打印流 | PrintStream/PrintWriter | 2 | 包含print和println的输出流 |
合并流 | SequenceInputStream | 1 | 可逻辑串联其他输入流 |
行号读入流 | LineNumberReader | 1 | 可得到一个携带行号的字符读入流 |
推回输入流 | PushbackInputStream/PushbackReader | 2 | 可将输入流push back或unread一个字节 |
字符串读写流 | StringWriter/StringReader | 2 | 可在缓冲区读写字符串 |
注:
- 默认命名规则:字节流是输入输出(Input/Output),字符流是读写(Writer/Reader),当又有输入输出又有读写的时候,那一定是转化流(InputStreamReader/OutputStreamWriter)。
- 默认都是操作字节,所有操作字符的类都需要先经过转化流将字节流转为字符流再使用。
- LineNumberInputStream已过时,因为它是基于字节输入流的,而错误假定字节能充分表示字符,现已被LineNumberReader取代。
- StringBufferInputStream已过时,因为此类未能正确地将字符转换为字节,现已被StringReader取代。
- 这里面还有一个过滤流的概念,它也包括输入输出字节字符四种流,我在上面没有表示出来。它是一个抽象类,作为“装饰器”的接口,其中,“装饰器”为其他输入输出字符字节类提供有用功能。
2. File对象
在 Java 中,File 类是 java.io 包中唯一代表磁盘文件本身的对象。File 类定义了一些与平台无关的方法来操作文件,File类主要用来获取或处理与磁盘文件相关的信息,像文件名、 文件路径、访问权限和修改日期等,还可以浏览子目录层次结构。
File 类表示处理文件和文件系统的相关信息。也就是说,File 类不具有从文件读取信息和向文件写入信息的功能,它仅描述文件本身的属性。
构造方法:
- File(String pathname) 通过将给定路径名字符串转换为抽象路径名来创建一个新 File 实例。
- File(String parent,String child) 根据指定的父路径和文件路径创建一个新File对象实例
- File(File parent,String child) 根据指定的父路径对象和文件路径创建一个新的File对象实例
public static void main(String[] args) {
//File(String pathname) 将指定路径名转换成一个File对象
File file = new File("D:\\1.txt");
System.out.println(file);
//File(String parent,String child) 根据指定的父路径和文件路径创建File对象
File file1 = new File("D:\\a", "1.txt");
System.out.println(file1);
//File(File parent,String child) 根据指定的父路径对象和文件路径创建File对象
File parent = new File("D:\\a");
File file2 = new File(parent, "1.txt");
System.out.println(file2);
File file3 = new File(new File("D:\\a"), "1.txt");
System.out.println(file3);
/*D:\1.txt
D:\a\1.txt
D:\a\1.txt
D:\a\1.txt*/
}
File类创建和删除功能
- boolean createNewFile();指定路径不存在该文件时创建文件,返回true 否则false
- boolean mkdir() 当指定的单击文件夹不存在时创建文件夹并返回true 否则false
- boolean mkdirs() 但指定的多级文件夹在某一级文件夹不存在时,创建多级文件夹并返回true 否则false
- boolean delete() 删除文件或者删除单级文件夹
- 删除文件夹,这个文件夹下面不能有其他的文件和文件夹
public static void main(String[] args) throws IOException {
File file = new File("D:\\a\\1.txt");
File file1 = new File("1.txt");
boolean flag = file1.createNewFile();
System.out.println(flag);
File file2 = new File("b");
boolean flag2 = file2.mkdir();
System.out.println(flag);
File file3 = new File("c\\d\\e");
boolean d = file1.mkdir();
boolean c = file1.mkdirs();
System.out.println(d);
System.out.println(c);
File file4 = new File("c.txt");
System.out.println(file4.mkdir());
File file5 = new File("b");
System.out.println(file2.delete());
/*true
true
false
false
true
true
*/
}
File类的判断功能
- boolean exists() 判断指定路径的文件或文件夹是否为空
- boolean isAbsolute() 判断当前路径是否是绝对路径
- boolean isDirectory() 判断当前的目录是否存在
- boolean isFile() 判断当前的目录是否是一个文件
- boolean isHidden() 判断当前路径是否是一隐藏文件
public static void main(String[] args) throws IOException {
/*判断文件是否存在*/
File file = new File("a.txt");
boolean newFile = file.createNewFile();
System.out.println("newFile = " + newFile);//newFile = true
boolean flag = file.exists();
System.out.println(flag);//true
/*判断文件是绝对路径*/
File file2 = new File("D:\\a\\1.txt");
boolean flag2 = file2.isAbsolute();
System.out.println(flag2);//true
/*判断是否是文件/目录*/
File file1 = new File("1.txt");
File b = new File("b");
System.out.println(file1.isDirectory());//false
System.out.println(b.isFile());//false
/*判断是否是隐藏文件*/
File file3 = new File("D:\\a\\1.txt");
System.out.println(file3.isHidden());//false
}
File类的获取功能和修改名字功能
- File getAbsoluteFile() 获取文件的绝对路径,返回File对象
- String getAbsolutePath() 获取文件的绝对路径,返回路径的字符串
- String getParent() 获取当前路径的父级路径,以字符串形式返回该父级路径
- String getName() 获取文件或文件夹的名称
- String getPath() 获取File对象中封装的路径
- long lastModified() 以毫秒值返回最后修改时间
- long length() 返回文件的字节数
- boolean renameTo(File dest) 将当前File对象所指向的路径修改为指定File所指向的路径
获取文件的绝对路径,返回File对象:
File file = new File("D:\\a\\1.txt");
File file1 = new File("1.txt");
//以File对象返回的形式返回当前File对象所指向的绝对路径
System.out.println(file1.getAbsoluteFile());
//返回File对象所指向的绝对路径
System.out.println(file1.getAbsolutePath());
/*E:\Java_code\JavaseStudy\1.txt
E:\Java_code\JavaseStudy\1.txt*/
获取文件的绝对路径,返回路径的字符串:
File file = new File("D:\\a.txt");
String absolutePath = file.getAbsolutePath();
System.out.println("absolutePath = " + absolutePath);
/*absolutePath = D:\a.txt*/
获取当前路径的父级路径,以字符串形式返回该父级路径:
File file = new File("D:\\a.txt");
String parent = file.getParent();
System.out.println("parent = " + parent);
/*parent = D:\*/
获取文件或文件夹的名称:
File file = new File("D:\\a.txt");
String name = file.getName();
System.out.println("name = " + name);
/*name = a.txt*/
获取File对象中封装的路径:
File file = new File("D:\\a.txt");
String path = file.getPath();
System.out.println("path = " + path);
/*path = D:\a.txt*/
以毫秒值返回最后修改时间:
File file = new File("D:\\a.txt");
long l = file.lastModified();
System.out.println("l = " + l);
/*l = 1584069256525*/
// 文件不存在则返回0
返回文件的字节数:
File file = new File("D:\\a.txt");
long length = file.length();
System.out.println("length = " + length);
//0
将当前File对象所指向的路径修改为指定File所指向的路径:
File file = new File("D:\\a.txt");
boolean b = file.renameTo(new File("E:\\a.txt"));
System.out.println("b = " + b);
/*b = true*/
// 此操作相当于把位于D盘的文件移到E盘了
File 类的其他获取功能
- String[] list(); 以字符串的形式返回当前路径下所有的文件和文件夹的名称
- File[] listFile 以File对象的形式返回当前路径下的所有文件和文件夹名称
- Static File[] listRoots() 获取计算机中的所有盘符
以字符串的形式返回当前路径下所有的文件和文件夹的名称
File file = new File("D:\\");
String[] list = file.list();
for (String s : list) {
System.out.println("s = " + s);
}
/*s = apps
s = Program Files
s = Program Files (x86)
s = PSAutoRecover*/
// 这是我演示时,D盘所存在的文件夹
以File对象的形式返回当前路径下的所有文件和文件夹名称
File file = new File("D:\\");
File[] files = file.listFiles();
for (File file1 : files) {
System.out.println(file1.isDirectory());//true
}
获取计算机中的所有盘符
File[] files = File.listRoots();
for (File file : files) {
System.out.println(file);
}
3. 字节流
IO流图示:
3.1. 字节流基类
InputStream
InputStream:字节输入流基类、抽象类,是表示字节输入流的所有类的超类。
常用方法
// 从输入流中读取数据的下一个字节
abstract int read()
// 从输入流中读取一定数量的字节,并将其存储在缓冲区数组 b中
int read(byte[] b)
// 将输入流中最多 len 个数据字节读入 byte 数组
int read(byte[] b, int off, int len)
// 跳过和丢弃此输入流中数据的 n个字节
long skip(long n)
// 关闭此输入流并释放与该流关联的所有系统资源
void close()
OutputStream
OutputStream:字节输出流基类、抽象类,是表示输出字节流的所有类的超类。
常用方法:
// 将 b.length 个字节从指定的 byte 数组写入此输出流
void write(byte[] b)
// 将指定 byte 数组中从偏移量 off 开始的 len 个字节写入此输出流
void write(byte[] b, int off, int len)
// 将指定的字节写入此输出流
abstract void write(int b)
// 关闭此输出流并释放与此流有关的所有系统资源
void close()
// 刷新此输出流并强制写出所有缓冲的输出字节
void flush()
3.2. 字节文件操作流
FileInputStream
FileInputStream:字节文件输入流,从文件系统中的某个文件中获得输入字节,用于读取诸如图像数据之类的原始字节流。
构造方法
// 通过打开一个到实际文件的连接来创建一个FileInputStream,该文件通过文件系统中的File对象file指定
FileInputStream(File file)
// 通过打开一个到实际文件的连接来创建一个FileInputStream,该文件通过文件系统中的路径name指定
FileInputStream(String name)
常用方法
// 读取f盘下该文件f://hell/test.txt
//构造方法1
InputStream inputStream = new FileInputStream(new File("f://hello//test.txt"));
int i = 0;
//一次读取一个字节
while ((i = inputStream.read()) != -1) {
// System.out.print(i + " ");// 65 66 67 68
//为什么会输出65 66 67 68?因为字符在底层存储的时候就是存储的数值。即字符对应的ASCII码。
System.out.print((char) i + " ");// A B C D
}
//关闭IO流
inputStream.close();
// 读取f盘下该文件f://hell/test.txt
//构造方法2
InputStream inputStream2 = new FileInputStream("f://hello/test.txt");
// 字节数组
byte[] b = new byte[2];
int i2 = 0;
// 一次读取一个字节数组
while ((i2 = inputStream2.read(b)) != -1) {
System.out.print(new String(b, 0, i2) + " ");// AB CD
}
//关闭IO流
inputStream2.close();
注意: 一次读取一个字节数组,提高了操作效率,IO流使用完毕一定要关闭。
FileOutputStream
FileOutputStream:字节文件输出流是用于将数据写入到File,从程序中写入到其他位置。
构造方法
// 创建一个向指定File对象表示的文件中写入数据的文件输出流
FileOutputStream(File file)
// 创建一个向指定File对象表示的文件中写入数据的文件输出流
FileOutputStream(File file, boolean append)
// 创建一个向具有指定名称的文件中写入数据的输出文件流
FileOutputStream(String name)
// 创建一个向具有指定name的文件中写入数据的输出文件流
FileOutputStream(String name, boolean append)
常用方法:覆盖和重写了父类的的常用方法。
OutputStream outputStream = new FileOutputStream(new File("test.txt"));
// 写出数据
outputStream.write("ABCD".getBytes());
// 关闭IO流
outputStream.close();
// 内容追加写入
OutputStream outputStream2 = new FileOutputStream("test.txt", true);
// 输出换行符
outputStream2.write("\r\n".getBytes());
// 输出追加内容
outputStream2.write("hello".getBytes());
// 关闭IO流
outputStream2.close();
注意:
1、输出的目的地文件不存在,则会自动创建,不指定盘符的话,默认创建在项目目录下;
2、输出换行符时一定要写\r\n不能只写\n,因为不同文本编辑器对换行符的识别存在差异性。
3.3. 字节缓冲流(高效流)
BufferedInputStream
BufferedInputStream:字节缓冲输入流,提高了读取效率。
构造方法
// 创建一个 BufferedInputStream并保存其参数,即输入流in,以便将来使用。
BufferedInputStream(InputStream in)
// 创建具有指定缓冲区大小的 BufferedInputStream并保存其参数,即输入流in以便将来使用
BufferedInputStream(InputStream in, int size)
InputStream in = new FileInputStream("test.txt");
// 字节缓存流
BufferedInputStream bis = new BufferedInputStream(in);
byte[] bs = new byte[20];
int len = 0;
while ((len = bis.read(bs)) != -1) {
System.out.print(new String(bs, 0, len));
// ABCD
// hello
}
// 关闭流
bis.close();
BufferedOutputStream
BufferedOutputStream:字节缓冲输出流,提高了写出效率。
构造方法
// 创建一个新的缓冲输出流,以将数据写入指定的底层输出流
BufferedOutputStream(OutputStream out)
// 创建一个新的缓冲输出流,以将具有指定缓冲区大小的数据写入指定的底层输出流
BufferedOutputStream(OutputStream out, int size)
常用方法
// 将指定 byte 数组中从偏移量 off 开始的 len 个字节写入此缓冲的输出流
void write(byte[] b, int off, int len)
// 将指定的字节写入此缓冲的输出流
void write(int b)
// 刷新此缓冲的输出流
void flush()
测试:
BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("test.txt", true));
// 输出换行符
bos.write("\r\n".getBytes());
// 输出内容
bos.write("Hello Android".getBytes());
// 刷新此缓冲的输出流
bos.flush();
// 关闭流
bos.close();
4. 字符流
4.1. 字符流基类
Reader
Reader:读取字符流的抽象类
常用方法
// 读取单个字符
int read()
// 将字符读入数组
int read(char[] cbuf)
// 将字符读入数组的某一部分
abstract int read(char[] cbuf, int off, int len)
// 跳过字符
long skip(long n)
// 关闭该流并释放与之关联的所有资源
abstract void close()
Writer
Writer:写入字符流的抽象类
常用方法
// 写入字符数组
void write(char[] cbuf)
// 写入字符数组的某一部分
abstract void write(char[] cbuf, int off, int len)
// 写入单个字符
void write(int c)
// 写入字符串
void write(String str)
// 写入字符串的某一部分
void write(String str, int off, int len)
// 将指定字符添加到此 writer
Writer append(char c)
// 将指定字符序列添加到此 writer
Writer append(CharSequence csq)
// 将指定字符序列的子序列添加到此 writer.Appendable
Writer append(CharSequence csq, int start, int end)
// 关闭此流,但要先刷新它
abstract void close()
// 刷新该流的缓冲
abstract void flush()
4.2. 字符转换流
InputStreamReader
InputStreamReader:字节流转字符流,它使用的字符集可以由名称指定或显式给定,否则将接受平台默认的字符集。
常用方法
// 创建一个使用默认字符集的 InputStreamReader
InputStreamReader(InputStream in)
// 创建使用给定字符集的 InputStreamReader
InputStreamReader(InputStream in, Charset cs)
// 创建使用给定字符集解码器的 InputStreamReader
InputStreamReader(InputStream in, CharsetDecoder dec)
// 创建使用指定字符集的 InputStreamReader
InputStreamReader(InputStream in, String charsetName)
特有方法
//返回此流使用的字符编码的名称
String getEncoding(
测试:
//使用默认编码
InputStreamReader reader = new InputStreamReader(new FileInputStream("test.txt"));
int len;
while ((len = reader.read()) != -1) {
System.out.print((char) len);//爱生活,爱Android
}
reader.close();
//指定编码
InputStreamReader reader = new InputStreamReader(new FileInputStream("test.txt"),"utf-8");
int len;
while ((len = reader.read()) != -1) {
System.out.print((char) len);//????????Android
}
reader.close();
OutputStreamWriter
OutputStreamWriter:字节流转字符流。
构造方法
// 创建使用默认字符编码的 OutputStreamWriter
OutputStreamWriter(OutputStream out)
// 创建使用给定字符集的 OutputStreamWriter
OutputStreamWriter(OutputStream out, Charset cs)
// 创建使用给定字符集编码器的 OutputStreamWriter
OutputStreamWriter(OutputStream out, CharsetEncoder enc)
// 创建使用指定字符集的 OutputStreamWriter
OutputStreamWriter(OutputStream out, String charsetName)
特有方法
//返回此流使用的字符编码的名称
String getEncoding()
4.3. 字符缓冲流(高效流)
BufferedReader
BufferedReader:字符缓冲流,从字符输入流中读取文本,缓冲各个字符,从而实现字符、数组和行的高效读取。
构造方法
// 创建一个使用默认大小输入缓冲区的缓冲字符输入流
BufferedReader(Reader in)
// 创建一个使用指定大小输入缓冲区的缓冲字符输入流
BufferedReader(Reader in, int sz)
特有方法
// 读取一个文本行
String readLine()
//生成字符缓冲流对象
BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream("test.txt")));
String str;
//一次性读取一行
while ((str = reader.readLine()) != null) {
System.out.println(str);// 爱生活,爱Android
}
//关闭流
reader.close();
BufferedWriter
BufferedWriter:字符缓冲流,将文本写入字符输出流,缓冲各个字符,从而提供单个字符、数组和字符串的高效写入。
构造方法
// 创建一个使用默认大小输出缓冲区的缓冲字符输出流
BufferedWriter(Writer out)
// 创建一个使用给定大小输出缓冲区的新缓冲字符输出流
BufferedWriter(Writer out, int sz)
特有方法
// 写入一个行分隔符
void newLine()
FileReader、FileWriter
FileReader:InputStreamReader类的直接子类,用来读取字符文件的便捷类,使用默认字符编码。
FileWriter:OutputStreamWriter类的直接子类,用来写入字符文件的便捷类,使用默认字符编码。