1. 程式人生 > >21 API-IO流(字元流(編碼表,轉換流,FileReader,FileWriter,BufferedReader,BufferedWriter),IO流總結)

21 API-IO流(字元流(編碼表,轉換流,FileReader,FileWriter,BufferedReader,BufferedWriter),IO流總結)

1:字元流(掌握)

       (1)位元組流操作中文資料不是特別的方便,所以就出現了轉換流。

          轉換流的作用就是把位元組流轉換字元流來使用。

       (2)轉換流其實是一個字元流

              字元流 = 位元組流 + 編碼表


(3)編碼表


              A:就是由字元和對應的數值組成的一張表

              B:常見的編碼表

                     ASCII

                     ISO-8859-1

                     GB2312

                     GBK

                     GB18030

                     UTF-8

              C:字串中的編碼問題

                     編碼

                            String-- byte[]

                     解碼

                            byte[]-- String


(4)IO流中的編碼問題(轉換流)

              A:OutputStreamWriter

                     OutputStreamWriter(OutputStreamos):預設編碼,GBK

                     OutputStreamWriter(OutputStreamos,String charsetName):指定編碼。

              B:InputStreamReader

                     InputStreamReader(InputStreamis):預設編碼,GBK

                     InputStreamReader(InputStreamis,String charsetName):指定編碼

              C:編碼問題其實很簡單

                     編碼只要一致即可


(5)字元流

              Reader

                     |--InputStreamReader

                            |--FileReader

                     |--BufferedReader

              Writer

                     |--OutputStreamWriter

                            |--FileWriter

                     |--BufferedWriter

 * 由於我們常見的操作都是使用本地預設編碼,所以,不用指定編碼。

 * 而轉換流的名稱有點長,所以,Java就提供了其子類供我們使用。

 * OutputStreamWriter = FileOutputStream + 編碼表(GBK)

 * FileWriter = FileOutputStream + 編碼表(GBK)

 * 

 * InputStreamReader = FileInputStream + 編碼表(GBK)

 * FileReader = FileInputStream + 編碼表(GBK)

  字元緩衝流的特殊方法:

 * BufferedWriter:

 * public void newLine():根據系統來決定換行符

 * BufferedReader:

 * public String readLine():一次讀取一行資料

 * 包含該行內容的字串,不包含任何行終止符,如果已到達流末尾,則返回 null

(6)複製文字檔案(5種方式)
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;

/*
 * 複製文字檔案
 * 
 * 分析:
 * 		複製資料,如果我們知道用記事本開啟並能夠讀懂,就用字元流,否則用位元組流。
 * 		通過該原理,我們知道我們應該採用字元流更方便一些。
 * 		而字元流有5種方式,所以做這個題目我們有5種方式。推薦掌握第5種。
 * 資料來源:
 * 		c:\\a.txt -- FileReader -- BufferdReader
 * 目的地:
 * 		d:\\b.txt -- FileWriter -- BufferedWriter
 */
public class CopyFileDemo {
	public static void main(String[] args) throws IOException {
		String srcString = "c:\\a.txt";
		String destString = "d:\\b.txt";
		// method1(srcString, destString);
		// method2(srcString, destString);
		// method3(srcString, destString);
		// method4(srcString, destString);
		method5(srcString, destString);
	}

	// 字元緩衝流一次讀寫一個字串
	private static void method5(String srcString, String destString)
			throws IOException {
		BufferedReader br = new BufferedReader(new FileReader(srcString));
		BufferedWriter bw = new BufferedWriter(new FileWriter(destString));

		String line = null;
		while ((line = br.readLine()) != null) {
			bw.write(line);
			bw.newLine();
			bw.flush();
		}

		bw.close();
		br.close();
	}

	// 字元緩衝流一次讀寫一個字元陣列
	private static void method4(String srcString, String destString)
			throws IOException {
		BufferedReader br = new BufferedReader(new FileReader(srcString));
		BufferedWriter bw = new BufferedWriter(new FileWriter(destString));

		char[] chs = new char[1024];
		int len = 0;
		while ((len = br.read(chs)) != -1) {
			bw.write(chs, 0, len);
		}

		bw.close();
		br.close();
	}

	// 字元緩衝流一次讀寫一個字元
	private static void method3(String srcString, String destString)
			throws IOException {
		BufferedReader br = new BufferedReader(new FileReader(srcString));
		BufferedWriter bw = new BufferedWriter(new FileWriter(destString));

		int ch = 0;
		while ((ch = br.read()) != -1) {
			bw.write(ch);
		}

		bw.close();
		br.close();
	}

