1、緩沖流
緩沖流槽驶,也叫高效流责嚷,是對(duì)4個(gè)基本的 FileXxx 流的增強(qiáng),所以也是4個(gè)流掂铐,按照數(shù)據(jù)類型分類:
- 字節(jié)緩沖流: BufferedInputStream 罕拂, BufferedOutputStream
- 字符緩沖流: BufferedReader , BufferedWriter
緩沖流的基本原理全陨,是在創(chuàng)建流對(duì)象時(shí)爆班,會(huì)創(chuàng)建一個(gè)內(nèi)置的默認(rèn)大小的緩沖區(qū)數(shù)組,通過(guò)緩沖區(qū)讀寫(xiě)辱姨,減少系統(tǒng)IO 次數(shù)柿菩,從而提高讀寫(xiě)的效率。
(1)字節(jié)緩沖流
java.io.BufferedOutputStream extends OutputStream
BufferedOutputStream:字節(jié)緩沖輸出流
java.io.BufferedInputStream extends InputStream
BufferedInputStream:字節(jié)緩沖輸入流
構(gòu)造方法:
- BufferedOutputStream(OutputStream out) 創(chuàng)建一個(gè)新的緩沖輸出流雨涛,以將數(shù)據(jù)寫(xiě)入指定的底層輸出流枢舶。
- BufferedInputStream(InputStream in) 創(chuàng)建一個(gè) BufferedInputStream 并保存其參數(shù),即輸入流 in替久,以便將來(lái)使用凉泄。
代碼實(shí)現(xiàn):
public void testBufferedStream() throws IOException {
// 創(chuàng)建流對(duì)象
String src = "E:\\javatest文件\\src\\1.png";
BufferedInputStream bis = new BufferedInputStream(new FileInputStream(src));
String des = "E:\\javatest文件\\des\\bis.png";
BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(des));
//int read(byte[] b) 從輸入流中讀取一定數(shù)量的字節(jié),并將其存儲(chǔ)在緩沖區(qū)數(shù)組 b 中侣肄。
byte[] bytes = new byte[1024];//存儲(chǔ)每次讀取的數(shù)據(jù)
int len = 0; //記錄每次讀取的有效字節(jié)個(gè)數(shù)
while ((len = bis.read(bytes)) != -1) {
bos.write(bytes, 0, len);
}
//釋放資源
bos.close();
bis.close();
}
(2)字符緩沖流
java.io.BufferedWriter extends Writer
BufferedWriter:字符緩沖輸出流
java.io.BufferedReader extends Reader
BufferedReader:字符緩沖輸入流
構(gòu)造方法:
- BufferedWriter(Writer out) 創(chuàng)建一個(gè)使用默認(rèn)大小輸出緩沖區(qū)的緩沖字符輸出流旧困。
- BufferedReader(Reader in) 創(chuàng)建一個(gè)使用默認(rèn)大小輸入緩沖區(qū)的緩沖字符輸入流。
特有方法:
- BufferedWriter: void newLine() 寫(xiě)入一個(gè)行分隔符稼锅。會(huì)根據(jù)不同的操作系統(tǒng),獲取不同的行分隔符吼具。
- BufferedReader: String readLine() 讀取一個(gè)文本行。讀取一行數(shù)據(jù)行的終止符號(hào):通過(guò)換行 ('\n')矩距、回車(chē) ('\r') 或回車(chē)后直接跟著換行(\r\n)字符之一即可認(rèn)為某行已終止拗盒。
代碼實(shí)現(xiàn):
/*
3.侍中、侍郎郭攸之锥债、費(fèi)祎陡蝇、董允等痊臭,此皆良實(shí),志慮忠純登夫,是以先帝簡(jiǎn)拔以遺陛下广匙。愚以為宮中之事,事無(wú)大小恼策,悉 以咨之鸦致,然后施行,必得裨補(bǔ)闕漏涣楷,有所廣益分唾。
2.宮中府中,俱為一體狮斗,陟罰臧否绽乔,不宜異同。若有作奸犯科及為忠善者碳褒,宜付有司論其刑賞折砸,以昭陛下平明之理,不 宜偏私沙峻,使內(nèi)外異法也鞍爱。
1.先帝創(chuàng)業(yè)未半而中道崩殂,今天下三分专酗,益州疲弊睹逃,此誠(chéng)危急存亡之秋也。然侍衛(wèi)之臣不懈于內(nèi)祷肯,忠志之士忘身于外 者沉填,蓋追先帝之殊遇,欲報(bào)之于陛下也佑笋。誠(chéng)宜開(kāi)張圣聽(tīng)翼闹,以光先帝遺德,恢弘志士之氣蒋纬,不宜妄自菲薄猎荠,引喻失義,以 塞忠諫之路也蜀备。
*/
public void testBuffered() throws IOException {
//1.創(chuàng)建一個(gè)HashMap集合對(duì)象,可以:存儲(chǔ)每行文本的序號(hào)(1,2,3,..);value:存儲(chǔ)每行的文本
HashMap<String, String> map = new HashMap<>();
//2.創(chuàng)建字符緩沖輸入流對(duì)象,構(gòu)造方法中綁定字符輸入流
String src = "E:\\javatest文件\\src\\in.txt";
BufferedReader br = new BufferedReader(new FileReader(src));
//3.創(chuàng)建字符緩沖輸出流對(duì)象,構(gòu)造方法中綁定字符輸出流
String des = "E:\\javatest文件\\des\\out.txt";
BufferedWriter bw = new BufferedWriter(new FileWriter(des));
//4.使用字符緩沖輸入流中的方法readline,逐行讀取文本
String line;
while ((line = br.readLine()) != null) {
//5.對(duì)讀取到的文本進(jìn)行切割,獲取行中的序號(hào)和文本內(nèi)容
String[] arr = line.split("\\.");
//6.把切割好的序號(hào)和文本的內(nèi)容存儲(chǔ)到HashMap集合中(key序號(hào)是有序的,會(huì)自動(dòng)排序1,2,3,4..)
map.put(arr[0], arr[1]);
}
//7.遍歷HashMap集合,獲取每一個(gè)鍵值對(duì)
for (String key : map.keySet()) {
String value = map.get(key);
//8.把每一個(gè)鍵值對(duì),拼接為一個(gè)文本行
line = key + "." + value;
//9.把拼接好的文本,使用字符緩沖輸出流中的方法write,寫(xiě)入到文件中
bw.write(line);
bw.newLine();//寫(xiě)換行
}
//10.釋放資源
bw.close();
br.close();
}
2关摇、轉(zhuǎn)換流
(1)字符編碼和字符集
- 字符編碼 Character Encoding:就是一套自然語(yǔ)言的字符與二進(jìn)制數(shù)之間的對(duì)應(yīng)規(guī)則。
- 字符集 Charset:也叫編碼表碾阁。是一個(gè)系統(tǒng)支持的所有字符的集合输虱,包括各國(guó)家文字、標(biāo)點(diǎn)符號(hào)脂凶、圖形符 號(hào)宪睹、數(shù)字等愁茁。
常見(jiàn)字符 集有ASCII字符集、GBK字符集亭病、Unicode字符集等鹅很。
- ASCII碼表:0~ 9(48~ 57)、A~ Z(65~ 90)罪帖、a~ z(97~ 122)
- GBK:最常用的中文碼表道宅。是在GB2312(簡(jiǎn)體中文碼表)標(biāo)準(zhǔn)基礎(chǔ)上的擴(kuò)展規(guī)范,使用了雙字節(jié)編碼方案胸蛛。
- UTF-8:它使用一至四個(gè)字節(jié)為每個(gè)字符編碼。
編碼規(guī)則:①128個(gè)US-ASCII字符樱报,只需一個(gè)字節(jié)編碼葬项。 ②拉丁文等字符,需要二個(gè)字節(jié)編碼迹蛤。 ③大部分常用字(含中文)民珍,使用三個(gè)字節(jié)編碼。④其他極少使用的Unicode輔助字符盗飒,使用四字節(jié)編碼嚷量。
(2)InputStreamReader和OutputStreamWriter
java.io.OutputStreamWriter extends Writer
OutputStreamWriter:是字符流通向字節(jié)流的橋梁:可使用指定的 charset 將要寫(xiě)入流中的字符編碼成字節(jié)。(編碼:把能看懂的變成看不懂)
java.io.InputStreamReader extends Reader
InputStreamReader:是字節(jié)流通向字符流的橋梁:它使用指定的 charset 讀取字節(jié)并將其解碼為字符逆趣。(解碼:把看不懂的變成能看懂的)
構(gòu)造方法:
- OutputStreamWriter(OutputStream out):創(chuàng)建使用默認(rèn)字符編碼的 OutputStreamWriter蝶溶。
- OutputStreamWriter(OutputStream out, String charsetName): 創(chuàng)建使用指定字符集的 OutputStreamWriter。
- InputStreamReader(InputStream in) 創(chuàng)建一個(gè)使用默認(rèn)字符集的 InputStreamReader宣渗。
- InputStreamReader(InputStream in, String charsetName) 創(chuàng)建使用指定字符集的 InputStreamReader抖所。
小貼士:String charsetName表示指定的編碼表名稱崔涂,不區(qū)分大小寫(xiě)辽狈,可以是utf-8/UTF-8、gbk/GBK竭缝、...不指定默認(rèn)使用UTF-8鞍恢。
代碼實(shí)現(xiàn):
/*將GBK編碼的文本文件傻粘,轉(zhuǎn)換為UTF-8編碼的文本文件。*/
public void testReverseStream() throws IOException {
//1.創(chuàng)建InputStreamReader對(duì)象,構(gòu)造方法中傳遞字節(jié)輸入流和指定的編碼表名稱GBK
String src = "E:\\javatest文件\\src\\gbk.txt";
InputStreamReader isr = new InputStreamReader(new FileInputStream(src), "GBK");
//2.創(chuàng)建OutputStreamWriter對(duì)象,構(gòu)造方法中傳遞字節(jié)輸出流和指定的編碼表名稱UTF-8
String des = "E:\\javatest文件\\des\\utf8.txt";
OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream(des), "UTF-8");
//3.使用InputStreamReader對(duì)象中的方法read讀取文件
char[] cs = new char[1024];//存儲(chǔ)讀取到的多個(gè)字符
int len = 0;//記錄的是每次讀取的有效字符個(gè)數(shù)
while ((len = isr.read(cs)) != -1) {
//4.使用OutputStreamWriter對(duì)象中的方法write,把讀取的數(shù)據(jù)寫(xiě)入到文件中
osw.write(cs, 0, len);
}
//5.釋放資源
osw.close();
isr.close();
}
3帮掉、序列化流
(1)ObjectOutputStream和ObjectInputStream
java.io.ObjectOutputStream extends OutputStream
ObjectOutputStream(對(duì)象的序列化流):把對(duì)象以流的方式寫(xiě)入到文件中保存弦悉。
java.io.ObjectInputStream extends InputStream
ObjectInputStream(對(duì)象的反序列化流):把文件中保存的對(duì)象,以流的方式讀取出來(lái)使用蟆炊。
構(gòu)造方法:
- ObjectOutputStream(OutputStream out) 創(chuàng)建寫(xiě)入指定 OutputStream 的 ObjectOutputStream警绩。
- ObjectInputStream(InputStream in) 創(chuàng)建從指定 InputStream 讀取的 ObjectInputStream。
特有方法:
- ObjectOutputStream:void writeObject(Object obj) 將指定的對(duì)象寫(xiě)入ObjectOutputStream盅称。
- ObjectInputStream:Object readObject() 從ObjectInputStream讀取對(duì)象肩祥。
序列化操作:
一個(gè)對(duì)象要想序列化后室,必須滿足兩個(gè)條件:
- 該類必須實(shí)現(xiàn) java.io.Serializable 接口, Serializable是一個(gè)標(biāo)記接口混狠,不實(shí)現(xiàn)此接口的類將不會(huì)使任何狀態(tài)序列化或反序列化岸霹,會(huì)拋出 NotSerializableException。
- 該類的所有屬性必須是可序列化的将饺。如果有一個(gè)屬性不需要可序列化的贡避,則該屬性必須注明是瞬態(tài)的,使用 transient 關(guān)鍵字修飾予弧。
反序列化操作:
對(duì)于JVM可以反序列化對(duì)象刮吧,它必須是能夠找到class文件的類。
如果找不到該類的class文件掖蛤,則拋出一個(gè) ClassNotFoundException 異常杀捻。
如果能找到class文件,但是class文件在序列化對(duì)象之后發(fā)生了修改蚓庭,那么反序列化操 作也會(huì)失敗致讥,拋出一個(gè) InvalidClassException 異常。
發(fā)生這個(gè)異常的原因如下:
- 該類的序列版本號(hào)與從流中讀取的類描述符的版本號(hào)不匹配
- 該類包含未知數(shù)據(jù)類型
- 該類沒(méi)有可訪問(wèn)的無(wú)參數(shù)構(gòu)造方法
小貼士:
Serializable 接口給需要序列化的類器赞,提供了一個(gè)序列版本號(hào)垢袱。 serialVersionUID 該版本號(hào)的目的在于驗(yàn)證序列化的對(duì)象和對(duì)應(yīng)類是否版本匹配。
被static修飾的成員變量不能被序列化的港柜,序列化的都是對(duì)象请契。
被transient修飾成員變量,不能被序列化夏醉。
代碼實(shí)現(xiàn):
/*將GBK編碼的文本文件姚糊,轉(zhuǎn)換為UTF-8編碼的文本文件。*/
public class Person implements Serializable {
private static final long serialVersionUID = 1L;
private String name;
//被transient修飾成員變量,不能被序列化
private transient String sex;
private int age;
public Person() {
}
public Person(String name, String sex, int age) {
this.name = name;
this.sex = sex;
this.age = age;
}
//省略toString方法和get授舟、set方法
}
public void testObjectStream() throws IOException, ClassNotFoundException {
//1.定義一個(gè)存儲(chǔ)Person對(duì)象的ArrayList集合
ArrayList<Person> list = new ArrayList<>();
//2.往ArrayList集合中存儲(chǔ)Person對(duì)象
list.add(new Person("張三", "男", 18));
list.add(new Person("李四", "男", 19));
list.add(new Person("王五", "男", 20));
//3.創(chuàng)建一個(gè)序列化流ObjectOutputStream對(duì)象
String pathName = "E:\\javatest文件\\obj.txt";
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(pathName));
//4.使用ObjectOutputStream對(duì)象中的方法writeObject,對(duì)集合進(jìn)行序列化
oos.writeObject(list);
//5.創(chuàng)建一個(gè)反序列化ObjectInputStream對(duì)象
ObjectInputStream ois = new ObjectInputStream(new FileInputStream(pathName));
//6.使用ObjectInputStream對(duì)象中的方法readObject讀取文件中保存的集合
Object o = ois.readObject();
//7.把Object類型的集合轉(zhuǎn)換為ArrayList類型
ArrayList<Person> list2 = (ArrayList<Person>) o;
//8.遍歷ArrayList集合
for (Person p : list2) {
System.out.println(p);
//打印輸出:
//Person{name='張三', sex='null', age=18}
//Person{name='李四', sex='null', age=19}
//Person{name='王五', sex='null', age=20}
}
//9.釋放資源
ois.close();
oos.close();
}
4救恨、打印流
(1)PrintStream
java.io.PrintStream extends OutputStream
PrintStream(打印流):為其他輸出流添加了功能,使它們能夠方便地打印各種數(shù)據(jù)值表示形式释树。
構(gòu)造方法:
- PrintStream(File file):輸出的目的地是一個(gè)文件
- PrintStream(OutputStream out):輸出的目的地是一個(gè)字節(jié)輸出流肠槽。
- InputStreamReader(InputStream in) 創(chuàng)建一個(gè)使用默認(rèn)字符集的 InputStreamReader。
- PrintStream(String fileName):輸出的目的地是一個(gè)文件路徑奢啥。
改變打印流向:
System.out 就是 PrintStream 類型的秸仙,只不過(guò)它的流向是系統(tǒng)規(guī)定的,打印在控制臺(tái)上桩盲。既然是流對(duì)象寂纪, 我們就可以改變它的流向。
使用System.setOut方法改變輸出語(yǔ)句的目的地改為參數(shù)中傳遞的打印流的目的地:static void setOut(PrintStream out)重新分配“標(biāo)準(zhǔn)”輸出流。
注意:
如果使用繼承自父類的write方法寫(xiě)數(shù)據(jù)捞蛋,那么查看數(shù)據(jù)的時(shí)候會(huì)查詢編碼表 97->a孝冒。
如果使用自己特有的方法print/println方法寫(xiě)數(shù)據(jù),寫(xiě)的數(shù)據(jù)原樣輸出 97->97拟杉。
代碼實(shí)現(xiàn):
public void testPrintStream() throws IOException {
//創(chuàng)建打印流PrintStream對(duì)象,構(gòu)造方法中綁定要輸出的目的地
String pathName = "E:\\javatest文件\\print.txt";
PrintStream ps = new PrintStream(pathName);
//如果使用繼承自父類的write方法寫(xiě)數(shù)據(jù),那么查看數(shù)據(jù)的時(shí)候會(huì)查詢編碼表 97->a
ps.write(97);
ps.write("\r\n".getBytes());
//如果使用自己特有的方法print/println方法寫(xiě)數(shù)據(jù),寫(xiě)的數(shù)據(jù)原樣輸出 97->97
ps.println(97);
ps.println(8.8);
ps.println('a');
ps.println("HelloWorld");
ps.println(true);
System.out.println("在控制臺(tái)輸出");
System.setOut(ps);//把輸出語(yǔ)句的目的地改變?yōu)榇蛴×鞯哪康牡? System.out.println("在打印流的目的地中輸出");
//釋放資源
ps.close();
}