IO概述
一种数据的流动,按照流动的方向,以内存为基准,分为输入input 和 输出output ,即流向内存是输入流,流出内存的输出流。
IO的分类
根据数据的流向分为:输入流和输出流。
- 输入流 :把数据从
其他设备
上读取到内存
中的流。 - 输出流 :把数据从
内存
中写出到其他设备
上的流。
格局数据的类型分为:字节流和字符流。
- 字节流 :以字节为单位,读写数据的流。
- 字符流 :以字符为单位,读写数据的流。
输入流 | 输出流 | |
---|---|---|
字节流 | 字节输入流 InputStream | 字节输出流 OutputStream |
字符流 | 字符输入流 Reader | 字符输出流 Writer |
字节流(万能流)
一切皆为字节
文本(用windows自带的文本文件可以打开并且不乱码)既可以用字节流也可以用字符流,而图片、视频只能用字节流。都是以二进制数字的形式保存。
如何选择字节流还是字节流,1. 看流向 2. 看文件类型
字节输出流【OutputStream】
java.io.OutputStream 抽象类是表示字节输出流的所有类的超类。
-
public void close() :关闭此输出流并释放与此流相关联的任何系统资源。
-
public void flush() :刷新此输出流并强制任何缓冲的输出字节被写出。
-
public void write(byte[] b) :将 b.length字节从指定的字节数组写入此输出流。
-
public void write(byte[] b, int off, int len) :从指定的字节数组写入 len字节,从偏移量 offff开始输出到此输出流。
-
public abstract void write(int b) :将指定的字节输出流。
close方法,当完成流的操作时,必须调用此方法,释放系统资源。
FileOutputStream类
OutputStream 有很多子类,java.io.FileOutputStream 类是文件输出流,用于将数据写出到文件。
构造方法
- public FileOutputStream(File file) :创建文件输出流以写入由指定的 File对象表示的文件。
- public FileOutputStream(String name) : 创建文件输出流以指定的名称写入文件。 、
当你创建一个流对象时,必须传入一个文件路径。该路径下,如果没有这个文件,会创建该文件。如果有这个文件,会清空这个文件的数据。
数据追加续写
- public FileOutputStream(File file, boolean append) : 创建文件输出流以写入由指定的 File对象表示的文件。
- public FileOutputStream(String name, boolean append) : 创建文件输出流以指定的名称写入文件。
参数中都需要传入一个boolean类型的值, true 表示追加数据, false 表示清空原有数据。
写出字节数据
- 写出字节: write(int b) 方法,每次可以写出一个字节数据。
虽然参数为int类型四个字节,但是只会保留一个字节的信息写出。
流操作完毕后,必须释放系统资源,调用close方法,千万记得。
- 写出字节数组: write(byte[] b) ,每次可以写出数组中的数据。
- 写出指定长度字节数组: write(byte[] b, int off, int len) ,每次写出从off索引开始,len个字节。
写出换行
Windows系统里,换行符号是 \r\n 。write("\r\n".getBytes())。
回车符 \r 和换行符 \n :
- 回车符:回到一行的开头(return)。
- 换行符:下一行(newline)。
系统中的换行:
- Windows系统里,每行结尾是 回车+换行 ,即 \r\n ;
- Unix系统里,每行结尾只有 换行 ,即 \n ;
- Mac系统里,每行结尾是 回车 ,即 \r 。从 Mac OS X开始与Linux统一。
public class FileOutputStreamDemo {
public static void main(String[] args) throws IOException {
File file = new File("aaa.txt");
//构造器:该路径下,如果没有这个文件,会创建该文件。如果有这个文 件,会清空这个文件的数据。
FileOutputStream outputStream = new FileOutputStream(file);
FileOutputStream outputStream1 = new FileOutputStream("bbb.txt");
//数据追加续写
FileOutputStream outputStream2 = new FileOutputStream(file,true);
FileOutputStream outputStream3 = new FileOutputStream("ccc.txt",true);
//write(int b)
outputStream.write(97);
outputStream.write(98);
outputStream.write(99);
//write(byte[] b)
byte[] b = new byte[]{97,98,99,100,101};
byte[] b1 = "哈哈哈".getBytes();
// outputStream1.write(b);
for (int i = 0;i<b.length;i++){
outputStream.write(b[i]);
//换行
outputStream.write("\r\n".getBytes());
}
outputStream1.write(b1);
outputStream3.write(b1);
//write(byte[] b, int off, int len)
outputStream.write(b,1,3);
outputStream3.write(b,2,3);
outputStream.flush();
outputStream.close();
outputStream1.flush();
outputStream1.close();
}
}
字节输入流【InputStream】
java.io.InputStream 抽象类是表示字节输入流的所有类的超类。
- public void close() :关闭此输入流并释放与此流相关联的任何系统资源。
- public abstract int read() : 从输入流读取数据的下一个字节。
- public int read(byte[] b) : 从输入流中读取一些字节数,并将它们存储到字节数组 b中 。
close方法,当完成流的操作时,必须调用此方法,释放系统资源。
FileInputStream类
java.io.FileInputStream 类是文件输入流,从文件中读取字节。
构造方法
- FileInputStream(File file) : 通过打开与实际文件的连接来创建一个 FileInputStream ,该文件由文件系统中的 File对象 fifile命名。
- FileInputStream(String name) : 通过打开与实际文件的连接来创建一个 FileInputStream ,该文件由文件系统中的路径名 name命名。
当你创建一个流对象时,必须传入一个文件路径。该路径下,如果没有该文件,会抛出 FileNotFoundException 。
读取字节数据
- 读取字节: read 方法,每次可以读取一个字节的数据,提升为int类型,读取到文件末尾,返回 -1
- 虽然读取了一个字节,但是会自动提升为int类型。
- 流操作完毕后,必须释放系统资源,调用close方法,千万记得。
- 使用字节数组读取: read(byte[] b) ,每次读取b的长度个字节到数组中,返回读取到的有效字节个数,读取到末尾时,返回 -1 ,
使用数组读取,每次读取多个字节,减少了系统间的IO操作次数,从而提高了读写的效率,建议开发中使用。
流的关闭原则:先开后关,后开先关。
public class FileInputStreamDemo {
public static void main(String[] args) throws IOException {
//构造方法
File file = new File("aaa.txt");
FileInputStream inputStream = new FileInputStream(file);
FileInputStream inputStream1 = new FileInputStream("ccc.txt");
//read 方法,每次可以读取一个字节的数据,提升为int类型,读取到文件末尾,返回 -1
/*int read = inputStream.read();
System.out.println((char)read);
int read1 = inputStream.read();
System.out.println((char)read1);
int read2 = inputStream.read();
System.out.println((char)read2);*/
int a ;
while((a = inputStream.read())!=-1){
System.out.print((char)a);
}
//read(byte[] b) ,每次读取b的长度个字节到数组中,返回读取到的有效字节个数,读 取到末尾时,返回 -1
int len = 0;
byte[] b = new byte[1024];
while((len = inputStream.read(b))!=-1){
System.out.println(new String(b,0,len));
}
inputStream.close();
}
}
public class Exam {
public static void main(String[] args) throws IOException {
File file = new File("递归求和原理图.png");
File file1 = new File("递归求和原理图1.png");
FileInputStream inputStream = new FileInputStream(file);
FileOutputStream outputStream = new FileOutputStream(file1);
byte[] bytes = new byte[1024*8];
int len = 0;
while((len = inputStream.read(bytes))!=-1){
outputStream.write(bytes,0,len);
}
outputStream.flush();
outputStream.close();
inputStream.close();
}
}
字符流
遇到中文字符时,可能不会显示完整的字符,因为一个中文字符可能占用多个字节存储。所以Java提供一些字符流类,以字符为单位读写数据,专门用于处理文本文件。
字符输入流【Reader】
java.io.Reader 抽象类是表示用于读取字符流的所有类的超类
- public void close() :关闭此流并释放与此流相关联的任何系统资源。
- public int read() : 从输入流读取一个字符。
- public int read(char[] cbuf) : 从输入流中读取一些字符,并将它们存储到字符数组 cbuf中 。
FileReader类
java.io.FileReader 类是读取字符文件的便利类。构造时使用系统默认的字符编码和默认字节缓冲区。
字符编码:字节与字符的对应规则。Windows系统的中文编码默认是GBK编码表。idea中UTF-8
字节缓冲区:一个字节数组,用来临时存储字节数据
构造方法
- FileReader(File file) : 创建一个新的 FileReader ,给定要读取的File对象。
- FileReader(String fileName) : 创建一个新的 FileReader ,给定要读取的文件的名称。
读取字符数据
- 读取字符: read 方法,每次可以读取一个字符的数据,提升为int类型,读取到文件末尾,返回 -1 ,循环读取,
虽然读取了一个字符,但是会自动提升为int类型。
- 使用字符数组读取: read(char[] cbuf) ,每次读取b的长度个字符到数组中,返回读取到的有效字符个数,读取到末尾时,返回 -1 ,
public class FileReaderDemo {
public static void main(String[] args) throws IOException {
//构造方法
File file = new File("aaa.txt");
FileReader fileReader = new FileReader(file);
FileReader fileReader1 = new FileReader("ccc.txt");
//read 方法,每次可以读取一个字符的数据,提升为int类型,读取到文件末尾,返回 -1
int buffer = 0;
while((buffer = (fileReader1.read()))!=-1){
System.out.print((char)buffer);
}
System.out.println();
//read(char[] cbuf) ,每次读取b的长度个字符到数组中,返回读取到的有效字符个数,读取到末尾时,返回 -1
char[] chars = new char[2];
int len = 0;
while((len = fileReader.read(chars))!=-1){
System.out.print(new String(chars,0,len));
}
fileReader.close();
fileReader1.close();
}
}
字符输出流【Writer】
java.io.Writer 抽象类是表示用于写出字符流的所有类的超类,将指定的字符信息写出到目的地
- void write(int c) 写入单个字符。
- void write(char[] cbuf) 写入字符数组。
- abstract void write(char[] cbuf, int off, int len) 写入字符数组的某一部分,off数组的开始索引,len 写的字符个数。
- void write(String str) 写入字符串。
- void write(String str, int off, int len) 写入字符串的某一部分,off字符串的开始索引,len写的字符个 数。
- void flush() 刷新该流的缓冲。
- void close() 关闭此流,但要先刷新它。
FileWriter类
java.io.FileWriter 类是写出字符到文件的便利类。
构造方法
- FileWriter(File file) : 创建一个新的 FileWriter,给定要读取的File对象。
- FileWriter(String fileName) : 创建一个新的 FileWriter,给定要读取的文件的名称。
基本写出数据
写出字符: write(int b) 方法,每次可以写出一个字符数据
- 虽然参数为int类型四个字节,但是只会保留一个字符的信息写出。
- 未调用close方法,数据只是保存到了缓冲区,并未写出到文件中。
关闭和刷新
因为内置缓冲区的原因,如果不关闭输出流,无法写出字符到文件中。但是关闭的流对象,是无法继续写出数据的。如果我们既想写出数据,又想继续使用流,就需要 flush 方法了。
- flush :刷新缓冲区,流对象可以继续使用。
- close :先刷新缓冲区,然后通知系统释放资源。流对象不可以再被使用了
即便是flflush方法写出了数据,操作的最后还是要调用close方法,释放系统资源。
写出其他数据
-
写出字符数组 : write(char[] cbuf) 和 write(char[] cbuf, int off, int len) ,每次可以写出字符数 组中的数据,用法类似FileOutputStream,
-
写出字符串: write(String str) 和 write(String str, int off, int len) ,每次可以写出字符串中的数据,更为方便
-
续写和换行:操作类似于FileOutputStream。
字符流,只能操作文本文件,不能操作图片,视频等非文本文件。
当我们单纯读或者写文本文件时 使用字符流 其他情况使用字节流
public class FileWriterDemo {
public static void main(String[] args) throws IOException {
//构造方法
File file = new File("ddd.txt");
FileWriter fileWriter = new FileWriter(file);
FileWriter fileWriter1 = new FileWriter("eee.txt",true);
// write(int b) 方法,每次可以写出一个字符数据
fileWriter.write(15);
fileWriter.write(97);
fileWriter.write('A');
fileWriter.write("asdsxad");
//写出字符数组 : write(char[] cbuf) 和 write(char[] cbuf, int off, int len)
char[] chars = "哈哈哈你好".toCharArray();
// System.out.println(chars);
fileWriter1.write(chars,0,2);
//写出字符串: write(String str) 和 write(String str, int off, int len)
String msg = "Hello Java";
fileWriter1.write(msg);
fileWriter1.write("\r\n");
fileWriter1.write(msg,0,5);
fileWriter.flush();
fileWriter.flush();
fileWriter.close();
fileWriter1.close();
}
}
IO异常的处理
public class IOExceptionDemo {
public static void main(String[] args) {
FileReader fileReader = null;
FileWriter fileWriter = null;
try {
fileReader = new FileReader("aaa.txt");
fileWriter = new FileWriter("ggg.txt");
int len = 0;
char[] c = new char[1024];
while ((len=fileReader.read(c))!=-1){
fileWriter.write(new String(c,0,len));
fileWriter.flush();
}
} catch (FileNotFoundException e) {
e.printStackTrace();
}catch (IOException e) {
e.printStackTrace();
}finally {
try {
if(fileReader!=null){
fileReader.close();
}
if(fileWriter!=null){
fileWriter.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
属性集
java.util.Properties 继承于 Hashtable ,来表示一个持久的属性集。它使用键值结构存储数据,每个键及其对应值都是一个字符串。该类也被许多Java类使用,比如获取系统属性时, System.getProperties 方法就是返回一个 Properties 对象。
构造方法
- public Properties() :创建一个空的属性列表。
基本的存储方法
-
public Object setProperty(String key, String value) : 保存一对属性。
-
public String getProperty(String key) :使用此属性列表中指定的键搜索属性值。
-
public Set stringPropertyNames() :所有键的名称的集合。
-
put(Object key, Object value):将指定的Key映射到此散列表中指定的value
与流相关的方法
- public void load(InputStream inStream) : 从字节输入流中读取键值对。
- store(OutputStream out, String comments):将此Properties表中的此属性列表(键和元素对)以适合于使用load(Input Stream)方法加载到Properties表格的格式写入输出流。
Properties继承于Hashtable ,而Hashtable 实现Map接口,所以具有entrySet()、keySet()等方法。
文本中的数据,必须是键值对形式,可以使用空格、等号、冒号等符号分隔。
public class PropertiesDemo {
public static void main(String[] args) throws IOException {
Properties properties = new Properties();
// properties.setProperty("姓名","李四");
// properties.setProperty("年龄","18");
// properties.setProperty("性别","男");
//
// System.out.println(properties);
//
// System.out.println(properties.getProperty("姓名"));
// System.out.println(properties.getProperty("年龄"));
// System.out.println(properties.getProperty("性别"));
//
// Set<String> strings = properties.stringPropertyNames();
// for (String string:strings
// ) {
// System.out.println(string+"----"+properties.getProperty(string));
// }
FileReader reader = new FileReader("properties.txt");
properties.load(reader);
Set<Map.Entry<Object, Object>> entries = properties.entrySet();
for (Map.Entry<Object, Object> entry:entries) {
System.out.println(entry.getKey()+"---"+entry.getValue());
}
properties.put("name","zhangbaizhi");
properties.put("age","40");
properties.put("sex","nv");
FileWriter writer = new FileWriter("properties1.txt");
properties.store(writer,"author:XXX data:2020-08-07");
}
}