	// 基本字元流一次讀寫一個字元陣列
	private static void method2(String srcString, String destString)
			throws IOException {
		FileReader fr = new FileReader(srcString);
		FileWriter fw = new FileWriter(destString);

		char[] chs = new char[1024];
		int len = 0;
		while ((len = fr.read(chs)) != -1) {
			fw.write(chs, 0, len);
		}

		fw.close();
		fr.close();
	}

	// 基本字元流一次讀寫一個字元
	private static void method1(String srcString, String destString)
			throws IOException {
		FileReader fr = new FileReader(srcString);
		FileWriter fw = new FileWriter(destString);

		int ch = 0;
		while ((ch = fr.read()) != -1) {
			fw.write(ch);
		}

		fw.close();
		fr.close();
	}
}



2:IO流總結(掌握)


       IO流

              |--位元組流

                     |--位元組輸入流

                            InputStream

                                   intread():一次讀取一個位元組

                                   intread(byte[] bys):一次讀取一個位元組陣列

                                   |--FileInputStream

                                   |--BufferedInputStream

                     |--位元組輸出流

                            OutputStream

                                   voidwrite(int by):一次寫一個位元組

                                   voidwrite(byte[] bys,int index,int len):一次寫一個位元組陣列的一部分

                                   |--FileOutputStream

                                   |--BufferedOutputStream

              |--字元流

                     |--字元輸入流

                            Reader

                                   intread():一次讀取一個字元

                                   intread(char[] chs):一次讀取一個字元陣列

                                   |--InputStreamReader

                                          |--FileReader

                                   |--BufferedReader

                                          StringreadLine():一次讀取一個字串

                     |--字元輸出流

                            Writer

                                   voidwrite(int ch):一次寫一個字元

                                   voidwrite(char[] chs,int index,int len):一次寫一個字元陣列的一部分

                                   |--OutputStreamWriter

                                          |--FileWriter

                                   |--BufferedWriter

                                          voidnewLine():寫一個換行符

                                          voidwrite(String line):一次寫一個字串




3:案例(理解 )

A:複製文字檔案 5種方式(掌握)

B:複製圖片(二進位制流資料) 4種方式(掌握)

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;

/*
 * 複製圖片
 * 
 * 分析:
 * 		複製資料,如果我們知道用記事本開啟並能夠讀懂,就用字元流,否則用位元組流。
 * 		通過該原理,我們知道我們應該採用位元組流。
 * 		而位元組流有4種方式,所以做這個題目我們有4種方式。推薦掌握第4種。
 * 
 * 資料來源:
 * 		c:\\a.jpg -- FileInputStream -- BufferedInputStream
 * 目的地:
 * 		d:\\b.jpg -- FileOutputStream -- BufferedOutputStream
 */
public class CopyImageDemo {
	public static void main(String[] args) throws IOException {
		// 使用字串作為路徑
		// String srcString = "c:\\a.jpg";
		// String destString = "d:\\b.jpg";
		// 使用File物件做為引數
		File srcFile = new File("c:\\a.jpg");
		File destFile = new File("d:\\b.jpg");

		// method1(srcFile, destFile);
		// method2(srcFile, destFile);
		// method3(srcFile, destFile);
		method4(srcFile, destFile);
	}

	// 位元組緩衝流一次讀寫一個位元組陣列
	private static void method4(File srcFile, File destFile) throws IOException {
		BufferedInputStream bis = new BufferedInputStream(new FileInputStream(
				srcFile));
		BufferedOutputStream bos = new BufferedOutputStream(
				new FileOutputStream(destFile));

		byte[] bys = new byte[1024];
		int len = 0;
		while ((len = bis.read(bys)) != -1) {
			bos.write(bys, 0, len);
		}

		bos.close();
		bis.close();
	}

	// 位元組緩衝流一次讀寫一個位元組
	private static void method3(File srcFile, File destFile) throws IOException {
		BufferedInputStream bis = new BufferedInputStream(new FileInputStream(
				srcFile));
		BufferedOutputStream bos = new BufferedOutputStream(
				new FileOutputStream(destFile));

		int by = 0;
		while ((by = bis.read()) != -1) {
			bos.write(by);
		}

		bos.close();
		bis.close();
	}

