java進階(33)--IO流
一、IO流概念:
1、基本概念
2、IO流分類
3、java.io流的四大家族
4、流的close和flush方法
5、java.id下常用的16個流
二、FileInputStream位元組輸入流
1、FileInputStream完成位元組流輸入的流程
2、使用while優化FileInputStream讀取流程
3、FileInputStream最終版,使用byte陣列讀取
4、FileInputStream的available
5、FileInputStream的skip方法
三、FileOutputStream位元組輸出流
1、FileOutputStream概念
2、FileOutputStream輸出流程
3、改進FileInputStream+String輸出
4、檔案複製(FileInputStream+FileOutputStream)
四、FileReader與FileWrite
1、FileReader概念
2、舉例FileReader
3、FileWrite概念:
4、舉例FileWrite:
5、複製普通檔案(FileReader與FileWrite):
五、帶有緩衝區的字元流
1、BufferedReader概念:
2、舉例說明BufferedReader:
3、節點流與包裝流
4、BufferedWrite:帶有緩衝區的字元輸出流
六、資料流
1、DataOutputStream概念
2、舉例說明:DataOutputStream
3、DataInputStream流概念:
4、舉例說明:DataInputStream
七、標準輸出流
1、PrintStream標準位元組輸出流:
2、更改標準輸出流的輸出方向
3、日誌工具生成原理:
八、File類
https://www.cnblogs.com/mrwhite2020/p/14322392.html
九、物件流
https://www.cnblogs.com/mrwhite2020/p/14322446.html
十、IO和Properties
https://www.cnblogs.com/mrwhite2020/p/14322463.html
------------------------------------------分割線,以下為正文-------------------------------------------------
一、IO流概念:
1、基本概念
I:Input
O:Output
通過IO可以完成硬碟檔案的讀和寫。
2、IO流分類
(1)按照流的方向分類:
輸入流(Input)或者稱為讀(Reader)
輸出流(Output)或者稱為寫(Write)
(2)按照讀取資料方式不同分類:
位元組流:按照位元組方式讀取資料,一次讀取1個位元組byte,等同一個8個二進位制位,這種流是萬能的,什麼型別的檔案都可以讀取,包括:文字檔案、圖片、視訊、聲音等
字元流:按照字元方式讀取資料,一次讀取一個字元,這種流是為了方便讀取普通檔案而存在,這種流不能讀取圖片、聲音、視訊、word等,只能讀取純文字檔案。
假設檔案test.txt 內容為:"h世界"
位元組流讀取:第一次讀'h'(佔用一個位元組),第二次讀'世'字元的一半(佔用一個位元組)
字元流讀取:第一次讀'h'(佔用一個位元組),第二次讀'世'字元(佔用一個字元)
3、java.io流的四大家族
(1)java.io.InputStream位元組輸入流
(2)java.io.OutputStream位元組輸出流
(3)java.io.Reader字元輸入流
(4)java.io.Writer字元輸出流
在java中以Stream結尾都是位元組流,以Reader/Writer結尾都是字元流
4、流的close和flush方法
(1)close()
所有的流都實現java.io.closeble介面,都是可以關閉的,都右close方法。
流是一個管理,是記憶體與硬碟之間的通道,用完之後一定要關閉,不然會耗費很多資源,養成好習慣,用完流一定要關閉。
(2)flush()
所有的輸出流都實現了java.io.Flushable介面,都可以重新整理,都包含flush方法。
養成好習慣,輸出流輸出玩都需要flush重新整理一下,表示將通道/管道當中的剩餘未輸出的資料強行輸出完,即清空管道,沒有使用flush方法會導致資料丟失
5、java.io下常用的16個流
(1)檔案專屬:
java.io.FileInputStream;
java.io.FileOutputStream;
java.io.FileReader;
java.io.FileWriter;
(2)轉換流:位元組流轉為字元流
java.io.InputStream;
java.io.OutputStream;
(3)緩衝區專屬:
java.io.BufferedInputStream;
java.io.BufferedOutputStream;
java.io.BufferedReader;
java.io.BufferedWriter;
(4)資料專屬:
java.io.DataInputStream;
java.io.DataOutputStream;
(5)標準輸出流:
java.io.PrintStream;
java.io.PrintWriter;
(6)物件專屬:
java.io.ObjectInputStream;
java.io.ObjectOutputStream;
二、FileInputStream位元組輸入流
1、FileInputStream完成位元組流輸入的流程
(1)準備檔案如下:
(2)建立位元組流輸入物件,Idea自動將路徑變斜槓
(3)alt+回車新增try+catch或者上報異常,這裡選擇try+catch
(4)第二種檔案路徑編寫方式"\"與"//"等價
(5)增加finally語句且初始化流放在try+catch語句外,並新增流的關閉,流關閉前需要增加流的判空ifn可預設生成 if(fis==null)
(6)增加流的讀取,並新增try+catch使用alt+回車,這裡建議選擇第一種異常
第一種:增加catch使用細粒度的異常
第二種:方法宣告異常
第三種:替換外層FIleNotFount為IO異常
第四種:內層在增加try+catch
(7)檢視列印結果為a對應的字元ascII碼值97
(8)可重複讀取6次,最後一次讀取不到返回-1
2、使用while優化FileInputStream讀取流程
3、FileInputStream最終版,使用byte陣列讀取
4、FileInputStream的available
(1)檢視剩餘的位元組數
(2)available的作用:可以不適用while迴圈,直接一次讀取全部的位元組,但是不適合大的檔案,因為byte陣列不能太大
5、FileInputStream的skip方法
三、FileOutputStream位元組輸出流
1、FileOutputStream概念:
位元組輸出流,從記憶體到硬碟
2、FileOutputStream輸出流程
(1)使用byte陣列+write方法+flush方法寫入
(2)檢查檔案結果,相對路徑為專案根目錄下:
(3)再次執行,會將原檔案中內容覆蓋,依然輸出:abcdab
(4)修改構造方法,true代表文字的追加
(5)檢視相對路徑下的文字執行結果:文字追加了內容
3、改進FileInputStream+String輸出
(1)使用String轉byte陣列輸出
(2)檢視輸出結果,追加一段String字串
4、檔案複製(FileInputStream+FileOutputStream)
(1)流程:拷貝過程是一邊讀、一邊寫;檔案型別任意,是萬能的
(2)舉例檔案複製:
1 package JAVAADVANCE; 2 3 import java.io.FileInputStream; 4 import java.io.FileNotFoundException; 5 import java.io.FileOutputStream; 6 import java.io.IOException; 7 8 public class TestAdvance33IOTest08 { 9 public static void main(String[] args) { 10 FileInputStream fis=null; 11 FileOutputStream fos=null; 12 try { 13 //建立一個輸入物件流 14 fis=new FileInputStream("D:\\javaTest\\inFile\\甜蜜家園第01集.mp4"); 15 //建立一個輸出物件流 16 fos=new FileOutputStream("D:\\javaTest\\outFile\\甜蜜家園第01集.mp4"); 17 //準備一個byte陣列,1024byte=1KB,*1024=1M,一次最多讀1M 18 byte[] bytes=new byte[1024*1024]; 19 int readCount=0; 20 while ((readCount=fis.read(bytes))!=-1) 21 { 22 //讀取多少,寫多少 23 fos.write(bytes,0,readCount); 24 } 25 fos.flush(); 26 } catch (FileNotFoundException e) { 27 e.printStackTrace(); 28 } catch (IOException e) { 29 e.printStackTrace(); 30 } finally { 31 //fos與fis的關閉分開try catch比較好,避免互相影響,有流沒有關閉 32 if (fos != null) { 33 try { 34 fos.close(); 35 } catch (IOException e) { 36 e.printStackTrace(); 37 } 38 } 39 if(fis!=null){ 40 try { 41 fis.close(); 42 } catch (IOException e) { 43 e.printStackTrace(); 44 } 45 } 46 } 47 48 49 50 51 } 52 }
檢視檔案輸出結果
\
四、FileReader與FileWrite
1、FileReader概念
檔案字元輸入流,只能讀取普通文字,讀取文字內容時比較方便、快捷
2、舉例FileReader
1 package JAVAADVANCE; 2 import java.io.*; 3 public class TestAdvance33IOTest09FileReader { 4 public static void main(String[] args) { 5 FileReader reader= null; 6 try { 7 //建立字元輸入流 8 reader=new FileReader("C:\\Users\\Mr.White\\IdeaProjects\\javaAdvance\\myTestFile01"); 9 //開始讀,使用char陣列,一次讀取4個字元 10 char[] chars=new char[4]; 11 int readCount=0; 12 while ((readCount=reader.read(chars))!=-1){ 13 System.out.println(new String(chars,0,readCount)); 14 } 15 } catch (FileNotFoundException e) { 16 e.printStackTrace(); 17 } catch (IOException e) { 18 e.printStackTrace(); 19 } finally { 20 if (reader != null) { 21 try { 22 reader.close(); 23 } catch (IOException e) { 24 e.printStackTrace(); 25 } 26 } 27 } 28 } 29 }
檢查原始檔與讀取的內容
3、FileWrite概念:
檔案字元輸出流,只能輸出普通文字
4、舉例FileWrite:
1 package JAVAADVANCE; 2 import java.io.FileWriter; 3 import java.io.IOException; 4 public class TestAdvance33IOTest10FileWrite { 5 public static void main(String[] args) { 6 FileWriter out = null; 7 try { 8 //建立字元輸出流物件,true使用追加寫入模式,否則重新執行會覆蓋 9 out=new FileWriter("fileWriteTest01",true); 10 //開始寫,使用char陣列 11 char[] chars={'我','是','中','國','人'}; 12 //寫入全部char陣列 13 out.write(chars); 14 //再寫入部分char陣列,第3個字元開始,3個字元 15 out.write(chars,2,3); 16 //直接寫入String 17 out.write("我是個JAVA工程師"); 18 out.flush(); 19 } catch (IOException e) { 20 e.printStackTrace(); 21 }finally { 22 if (out != null) { 23 try { 24 out.close(); 25 } catch (IOException e) { 26 e.printStackTrace(); 27 } 28 } 29 } 30 } 31 }
檢視輸出結果,相對路徑為工程根目錄下
5、複製普通檔案(FileReader與FileWrite):
(1)使用FileReader與FileWrite進行拷貝,只能拷貝普通文字檔案
(2)舉例複製普通檔案:
1 package JAVAADVANCE; 2 import java.io.FileNotFoundException; 3 import java.io.FileReader; 4 import java.io.FileWriter; 5 import java.io.IOException; 6 public class TestAdvance33IOTest11NormalFileCopy { 7 public static void main(String[] args) { 8 FileReader in =null; 9 FileWriter out = null; 10 try { 11 //建立字元輸入流 12 in=new FileReader("D:\\javaTest\\inFile\\fileWriteTest01"); 13 //建立字元輸出流 14 out=new FileWriter("D:\\javaTest\\outFile\\fileWriteTest01"); 15 //一邊讀一邊寫 16 char[] chars=new char[1024*512]; //一次讀取1M檔案 17 int readCount=0; 18 while ((readCount=in.read(chars))!=-1){ 19 out.write(new String(chars,0,readCount)); 20 } 21 //寫完後重新整理 22 out.flush(); 23 } catch (FileNotFoundException e) { 24 e.printStackTrace(); 25 } catch (IOException e) { 26 e.printStackTrace(); 27 } finally { 28 if (in != null) { 29 try { 30 in.close(); 31 } catch (IOException e) { 32 e.printStackTrace(); 33 } 34 } 35 if (out != null) { 36 try { 37 out.close(); 38 } catch (IOException e) { 39 e.printStackTrace(); 40 } 41 } 42 } 43 } 44 }
檢視複製後結果
五、帶有緩衝區的字元流
1、BufferedReader概念:
帶有緩衝區的字元輸入流,使用此流不需要自定義char陣列,或者說不需要自定義byte陣列,自帶緩衝區。
2、舉例說明BufferedReader:
(1)準備檔案
(2)按照行讀取檔案
package JAVAADVANCE; import java.io.*; public class TestAdvance33IOTest12BufferedReader { //異常先丟擲 public static void main(String[] args) throws IOException { FileReader reader=new FileReader("D:\\javaTest\\inFile\\fileWriteTest02"); //FileReader為節點流,BufferedReader為包裝流/處理流 BufferedReader br=new BufferedReader(reader); //讀第一行 String firstLine = br.readLine(); System.out.println(firstLine); //讀第二行 String secondLine = br.readLine(); System.out.println(secondLine); //讀第三行 String thirdLine = br.readLine(); System.out.println(thirdLine); //只需要關閉最外層的流,對於包裝流來說 br.close(); } }
(3)迴圈讀取全部檔案
1 package JAVAADVANCE; 2 import java.io.BufferedReader; 3 import java.io.FileReader; 4 import java.io.IOException; 5 public class TestAdvance33IOTest12BufferedReader02 { 6 //異常先丟擲 7 public static void main(String[] args) throws IOException { 8 FileReader reader=new FileReader("D:\\javaTest\\inFile\\fileWriteTest02"); 9 //FileReader為節點流,BufferedReader為包裝流/處理流 10 BufferedReader br=new BufferedReader(reader); 11 //使用String字串讀取 12 String s=null; 13 while ((s=br.readLine())!=null){ 14 System.out.println(s); 15 } 16 //只需要關閉最外層的流,對於包裝流來說 17 br.close(); 18 } 19 }
檢查執行結果
(4)print方法測試readLine方法不帶換行符
1 package JAVAADVANCE; 2 import java.io.BufferedReader; 3 import java.io.FileReader; 4 import java.io.IOException; 5 public class TestAdvance33IOTest12BufferedReader02 { 6 //異常先丟擲 7 public static void main(String[] args) throws IOException { 8 FileReader reader=new FileReader("D:\\javaTest\\inFile\\fileWriteTest02"); 9 //FileReader為節點流,BufferedReader為包裝流/處理流 10 BufferedReader br=new BufferedReader(reader); 11 //使用String字串讀取 12 String s=null; 13 while ((s=br.readLine())!=null){ 14 //print方法測試readLine方法不帶換行符 15 System.out.print(s); 16 } 17 //只需要關閉最外層的流,對於包裝流來說 18 br.close(); 19 } 20 }
檢視執行結果
3、節點流與包裝流
(1)節點流與包裝流都是相對而言
(2)舉例說明:節點流與包裝流
以下例子:
in與reader:in為節點流,reader為包裝流
reader與br:reader與節點流,br為包裝流
package JAVAADVANCE; import java.io.*; public class TestAdvance33IOTest12BufferedReader03 { //異常先丟擲 public static void main(String[] args) throws IOException { FileInputStream in=new FileInputStream("D:\\javaTest\\inFile\\fileWriteTest02"); //通過InputStreamReader轉換流將位元組流轉換為字元流 InputStreamReader reader=new InputStreamReader(in); BufferedReader br =new BufferedReader(reader); //使用String字串讀取 String line=null; while ((line=br.readLine())!=null){ //print方法測試readLine方法不帶換行符 System.out.println(line); } //只需要關閉最外層的流,對於包裝流來說 br.close(); } }
檢視執行結果:
4、BufferedWrite:帶有緩衝區的字元輸出流
(1)舉例說明BufferedWrite
package JAVAADVANCE; import java.io.*; public class TestAdvance33IOTest13BufferedWrite { //異常先丟擲 public static void main(String[] args) throws IOException { BufferedWriter out=new BufferedWriter(new FileWriter("D:\\javaTest\\outFile\\fileWriteTest02")); out.write("hello world"); out.write("\nhello kitty"); out.flush(); out.close(); } }
檢視執行結果
(2)使用包裝流追加輸出
package JAVAADVANCE; import java.io.*; public class TestAdvance33IOTest13BufferedWrite02 { //異常先丟擲 public static void main(String[] args) throws IOException { BufferedWriter out=new BufferedWriter(new OutputStreamWriter(new FileOutputStream("D:\\javaTest\\outFile\\fileWriteTest02",true))); out.write("\n這是繼續使用包裝流追加的文字"); out.flush(); out.close(); } }
檢視執行結果:
六、資料流
1、DataOutputStream概念:
資料專屬的流
這個流可以將資料連同資料的型別一併寫入檔案
注意:這個檔案不同普通文字文件(這個檔案使用記事本打不開)
2、舉例說明:DataOutputStream
package JAVAADVANCE; import java.io.*; public class TestAdvance33IOTest14DataOutputStream01 { //異常先丟擲 public static void main(String[] args) throws IOException { //建立資料專屬位元組輸出流 DataOutputStream dos=new DataOutputStream(new FileOutputStream("D:\\javaTest\\outFile\\fileWriteTest03")); byte b=100; short s=200; int i=300; float t=400F; double d=3.14; boolean sex=false; char c='a'; dos.writeByte(b); dos.writeShort(s); dos.writeInt(i); dos.writeFloat(t); dos.writeDouble(d); dos.writeBoolean(sex); dos.writeChar(c); dos.flush(); dos.close(); } }
檢視檔案結果:無法使用記事本或notepad++開啟,需要使用DataInputStream流開啟,並且讀的順序需要與寫的順序一致。
3、DataInputStream流概念:
資料位元組輸入流,DataOutputStream流寫的檔案只能使用DataInputStream開啟,並且讀的順序需要和寫的順序一致。
4、舉例說明:DataInputStream
1 package JAVAADVANCE; 2 import java.io.*; 3 public class TestAdvance33IOTest15DataInputStream01 { 4 //異常先丟擲 5 public static void main(String[] args) throws IOException { 6 //建立資料專屬位元組輸入流 7 DataInputStream dis =new DataInputStream(new FileInputStream("D:\\javaTest\\outFile\\fileWriteTest03")); 8 //開始讀,讀的順序要與寫的時候保持一致。 9 byte b= dis.readByte(); 10 short s=dis.readShort(); 11 int i=dis.readInt(); 12 float t =dis.readFloat(); 13 double d=dis.readDouble(); 14 boolean sex=dis.readBoolean(); 15 char c=dis.readChar(); 16 dis.close(); 17 System.out.println(b); 18 System.out.println(s); 19 System.out.println(i); 20 System.out.println(t); 21 System.out.println(d); 22 System.out.println(sex); 23 System.out.println(c); 24 25 } 26 }
檢視顯示結果
七、標準輸出流
1、PrintStream標準位元組輸出流:
標準位元組輸出流,預設輸出到控制檯。
1 package JAVAADVANCE; 2 import java.io.IOException; 3 import java.io.PrintStream; 4 public class TestAdvance33IOTest16PrintStream01 { 5 //異常先丟擲 6 public static void main(String[] args) throws IOException { 7 //標準輸出流直接輸出到控制檯 8 System.out.println("hello world"); 9 //換一種寫法 10 PrintStream ps=System.out; 11 ps.println("hello zhangsan"); 12 ps.println("hello lisi"); 13 //標準輸出流不需要手動關閉 14 } 15 }
2、更改標準輸出流的輸出方向
1 package JAVAADVANCE; 2 import java.io.FileOutputStream; 3 import java.io.IOException; 4 import java.io.PrintStream; 5 public class TestAdvance33IOTest16PrintStream01 { 6 //異常先丟擲 7 public static void main(String[] args) throws IOException { 8 //更改標準輸出流的輸出方向,指向log檔案 9 System.setOut(new PrintStream(new FileOutputStream("log"))); 10 //再次輸出 11 System.out.println("hello world"); 12 System.out.println("hello kitty"); 13 } 14 }
檢視專案根目錄的輸出檔案
3、日誌工具生成原理:
package JAVAADVANCE; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.PrintStream; import java.text.SimpleDateFormat; import java.util.Date; public class TestAdvance33IOTest16LogTest { //異常先丟擲 public static void log(String msg) { try { PrintStream out=new PrintStream(new FileOutputStream("log.txt",true)); //改變檔案的輸出方向 System.setOut(out); //當前日期格式 Date nowTime = new Date(); SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss SSS"); String strTime=sdf.format(nowTime); } catch (FileNotFoundException e) { e.printStackTrace(); } } }
編寫測試程式開始測試日誌列印
1 package JAVAADVANCE; 2 public class TestAdvance33IOTest16LogTest { 3 public static void main(String[] args) { 4 //測試工具是否好用 5 TestAdvance33IOTest16PrintStream02.log("呼叫了Sytem的gc()方法,建議啟動垃圾回收"); 6 TestAdvance33IOTest16PrintStream02.log("呼叫了userService的dosome()方法"); 7 TestAdvance33IOTest16PrintStream02.log("使用者正在嘗試登入,驗證失敗"); 8 } 9 }
檢視輸出結果
八、File類
File類參照如下章節:
https://www.cnblogs.com/mrwhite2020/p/14322392.html
九、物件流
物件流、序列化與反序列化參考如下章節:
https://www.cnblogs.com/mrwhite2020/p/14322446.html
十、IO和Properties
IO和Properties聯合使用參考如下章節:
https://www.cnblogs.com/mrwhite2020/p/14322463.html
&n