IO流
UTF-8 BOM
BOM含义:BOM(Byte Order Mark),字节顺序标记,出现在文本文件头部,Unicode编码标准中用于标识文件是采用哪种格式的编码。
BOM签名的意思就是告诉编辑器当前文件采用何种编码,方便编辑器识别。
在我们通常使用的windows系统中,我发现了一个有趣的现象。我新建一个空的文本文档,点击文件-另存为-编码选择UTF-8,然后保存。此时这个文件明明是空的,却占了3字节大小。原因在于:此时保存的编码方式自动会变为UTF-8 BOM
BOM即byte order mark,具体含义可百度百科或维基百科,UTF-8文件中放置BOM主要是微软的习惯,但是放在别的系统上会出现问题。不含BOM的UTF-8才是标准形式,UTF-8不需要BOM带BOM的UTF-8文件的开头会有U+FEFF,所以我新建的空文件会有3字节的大小。
1.File类
File类在java中表示(带路径的)文件或者目录。
1.1 File常用属性和方法
public static void main(String[] args) { // 给定路径创建File对象 // File file = new File("D:"+File.separator+"javatest"+File.separator+"a.txt"); File file = new File("d:\\javatest\\b.mp3"); System.out.println(file); // 文件基本属性 System.out.println(file.canExecute()); System.out.println(file.canRead()); System.out.println(file.canWrite()); // 文件的创建、删除 if(!file.exists()) { boolean r; try { r = file.createNewFile(); if(r) { System.out.println("文件创建成功"); } } catch (IOException e) { e.printStackTrace(); } } // 删除文件 file.delete(); }
1.2 File的路径相关
1 public static void main(String[] args) { 2 3 File file = new File("d:\\javatest\\a"); 4 // File file = new File("a.txt"); 5 6 // 获取file的绝对路径 7 System.out.println(file.getAbsolutePath()); 8 // 获取file的创建时的路径字符串 9 System.out.println(file.getPath()); 10 // 获取文件或者目录的名字 11 System.out.println(file.getName()); 12 // 获取文件或者目录的父目录 13 System.out.println(file.getParent()); 14 15 }
注意:如果file是相对路径,相对路径的当前路径是工程目录(java17)
1.3 目录的创建
1 public static void main(String[] args) { 2 3 File file = new File("d:\\javatest\\c\\d\\e"); 4 5 if(!file.exists()) { 6 boolean r; 7 8 try { 9 // 一次只能创建一个目录 10 // r = file.mkdir(); 11 r = file.mkdirs(); 12 if(r) { 13 System.out.println("目录创建成功"); 14 } 15 } catch (Exception e) { 16 e.printStackTrace(); 17 } 18 19 } 20 }
1.4目录的遍历
list():返回一个file表示的目录中的子目录或者文件,字符串数组类型
listFiles():返回一个file表示的目录中的子目录或者文件,File数组类型
1 public static void main(String[] args) { 2 3 // 需求:遍历d:\javatest目录 4 // list() 5 File file = new File("d:\\javatest"); 6 7 8 /* 9 String[] list = file.list(); 10 11 for (String str : list) { 12 System.out.print(str); 13 File f = new File(file.getPath()+"\\"+str); 14 if(f.isDirectory()) { 15 System.out.println(" 目录"); 16 }else { 17 System.out.println(" 文件"); 18 } 19 }*/ 20 21 22 // listFiles(); 23 File[] listFiles = file.listFiles(); 24 for (File f : listFiles) { 25 System.out.print(f.getName()); 26 if(f.isDirectory()) { 27 System.out.println(" 目录"); 28 }else { 29 System.out.println(" 文件"); 30 } 31 } 32 }
2.IO流
流(stream):流是一连串流动的数据(字节、字符),以先进先出的方式发送的信息的通道中。
1.1.1 输入流和输出流
输入流
数据从源数据源流入程序的过程称为输入流。可以理解为从源数据源读取数据到程序的过程
输出流
数据从程序流出到目的地的过程称为输出流。可以理解为把数据从程序写入目的地的过程
数据源一般指提供数据的原始媒介,一般常见有文件、数据库、云端、其他硬件等能提供数据的媒介。
2.1 流的分类
按照流向分为输入流和输出流
按照处理单元分为字节流和字符流
按照功能分为节点流和转换流。
2.2 InputStream/OutputStream
InputStream 是所有字节输入流的抽象父类,提供了
read 读取一个字节
read(byte[] buf) 读取一定量的字节到缓冲区数组 buf中。
OutputStream 是所有字节输出流的抽象父类,提供了
write() 写入一个字节
write(byte[] buf) 写入一定量的字节到输出流
FileInputStream 文件字节输入流,专门用于从文件中读取字节到程序内存中。
FileOutputStream 文件字节输出流,专门用于从内存中写入字节到文件中。
需求:从文件读取一个字节
1 public static void main(String[] args) { 2 3 // 需求:读取一个文件中的一个字节 4 File file = new File("d:\\javatest\\a.txt"); 5 6 // 【1】创建管道 7 FileInputStream in = null; 8 9 try { 10 in = new FileInputStream(file); 11 12 // 【2】从管道读取一个字节 13 /* 14 int t; 15 t = in.read(); 16 t = in.read(); 17 t = in.read(); 18 t = in.read(); 19 */ 20 // System.out.println(t); 21 22 // 循环读取一个字节 23 int t; 24 StringBuilder sb = new StringBuilder(); 25 while( (t=in.read()) != -1 ) { 26 sb.append((char)t); 27 } 28 29 System.out.println(sb.toString()); 30 31 32 33 } catch (FileNotFoundException e) { 34 e.printStackTrace(); 35 } catch(IOException e) { 36 e.printStackTrace(); 37 } 38 39 // 【3】关闭流管道 40 try { 41 in.close(); 42 } catch (IOException e) { 43 e.printStackTrace(); 44 } 45 }
一次读取多个字节
1 public static void main(String[] args) { 2 3 // 需求:一次读取多个字节 4 File file = new File("d:\\javatest\\a.txt"); 5 6 // 【1】创建管道 7 FileInputStream in = null; 8 9 try { 10 in = new FileInputStream(file); 11 12 // 【2】从管道读取多个字节到缓冲区 13 /* 14 byte[] buf = new byte[5]; 15 int len; 16 len = in.read(buf); 17 len = in.read(buf); 18 len = in.read(buf); 19 len = in.read(buf); 20 21 for(byte b:buf) { 22 System.out.print((char)b+"\t"); 23 } 24 System.out.println(len); 25 */ 26 27 // 通过循环读取文件 28 byte[] buf = new byte[5]; 29 int len; 30 StringBuilder sb = new StringBuilder(); 31 while( (len=in.read(buf)) != -1 ) { 32 // 读取的内容是原始二进制流,需要根据编码的字符集解码成对于字符 33 String str = new String(buf,0,len); 34 sb.append(str); 35 } 36 System.out.println(sb.toString()); 37 38 39 40 41 } catch (FileNotFoundException e) { 42 e.printStackTrace(); 43 } catch(IOException e) { 44 e.printStackTrace(); 45 } 46 47 // 【3】关闭流管道 48 try { 49 in.close(); 50 } catch (IOException e) { 51 e.printStackTrace(); 52 } 53 }
需求:按照指定编码写入文件
1 public static void main(String[] args) { 2 3 4 File file = new File("d:\\javatest\\c.txt"); 5 6 FileOutputStream out = null; 7 8 try { 9 // 【1】创建输出流管道 10 out = new FileOutputStream(file); 11 12 // 【2】写入数据到管道中 13 // 一次写入一个字节 14 /* 15 out.write(97); 16 out.write(98); 17 out.write(99); 18 */ 19 20 // 一次写入多个字节 21 String str = "hello world"; 22 // gbk 23 /* 24 byte[] buf = str.getBytes(); 25 out.write(buf); 26 */ 27 28 byte[] buf = str.getBytes("UTF-8"); 29 out.write(buf); 30 31 System.out.println("写入完成!"); 32 33 } catch (FileNotFoundException e) { 34 e.printStackTrace(); 35 } catch (IOException e) { 36 e.printStackTrace(); 37 } 38 39 // 【3】关闭流 40 try { 41 out.close(); 42 } catch (IOException e) { 43 e.printStackTrace(); 44 } 45 }
注意:
[1]字符串写入文件时一定会存在编码问题
[2]使用utf8编码写入文件时,如果不含中文时,win系统会对文件的编码造成误判。
[3] 通过字节流写入文件时,向管道写入一个字节,该字节立即写入文件中。
总结
InputStream/OutputStream 用于字节的读写。主要用于读取二进制文件(图片、音频、视频),也可以读取文件性文件。
需求:请把d:\\javatest\\logo.png 复制到工程目录中,并显示复制进度。
1 public static void main(String[] args) throws FileNotFoundException,IOException { 2 3 4 File oriFile = new File("d:\\javatest\\logo.jpg"); 5 File toFile = new File("logo.jpg"); 6 7 long totalLen = oriFile.length(); // 文件大小 8 long cpyedLen = 0; // 已复制完成的大小 9 float progress = 0.0f; 10 11 FileInputStream in = new FileInputStream(oriFile); 12 FileOutputStream out = new FileOutputStream(toFile); 13 14 // 一次读取1kb 15 byte[] buf = new byte[512]; 16 int len; 17 while( (len=in.read(buf)) != -1) { 18 out.write(buf, 0, len); 19 cpyedLen += len; 20 progress = cpyedLen*1.0f/totalLen; 21 System.out.println(progress); 22 23 24 } 25 26 in.close(); 27 out.close(); 28 29 System.out.println("复制完成!"); 30 31 }
2.3 Reader/Writer
Reader 是字符输入流的抽象父类,提供了
read 一次读取一个字符
read(char[] cbuf) 一次读取多个字符到字符缓冲区cbuf,返回长度表示读取的字符个数。
Writer 是字符输出流的抽象父类,提供了
write
write(char[] cbuf)
write(string)
FileReader 文件字符输入流,专门用于读取默认字符编码文本性文件。
FileWriter 文件字符输出流,专门用于写入默认字符编码的文本性文件。为了提高效率,FileWriter内部存在一个字节缓冲区,用于对待写入的字符进行统一编码到字节缓冲区,一定要在关闭流之前,调用flush方法刷新缓冲区。
需求:一次读取一个字符/多个字符到cbuf
1 public static void main(String[] args) throws IOException { 2 3 File file = new File("d:\\javatest\\d.txt"); 4 5 FileReader reader = new FileReader(file); 6 7 // 【1】一次读取一个字符 8 /* 9 int c; 10 c = reader.read(); 11 c = reader.read(); 12 c = reader.read(); 13 c = reader.read(); 14 c = reader.read(); 15 System.out.println((char)c); 16 */ 17 18 // 【2】一次读取多个字符到cbuf中 19 /* 20 char[] cbuf = new char[2]; 21 int len; 22 len = reader.read(cbuf); 23 len = reader.read(cbuf); 24 len = reader.read(cbuf); 25 len = reader.read(cbuf); 26 System.out.println(Arrays.toString(cbuf)); 27 System.out.println(len); 28 */ 29 30 char[] cbuf = new char[2]; 31 int len; 32 StringBuilder sb = new StringBuilder(); 33 while( (len=reader.read(cbuf)) != -1 ) { 34 sb.append(cbuf,0,len); 35 } 36 37 System.out.println(sb); 38 }
需求:写入字符到文件中
1 public static void main(String[] args) throws IOException { 2 3 4 File file = new File("d:\\javatest\\f.txt"); 5 6 FileWriter writer = new FileWriter(file); 7 8 // 【1】一次写入一个字符 9 /*writer.write('中'); 10 writer.write('国');*/ 11 12 // 【2】一次写入多个字符 13 /*char[] cbuf = {'h','e','l','l','o','中','国'}; 14 writer.write(cbuf);*/ 15 16 // 【3】一次写入一个字符串 17 String str = "hello你好"; 18 writer.write(str); 19 20 21 // 刷新字节缓冲区 22 writer.flush(); 23 24 // 关闭流通道 25 writer.close(); 26 27 System.out.println("写入完成"); 28 }
3 转换流
InputStreamReader 继承于Reader,是字节流通向字符流的桥梁,可以把字节流按照指定编码 解码 成字符流。
OutputStreamWriter 继承于Writer,是字符流通向字节流的桥梁,可以把字符流按照指定的编码 编码 成字节流。
3.1 转换流工作原理
需求:写入utf8文件
1 /** 2 * 把一个字符串以utf8编码写入文件 3 */ 4 public class Test01 { 5 public static void main(String[] args) throws IOException { 6 7 8 String str = "hello中国"; 9 File file = new File("d:\\javatest\\g.txt"); 10 11 // 【1】创建管道 12 FileOutputStream out = new FileOutputStream(file); 13 OutputStreamWriter writer = new OutputStreamWriter(out, "utf8"); 14 15 // 【2】写入管道 16 writer.write(str); 17 18 // 【3】刷新缓冲区 19 writer.flush(); 20 21 // 【4】关闭管道 22 out.close(); 23 writer.close(); 24 25 System.out.println("写入完成"); 26 } 27 }
需求:读取utf8文件
1 /**
2 * 读取utf8编码的文本文件
3 */
4 public class Test01 {
5 public static void main(String[] args) throws IOException {
6
7 File file = new File("d:\\javatest\\g.txt");
8
9 // 【1】建立管道
10 FileInputStream in = new FileInputStream(file);
11 InputStreamReader reader = new InputStreamReader(in, "UTF-8");
12
13 char[] cbuf = new char[2];
14 int len;
15
16 StringBuilder sb = new StringBuilder();
17 while( (len=reader.read(cbuf))!=-1 ) {
18 sb.append(cbuf, 0, len);
19 }
20 System.out.println(sb.toString());
21
22 }
23 }
注意:
[1]win平台默认的utf8编码的文本性文件带有BOM,java转换流写入的utf8文件不带BOM。所以用java读取手动创建的utf8文件会出现一点乱码(?hello中国,?是bom导致的)
[2] 一句话:用字符集编码,一定用字符集解码!!
FileReader = InputStreamReader + GBK
1 package cn.sxt07.outputstreamwriter; 2 3 import java.io.File; 4 import java.io.FileInputStream; 5 import java.io.FileReader; 6 import java.io.IOException; 7 import java.io.InputStreamReader; 8 9 /** 10 * 读取一个gbk编码的文本性文件 11 */ 12 public class Test02 { 13 public static void main(String[] args) throws IOException { 14 15 16 File file = new File("d:\\javatest\\f.txt"); 17 18 // 【1】建立管道 19 /* 20 * FileInputStream in = new FileInputStream(file); 21 * InputStreamReader reader = new InputStreamReader(in, "GBK"); 22 */ 23 24 FileReader reader = new FileReader(file); 25 26 char[] cbuf = new char[2]; 27 int len; 28 29 StringBuilder sb = new StringBuilder(); 30 while( (len=reader.read(cbuf))!=-1 ) { 31 sb.append(cbuf, 0, len); 32 } 33 34 reader.close(); 35 36 System.out.println(sb.toString()); 37 } 38 }
3.2 BufferedReader/BufferedWriter
BufferedReader 继承于Reader,提供了
read
read(char[] cbuf)
readLine() 用于读取一行文本,实现对文本的高效读取。
BufferedReader 初始化时需要一个reader,本质上BufferedReader在reader的基础上增加readLine()的功能。
BufferedWriter继承于Writer,提供了
write
write(char[] cbuf)
write(string)
newline() 写入一个行分隔符。
需求:读取一首诗
1 public static void main(String[] args) throws IOException { 2 3 // 按行读取gbk文本性文件 4 5 File file = new File("d:\\javatest\\i.txt"); 6 7 // 【1】创建管道 8 FileReader reader = new FileReader(file); 9 BufferedReader br = new BufferedReader(reader); 10 11 // 【2】读取一行 12 /* 13 String line = br.readLine(); 14 line = br.readLine(); 15 line = br.readLine(); 16 line = br.readLine(); 17 */ 18 19 String line; 20 while( (line=br.readLine()) != null) { 21 System.out.println(line); 22 } 23 }
需求:以gbk编码写入一首诗到文件
1 public static void main(String[] args) throws IOException { 2 3 File file = new File("d:\\javatest\\j.txt"); 4 5 // 【1】创建gbk管道 6 FileWriter writer = new FileWriter(file); 7 BufferedWriter bw = new BufferedWriter(writer); 8 9 // 【2】写入一行 10 bw.write("窗前明月光,"); 11 bw.newLine(); 12 13 bw.write("疑似地上霜。"); 14 15 // for win 16 // bw.write("\r\n"); 17 18 // for unix/linux/mac 19 // bw.write("\n"); 20 21 bw.write("举头望明月,"); 22 bw.newLine(); 23 24 // 【3】flush 25 bw.flush(); 26 27 // 【4】关闭管道 28 bw.close(); 29 writer.close(); 30 }
需求:以utf8编码高效写入文件
1 /** 2 * 以utf8写入一首诗 3 * @author Administrator 4 * 5 */ 6 public class Test02 { 7 public static void main(String[] args) throws IOException { 8 9 File file = new File("d:\\javatest\\j-utf8.txt"); 10 11 // 【1】创建utf8管道 12 FileOutputStream out = new FileOutputStream(file); 13 OutputStreamWriter writer = new OutputStreamWriter(out, "UTF-8"); 14 BufferedWriter bw = new BufferedWriter(writer); 15 16 // 【2】写入一行 17 bw.write("窗前明月光,"); 18 bw.newLine(); 19 20 bw.write("疑似地上霜。"); 21 22 // for win 23 bw.write("\r\n"); 24 25 // for unix/linux/mac 26 // bw.write("\n"); 27 28 bw.write("举头望明月,"); 29 bw.newLine(); 30 31 // 【3】flush 32 bw.flush(); 33 34 // 【4】关闭管道 35 bw.close(); 36 writer.close(); 37 } 38 }