	// 基本位元組流一次讀寫一個位元組陣列
	private static void method2(File srcFile, File destFile) throws IOException {
		FileInputStream fis = new FileInputStream(srcFile);
		FileOutputStream fos = new FileOutputStream(destFile);

		byte[] bys = new byte[1024];
		int len = 0;
		while ((len = fis.read(bys)) != -1) {
			fos.write(bys, 0, len);
		}

		fos.close();
		fis.close();
	}

	// 基本位元組流一次讀寫一個位元組
	private static void method1(File srcFile, File destFile) throws IOException {
		FileInputStream fis = new FileInputStream(srcFile);
		FileOutputStream fos = new FileOutputStream(destFile);

		int by = 0;
		while ((by = fis.read()) != -1) {
			fos.write(by);
		}

		fos.close();
		fis.close();
	}
}


C:把集合中的資料儲存到文字檔案

D:把文字檔案中的資料讀取到集合並遍歷集合

E:複製單級資料夾

F:複製單級資料夾中指定的檔案並修改名稱

回顧一下批量修改名稱

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.FilenameFilter;
import java.io.IOException;

/*
 * 需求:複製指定目錄下的指定檔案,並修改後綴名。
 * 指定的檔案是:.java檔案。
 * 指定的字尾名是:.jad
 * 指定的目錄是:jad
 * 
 * 資料來源:e:\\java\\A.java
 * 目的地:e:\\jad\\A.jad
 * 
 * 分析:
 * 		A:封裝目錄
 * 		B:獲取該目錄下的java檔案的File陣列
 * 		C:遍歷該File陣列,得到每一個File物件
 * 		D:把該File進行復制
 * 		E:在目的地目錄下改名
 */
public class CopyFolderDemo {
	public static void main(String[] args) throws IOException {
		// 封裝目錄
		File srcFolder = new File("e:\\java");
		// 封裝目的地
		File destFolder = new File("e:\\jad");
		// 如果目的地目錄不存在,就建立
		if (!destFolder.exists()) {
			destFolder.mkdir();
		}

		// 獲取該目錄下的java檔案的File陣列
		File[] fileArray = srcFolder.listFiles(new FilenameFilter() {
			@Override
			public boolean accept(File dir, String name) {
				return new File(dir, name).isFile() && name.endsWith(".java");
			}
		});

		// 遍歷該File陣列,得到每一個File物件
		for (File file : fileArray) {
			// System.out.println(file);
			// 資料來源:e:\java\DataTypeDemo.java
			// 目的地:e:\\jad\DataTypeDemo.java
			String name = file.getName();
			File newFile = new File(destFolder, name);
			copyFile(file, newFile);
		}

		// 在目的地目錄下改名
		File[] destFileArray = destFolder.listFiles();
		for (File destFile : destFileArray) {
			// System.out.println(destFile);
			// e:\jad\DataTypeDemo.java
			// e:\\jad\\DataTypeDemo.jad
			String name =destFile.getName(); //DataTypeDemo.java
			String newName = name.replace(".java", ".jad");//DataTypeDemo.jad
			
			File newFile = new File(destFolder,newName);
			destFile.renameTo(newFile);
		}
	}

	private static void copyFile(File file, File newFile) throws IOException {
		BufferedInputStream bis = new BufferedInputStream(new FileInputStream(
				file));
		BufferedOutputStream bos = new BufferedOutputStream(
				new FileOutputStream(newFile));

		byte[] bys = new byte[1024];
		int len = 0;
		while ((len = bis.read(bys)) != -1) {
			bos.write(bys, 0, len);
		}

		bos.close();
		bis.close();
	}
}


G:複製多級資料夾

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;

/*
 * 需求:複製多極資料夾
 * 
 * 資料來源:E:\JavaSE\day21\code\demos
 * 目的地:E:\\
 * 
 * 分析:
 * 		A:封裝資料來源File
 * 		B:封裝目的地File
 * 		C:判斷該File是資料夾還是檔案
 * 			a:是資料夾
 * 				就在目的地目錄下建立該資料夾
 * 				獲取該File物件下的所有檔案或者資料夾File物件
 * 				遍歷得到每一個File物件
 * 				回到C
 * 			b:是檔案
 * 				就複製(位元組流)
 */
