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)
(6)複製文字檔案(5種方式)字元緩衝流的特殊方法:
* BufferedWriter:
* public void newLine():根據系統來決定換行符
* BufferedReader:
* public String readLine():一次讀取一行資料
* 包含該行內容的字串,不包含任何行終止符,如果已到達流末尾,則返回 null
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();
}
}