一、IO流
Java中I/O操作主要是指使用java.io
包下的内容,进行输入、输出操作。输入、输出是相对于内存来说的。
类型 | 输入流 | 输出流 |
---|---|---|
字节流 | InputStream | OutputStream |
字符流 | Reader | Writer |
字节流按照字节为单位进行读取,字符流按照字符为单位进行读取。
1.字符流是根据什么判断,多少个字节是一个字符?
答:根据字符流指定的编码表。Reader
和Writer
是系统默认编码集。
字符流可以理解为字节流按照编码表的规则来读写。
2.如果需要变更编码表,
Reader
和Writer
不能满足,应该如何操作?
答:方式一:使用DataOutputStream
和DataInputStream
。
示例:new DataOutputStream(fileOut).writeUTF("టుటోరియల్స్ పాయింట్ కి స్వాగతిం")
方式二:使用字节流+ 转换流。
建议文本文件使用字符流读写,但是使用字节流也可以(按照规律读写即可)。
public static void main(String[] args) throws Exception {
//测试使用字节流读取文本文件,a.txt的内容:“你好”
File file = new File("/Users/xuezengbo/ABO/myself/myCode/a.txt");
FileInputStream is = new FileInputStream(file);
byte[] bytes = new byte[3];
int len = 0;
while((len = is.read(bytes))!=-1){
//4.使用字节输出流中的方法write,把读取到的字节写入到目的地的文件中
System.out.println(new String(bytes,"UTF-8"));
}
is.close();
}
扫描器与IO流结合
public static void main(String[] args) throws Exception {
InputStream in = System.in;
Scanner scanner = new Scanner(in);
File file = new File("/Users/xuezengbo/ABO/myself/myCode/a.txt");
FileWriter writer = new FileWriter(file);
if (scanner.hasNext()) {
writer.write(scanner.next());
writer.flush();
}
//流的关闭需要按照倒序进行关闭,如果有缓冲或者转换流,只需要关闭这些高级流。
in.close();
writer.close();
}
二、字节流
OutputStream
抽象类是表示字节输出流的所有类的超类,将指定的字节信息写出到目的地。它定义了字节输出流的基本共性功能方法。
InputStream
抽象类是表示字节输入流的所有类的超类,可以读取字节信息到内存中。它定义了字节输入流的基本共性功能方法。
三、字符流
ASCII码文本文件在计算机中存储是二进制,使用文本编辑器打开的时候,编辑器会根据设置的编码集进行转码。
如:ASCII码的A在计算机中的二进制是0100 001,对应的十进制是65。
java.io.Reader
抽象类是表示用于读取字符流的所有类的超类,可以读取字符信息到内存中。它定义了字符输入流的基本共性功能方法。
java.io.Writer
抽象类是表示用于写出字符流的所有类的超类,将指定的字符信息写出到目的地。它定义了字节输出流的基本共性功能方法。
字节流读取中文的问题:1个中文(GBK:占用两个字节、UTF-8:占用3个字节)。
当使用字节流读取"你好" 的时候,字节流按照字节“逐个”进行读取,“你好”按照UTF-8编码表来说,占六个字节,字节流会读取六次,2个字被6次拆开读并打印出来,编辑器在UTF-8编码表中找不到被拆开后对应6的字符,故而出现乱码。
四、装饰流
4.1 缓冲流
缓冲流在创建流对象时,会创建一个内置的默认大小的缓冲区数组,通过缓冲区读写,减少系统IO次数,从而提高读写的效率。
-
字节缓冲流:
BufferedIn putStream
,BufferedOutputStream
自带的有8K的Buffer (即64个字节)。 -
字符缓冲流:
BufferedReader
,BufferedWriter
4.2 转换流
当要使用指定编码进行读取文本文件的时候,Reader
和Writer
是系统默认编码集读写,不能满足条件,这个时候只有字节流可以使用,使用字节流的时候需要InputStreamReader
和OutputStreamWriter
配合,可以指定编码集进行读写。
OutputStreamWriter:是字符流通向字节流的桥梁。
InputStreamReader:是字节流通向字符流的桥梁。
4.3 数据流
DataInputStream
和DataOutputStream
操作基本数据类型和字符串。
4.4 对象流
ObjectOutputStream
和ObjectInputStream
操作对象。
- 序列化和反序列化需要实现 java.io.Serializable 接口以启用其序列化功能。
- 被static(静态关键字)或transient(瞬态关键字)修饰的成员变量不能被序列化。
ObjectOutputStream:对象的序列化流
public class Demo01ObjectOutputStream {
public static void main(String[] args) throws IOException {
//1.创建ObjectOutputStream对象,构造方法中传递字节输出流
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("10_IO\\person.txt"));
//2.使用ObjectOutputStream对象中的方法writeObject,把对象写入到文件中
oos.writeObject(new Person("小美女",18));
//3.释放资源
oos.close();
}
}
ObjectInputStream:对象的反序列化流
public class Demo02ObjectInputStream {
public static void main(String[] args) throws IOException, ClassNotFoundException {
//1.创建ObjectInputStream对象,构造方法中传递字节输入流
ObjectInputStream ois = new ObjectInputStream(new FileInputStream("10_IO\\person.txt"));
//2.使用ObjectInputStream对象中的方法readObject读取保存对象的文件
Object o = ois.readObject();
//3.释放资源
ois.close();
//4.使用读取出来的对象(打印)
System.out.println(o);
Person p = (Person)o;
System.out.println(p.getName()+p.getAge());
}
}
五、属性集:Properties类
java.util.Properties
继承于Hashtable
,来表示一个持久的属性集。它使用键值结构存储数据,每个键及其对应值都是一个字符串。该类也被许多Java类使用,比如获取系统属性时,System.getProperties
方法就是返回一个Properties
对象。
Properties类就是一个集合,而且是一个持久的集合(保存在硬盘中),是唯一与IO流结合的集合。
package com.itheima.demo07.Prop;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.IOException;
import java.util.Properties;
import java.util.Set;
/*
java.util.Properties集合 extends Hashtable<k,v> implements Map<k,v>
Properties 类表示了一个持久的属性集。Properties 可保存在流中或从流中加载。
Properties集合是一个唯一和IO流相结合的集合
可以使用Properties集合中的方法store,把集合中的临时数据,持久化写入到硬盘中存储
可以使用Properties集合中的方法load,把硬盘中保存的文件(键值对),读取到集合中使用
属性列表中每个键及其对应值都是一个字符串。
Properties集合是一个双列集合,key和value默认都是字符串
*/
public class Demo01Properties {
public static void main(String[] args) throws IOException {
show03();
}
/*
可以使用Properties集合中的方法load,把硬盘中保存的文件(键值对),读取到集合中使用
void load(InputStream inStream)
void load(Reader reader)
参数:
InputStream inStream:字节输入流,不能读取含有中文的键值对
Reader reader:字符输入流,能读取含有中文的键值对
使用步骤:
1.创建Properties集合对象
2.使用Properties集合对象中的方法load读取保存键值对的文件
3.遍历Properties集合
注意:
1.存储键值对的文件中,键与值默认的连接符号可以使用=,空格(其他符号)
2.存储键值对的文件中,可以使用#进行注释,被注释的键值对不会再被读取
3.存储键值对的文件中,键与值默认都是字符串,不用再加引号
*/
private static void show03() throws IOException {
//1.创建Properties集合对象
Properties prop = new Properties();
//2.使用Properties集合对象中的方法load读取保存键值对的文件
prop.load(new FileReader("09_IOAndProperties\\prop.txt"));
//prop.load(new FileInputStream("09_IOAndProperties\\prop.txt"));
//3.遍历Properties集合
Set<String> set = prop.stringPropertyNames();
for (String key : set) {
String value = prop.getProperty(key);
System.out.println(key+"="+value);
}
}
/*
可以使用Properties集合中的方法store,把集合中的临时数据,持久化写入到硬盘中存储
void store(OutputStream out, String comments)
void store(Writer writer, String comments)
参数:
OutputStream out:字节输出流,不能写入中文
Writer writer:字符输出流,可以写中文
String comments:注释,用来解释说明保存的文件是做什么用的
不能使用中文,会产生乱码,默认是Unicode编码
一般使用""空字符串
使用步骤:
1.创建Properties集合对象,添加数据
2.创建字节输出流/字符输出流对象,构造方法中绑定要输出的目的地
3.使用Properties集合中的方法store,把集合中的临时数据,持久化写入到硬盘中存储
4.释放资源
*/
private static void show02() throws IOException {
//1.创建Properties集合对象,添加数据
Properties prop = new Properties();
prop.setProperty("赵丽颖","168");
prop.setProperty("迪丽热巴","165");
prop.setProperty("古力娜扎","160");
//2.创建字节输出流/字符输出流对象,构造方法中绑定要输出的目的地
//FileWriter fw = new FileWriter("09_IOAndProperties\\prop.txt");
//3.使用Properties集合中的方法store,把集合中的临时数据,持久化写入到硬盘中存储
//prop.store(fw,"save data");
//4.释放资源
//fw.close();
prop.store(new FileOutputStream("09_IOAndProperties\\prop2.txt"),"");
}
/*
使用Properties集合存储数据,遍历取出Properties集合中的数据
Properties集合是一个双列集合,key和value默认都是字符串
Properties集合有一些操作字符串的特有方法
Object setProperty(String key, String value) 调用 Hashtable 的方法 put。
String getProperty(String key) 通过key找到value值,此方法相当于Map集合中的get(key)方法
Set<String> stringPropertyNames() 返回此属性列表中的键集,其中该键及其对应值是字符串,此方法相当于Map集合中的keySet方法
*/
private static void show01() {
//创建Properties集合对象
Properties prop = new Properties();
//使用setProperty往集合中添加数据
prop.setProperty("赵丽颖","168");
prop.setProperty("迪丽热巴","165");
prop.setProperty("古力娜扎","160");
//prop.put(1,true);
//使用stringPropertyNames把Properties集合中的键取出,存储到一个Set集合中
Set<String> set = prop.stringPropertyNames();
//遍历Set集合,取出Properties集合的每一个键
for (String key : set) {
//使用getProperty方法通过key获取value
String value = prop.getProperty(key);
System.out.println(key+"="+value);
}
}
}
六、处理异常
实例1:
package com.itheima.demo06.trycatch;
import java.io.FileWriter;
import java.io.IOException;
public class Demo01TryCatch {
public static void main(String[] args) {
//提高变量fw的作用域,让finally可以使用
//变量在定义的时候,可以没有值,但是使用的时候必须有值
//fw = new FileWriter("09_IOAndProperties\\g.txt",true); 执行失败,fw没有值,fw.close会报错
FileWriter fw = null;
try{
//可能会产出异常的代码
fw = new FileWriter("w:\\09_IOAndProperties\\g.txt",true);
for (int i = 0; i <10 ; i++) {
fw.write("HelloWorld"+i+"\r\n");
}
}catch(IOException e){
//异常的处理逻辑
System.out.println(e);
}finally {
//一定会指定的代码
//创建对象失败了,fw的默认值就是null,null是不能调用方法的,会抛出NullPointerException,需要增加一个判断,不是null在把资源释放
if(fw!=null){
try {
//fw.close方法声明抛出了IOException异常对象,所以我们就的处理这个异常对象,要么throws,要么try catch
fw.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
实例2:
package com.itheima.demo06.trycatch;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
/*
JDK7的新特性
在try的后边可以增加一个(),在括号中可以定义流对象
那么这个流对象的作用域就在try中有效
try中的代码执行完毕,会自动把流对象释放,不用写finally
格式:
try(定义流对象;定义流对象....){
可能会产出异常的代码
}catch(异常类变量 变量名){
异常的处理逻辑
}
*/
public class Demo02JDK7 {
public static void main(String[] args) {
try(//1.创建一个字节输入流对象,构造方法中绑定要读取的数据源
FileInputStream fis = new FileInputStream("c:\\1.jpg");
//2.创建一个字节输出流对象,构造方法中绑定要写入的目的地
FileOutputStream fos = new FileOutputStream("d:\\1.jpg");){
//可能会产出异常的代码
//一次读取一个字节写入一个字节的方式
//3.使用字节输入流对象中的方法read读取文件
int len = 0;
while((len = fis.read())!=-1){
//4.使用字节输出流中的方法write,把读取到的字节写入到目的地的文件中
fos.write(len);
}
}catch (IOException e){
//异常的处理逻辑
System.out.println(e);
}
}
}
网友评论