public class CopyFoldersDemo {
	public static void main(String[] args) throws IOException {
		// 封裝資料來源File
		File srcFile = new File("E:\\JavaSE\\day21\\code\\demos");
		// 封裝目的地File
		File destFile = new File("E:\\");

		// 複製資料夾的功能
		copyFolder(srcFile, destFile);
	}

	private static void copyFolder(File srcFile, File destFile)
			throws IOException {
		// 判斷該File是資料夾還是檔案
		if (srcFile.isDirectory()) {
			// 資料夾
			File newFolder = new File(destFile, srcFile.getName());
			newFolder.mkdir();

			// 獲取該File物件下的所有檔案或者資料夾File物件
			File[] fileArray = srcFile.listFiles();
			for (File file : fileArray) {
				copyFolder(file, newFolder);
			}
		} else {
			// 檔案
			File newFile = new File(destFile, srcFile.getName());
			copyFile(srcFile, newFile);
		}
	}

	private static void copyFile(File srcFile, File newFile) throws IOException {
		BufferedInputStream bis = new BufferedInputStream(new FileInputStream(
				srcFile));
		BufferedOutputStream bos = new BufferedOutputStream(
				new FileOutputStream(newFile));

		byte[] bys = new byte[1024];
		int len = 0;
		while ((len = bis.read(bys)) != -1) {
			bos.write(bys, 0, len);
		}

		bos.close();
		bis.close();
	}
}


H:鍵盤錄入學生資訊按照總分從高到低儲存到文字檔案

學生類略

import java.io.BufferedWriter;
import java.io.FileWriter;
import java.io.IOException;
import java.util.Comparator;
import java.util.Scanner;
import java.util.TreeSet;

/*
 * 鍵盤錄入5個學生資訊(姓名,語文成績,數學成績,英語成績),按照總分從高到低存入文字檔案
 * 
 * 分析:
 * 		A:建立學生類
 * 		B:建立集合物件
 * 			TreeSet<Student>
 * 		C:鍵盤錄入學生資訊儲存到集合
 * 		D:遍歷集合,把資料寫到文字檔案
 */
public class StudentDemo {
	public static void main(String[] args) throws IOException {
		// 建立集合物件
		TreeSet<Student> ts = new TreeSet<Student>(new Comparator<Student>() {
			@Override
			public int compare(Student s1, Student s2) {
				int num = s2.getSum() - s1.getSum();
				int num2 = num == 0 ? s1.getChinese() - s2.getChinese() : num;
				int num3 = num2 == 0 ? s1.getMath() - s2.getMath() : num2;
				int num4 = num3 == 0 ? s1.getEnglish() - s2.getEnglish() : num3;
				int num5 = num4 == 0 ? s1.getName().compareTo(s2.getName())
						: num4;
				return num5;
			}
		});

		// 鍵盤錄入學生資訊儲存到集合
		for (int x = 1; x <= 5; x++) {
			Scanner sc = new Scanner(System.in);
			System.out.println("請錄入第" + x + "個的學習資訊");
			System.out.println("姓名:");
			String name = sc.nextLine();
			System.out.println("語文成績:");
			int chinese = sc.nextInt();
			System.out.println("數學成績:");
			int math = sc.nextInt();
			System.out.println("英語成績:");
			int english = sc.nextInt();

			// 建立學生物件
			Student s = new Student();
			s.setName(name);
			s.setChinese(chinese);
			s.setMath(math);
			s.setEnglish(english);

			// 把學生資訊新增到集合
			ts.add(s);
		}

		// 遍歷集合,把資料寫到文字檔案
		BufferedWriter bw = new BufferedWriter(new FileWriter("students.txt"));
		bw.write("學生資訊如下:");
		bw.newLine();
		bw.flush();
		bw.write("姓名,語文成績,數學成績,英語成績");
		bw.newLine();
		bw.flush();
		for (Student s : ts) {
			StringBuilder sb = new StringBuilder();
			sb.append(s.getName()).append(",").append(s.getChinese())
					.append(",").append(s.getMath()).append(",")
					.append(s.getEnglish());
			bw.write(sb.toString());
			bw.newLine();
			bw.flush();
		}
		// 釋放資源
		bw.close();
		System.out.println("學習資訊儲存完畢");
	}
}


I:把某個檔案中的字串排序後輸出到另一個文字檔案中

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.util.Arrays;

/*
 * 已知s.txt檔案中有這樣的一個字串:“hcexfgijkamdnoqrzstuvwybpl”
 * 請編寫程式讀取資料內容,把資料排序後寫入ss.txt中。
 * 
 * 分析:
 * 		A:把s.txt這個檔案給做出來
 * 		B:讀取該檔案的內容,儲存到一個字串中
 * 		C:把字串轉換為字元陣列
 * 		D:對字元陣列進行排序
 * 		E:把排序後的字元陣列轉換為字串
 * 		F:把字串再次寫入ss.txt中
 */
public class StringDemo {
	public static void main(String[] args) throws IOException {
		// 讀取該檔案的內容,儲存到一個字串中
		BufferedReader br = new BufferedReader(new FileReader("s.txt"));
		String line = br.readLine();
		br.close();

		// 把字串轉換為字元陣列
		char[] chs = line.toCharArray();

		// 對字元陣列進行排序
		Arrays.sort(chs);

		// 把排序後的字元陣列轉換為字串
		String s = new String(chs);

		// 把字串再次寫入ss.txt中
		BufferedWriter bw = new BufferedWriter(new FileWriter("ss.txt"));
		bw.write(s);
		bw.newLine();
		bw.flush();

		bw.close();
	}
}


J:用Reader模擬BufferedReader的特有功能

import java.io.IOException;
import java.io.Reader;

/*
 * 用Reader模擬BufferedReader的readLine()功能
 * 
 * readLine():一次讀取一行,根據換行符判斷是否結束,只返回內容,不返回換行符
 */
public class MyBufferedReader {
	private Reader r;

	public MyBufferedReader(Reader r) {
		this.r = r;
	}

	/*
	 * 思考:寫一個方法,返回值是一個字串。
	 */
	public String readLine() throws IOException {
		/*
		 * 我要返回一個字串,我該怎麼辦呢? 我們必須去看看r物件能夠讀取什麼東西呢? 兩個讀取方法,一次讀取一個字元或者一次讀取一個字元陣列
		 * 那麼,我們要返回一個字串,用哪個方法比較好呢? 我們很容易想到字元陣列比較好,但是問題來了,就是這個陣列的長度是多長呢?
		 * 根本就沒有辦法定義陣列的長度,你定義多長都不合適。 所以,只能選擇一次讀取一個字元。
		 * 但是呢,這種方式的時候,我們再讀取下一個字元的時候,上一個字元就丟失了 所以,我們又應該定義一個臨時儲存空間把讀取過的字元給儲存起來。
		 * 這個用誰比較和是呢?陣列,集合,字串緩衝區三個可供選擇。
		 * 經過簡單的分析,最終選擇使用字串緩衝區物件。並且使用的是StringBuilder
		 */
		StringBuilder sb = new StringBuilder();

		// 做這個讀取最麻煩的是判斷結束,但是在結束之前應該是一直讀取,直到-1
		
		
		/*
		hello
		world
		java	
		
		104101108108111
		119111114108100
		1069711897
		 */
		
		int ch = 0;
		while ((ch = r.read()) != -1) { //104,101,108,108,111
			if (ch == '\r') {
				continue;
			}

			if (ch == '\n') {
				return sb.toString(); //hello
			} else {
				sb.append((char)ch); //hello
			}
		}

		// 為了防止資料丟失,判斷sb的長度不能大於0
		if (sb.length() > 0) {
			return sb.toString();
		}

		return null;
	}

	/*
	 * 先寫一個關閉方法
	 */
	public void close() throws IOException {
		this.r.close();
	}
}


K:模擬LineNumberReader的特有功能

import java.io.IOException;
import java.io.Reader;

public class MyLineNumberReader {
	private Reader r;
	private int lineNumber = 0;

	public MyLineNumberReader(Reader r) {
		this.r = r;
	}

	public int getLineNumber() {
		// lineNumber++;
		return lineNumber;
	}

	public void setLineNumber(int lineNumber) {
		this.lineNumber = lineNumber;
	}

	public String readLine() throws IOException {
		lineNumber++;

		StringBuilder sb = new StringBuilder();

		int ch = 0;
		while ((ch = r.read()) != -1) {
			if (ch == '\r') {
				continue;
			}

			if (ch == '\n') {
				return sb.toString();
			} else {
				sb.append((char) ch);
			}
		}

		if (sb.length() > 0) {
			return sb.toString();
		}

		return null;
	}

	public void close() throws IOException {
		this.r.close();
	}
}