编码与解码

编码就是把字符转换为字节,而解码是把字节重新组合成字符。

如果编码和解码过程使用不同的编码方式那么就出现了乱码。

  • GBK 编码中,中文字符占 2 个字节,英文字符占 1 个字节;
  • UTF-8 编码中,中文字符占 3 个字节,英文字符占 1 个字节;
  • UTF-16be 编码中,中文字符和英文字符都占 2 个字节。

UTF-16be 中的 be 指的是 Big Endian,也就是大端。相应地也有 UTF-16le,le 指的是 Little Endian,也就是小端。

Java 的内存编码使用双字节编码 UTF-16be,这不是指 Java 只支持这一种编码方式,
而是说 char 这种类型使用 UTF-16be 进行编码。char 类型占 16 位,也就是两个字节,
Java 使用这种双字节编码是为了让一个中文或者一个英文都能使用一个 char 来存储。

String 的编码方式

String 可以看成一个字符序列,可以指定一个编码方式将它编码为字节序列,也可以指定一个编码方式将一个字节序列解码为 String。

  1. String str1 = "中文";
  2. byte[] bytes = str1.getBytes("UTF-8");
  3. String str2 = new String(bytes, "UTF-8");
  4. System.out.println(str2);

在调用无参数 getBytes() 方法时,默认的编码方式不是 UTF-16be。
双字节编码的好处是可以使用一个 char 存储中文和英文,
而将 String 转为 bytes[] 字节数组就不再需要这个好处,因此也就不再需要双字节编码
。getBytes() 的默认编码方式与平台有关,一般为 UTF-8。

  1. byte[] bytes = str1.getBytes();
  1. public class StringDemo {
  2. public static void main(String[] args) throws UnsupportedEncodingException {
  3. String s = "你好";
  4. // String -- byte[]
  5. //byte[] bys = s.getBytes(); // [-28, -67, -96, -27, -91, -67]
  6. //byte[] bys = s.getBytes("GBK");// [-60, -29, -70, -61]
  7. //byte[] bys = s.getBytes("UTF-8");// [-28, -67, -96, -27, -91, -67]
  8. byte[] bys = s.getBytes("UTF-16be");//[79, 96, 89, 125]
  9. System.out.println(Arrays.toString(bys));
  10. // byte[] -- String
  11. //String ss = new String(bys); // 你好
  12. //String ss = new String(bys, "GBK"); // 你好
  13. //String ss = new String(bys, "UTF-8"); // 你好
  14. String ss = new String(bys, "UTF-16be"); //你好
  15. System.out.println(ss);
  16. }
  17. }

Reader 与 Writer

不管是磁盘还是网络传输,最小的存储单元都是字节,而不是字符。
但是在程序中操作的通常是字符形式的数据,因此需要提供对字符进行操作的方法。

  • InputStreamReader 实现从字节流解码成字符流;
  • OutputStreamWriter 实现字符流编码成为字节流。

InputStreamReader

  • InputStreamReader的构造方法:
  1. InputStreamReader(InputStream is) //用默认的编码读取数据
  2. InputStreamReader(InputStream is,String charsetName) //用指定的编码读取数据
  1. public class ReaderDemo {
  2. public static void main(String[] args) throws IOException {
  3. InputStreamReader isr=new InputStreamReader(
  4. new FileInputStream("demo1.txt"),"UTF-8");
  5. int ch=0;
  6. while((ch=isr.read())!=-1){
  7. System.out.print((char)ch);
  8. }
  9. isr.close();
  10. }
  11. }
  • 读入字符
  1. int read() //一次读取一个字符
  2. int read(char[] chs) //一次读取一个字符数组
  1. public class ReaderDemo2 {
  2. public static void main(String[] args) throws IOException {
  3. InputStreamReader isr=new InputStreamReader(
  4. new FileInputStream("demo1.txt"),"UTF-8");
  5. // 一次读取一个字符
  6. // int ch = 0;
  7. // while ((ch = isr.read()) != -1) {
  8. // System.out.print((char) ch);
  9. // }
  10. // 一次读取一个字符数组
  11. char[] chs = new char[1024];
  12. int len = 0;
  13. while ((len = isr.read(chs)) != -1) {
  14. System.out.print(new String(chs, 0, len));
  15. }
  16. isr.close();
  17. }
  18. }

OutputStreamWriter

  • OutputStreamWriter的构造方法:
  1. OutputStreamWriter(OutputStream out) //根据默认编码把字节流的数据转换为字符流
  2. OutputStreamWriter(OutputStream out,String charsetName) //根据指定编码把字节流数据转换为字符流
  3. //字符流 = 字节流 +编码表。
  1. public class WriterDemo {
  2. public static void main(String[] args) throws IOException {
  3. OutputStreamWriter osw=new OutputStreamWriter(new FileOutputStream("demo1.txt"),"UTF-8");
  4. //直接写字符
  5. osw.write("你好 hello\r\n");
  6. osw.close();
  7. }
  8. }
  • 写出方法:
  1. public void write(int c) //写一个字符
  2. public void write(char[] cbuf) //写一个字符数组
  3. public void write(char[] cbuf,int off,int len) //写一个字符数组的一部分
  4. public void write(String str) //写一个字符串
  5. public void write(String str,int off,int len) //写一个字符串的一部分
  1. public class WriterDemo2 {
  2. public static void main(String[] args) throws IOException {
  3. OutputStreamWriter osw=new OutputStreamWriter(new FileOutputStream("demo1.txt"),"UTF-8");
  4. // 写数据
  5. // public void write(int c):写一个字符
  6. osw.write('a');
  7. osw.write(97);
  8. // 为什么数据没有进去呢?
  9. // 原因是:字符 = 2字节
  10. // 文件中数据存储的基本单位是字节。
  11. // void flush()
  12. // public void write(char[] cbuf):写一个字符数组
  13. // char[] chs = {'a','b','c','d','e'};
  14. // osw.write(chs);
  15. // public void write(char[] cbuf,int off,int len):写一个字符数组的一部分
  16. // osw.write(chs,1,3);
  17. // public void write(String str):写一个字符串
  18. // osw.write("我爱林青霞");
  19. // public void write(String str,int off,int len):写一个字符串的一部分
  20. osw.write("我爱林青霞", 2, 3);
  21. // 刷新缓冲区
  22. osw.flush();
  23. // osw.write("我爱林青霞", 2, 3);
  24. // 释放资源
  25. osw.close();
  26. // java.io.IOException: Stream closed
  27. // osw.write("我爱林青霞", 2, 3);
  28. }
  29. }

注意:close()和flush()的区别?

  1. close()关闭流对象,但是先刷新一次缓冲区。关闭之后,流对象不可以继续再使用了。
  2. flush()仅仅刷新缓冲区,刷新之后,流对象还可以继续使用。

BufferedReader

从字符输入流中读取文本,缓冲各个字符,从而实现字符、数组和行的高效读取。
可以指定缓冲区的大小,或者可使用默认的大小。大多数情况下,默认值就足够大了。

  1. public class ReaderDemo3 {
  2. public static void main(String[] args) throws IOException {
  3. // 创建字符缓冲输入流对象
  4. BufferedReader br = new BufferedReader(new FileReader("demo1.txt"));
  5. // 方式1
  6. // int ch = 0;
  7. // while ((ch = br.read()) != -1) {
  8. // System.out.print((char) ch);
  9. // }
  10. // 方式2
  11. char[] chs = new char[1024];
  12. int len = 0;
  13. while ((len = br.read(chs)) != -1) {
  14. System.out.print(new String(chs, 0, len));
  15. }
  16. // 释放资源
  17. br.close();
  18. }
  19. }

BufferedWriter

  1. public class WriterDemo3 {
  2. public static void main(String[] args) throws IOException {
  3. BufferedWriter bw = new BufferedWriter(new FileWriter("bw.txt"));
  4. bw.write("hello");
  5. bw.write("world");
  6. bw.write("java");
  7. bw.flush();
  8. bw.close();
  9. }
  10. }

字符缓冲流中的特殊方法

  1. //BufferedWriter:
  2. public void newLine() //根据系统来决定换行符
  3. //BufferedReader:
  4. public String readLine() //一次读取一行数据
  5. //包含该行内容的字符串,不包含任何行终止符,如果已到达流末尾,则返回 null
  1. /**
  2. * 字符缓冲流的特殊方法:
  3. * BufferedWriter:
  4. * public void newLine():根据系统来决定换行符
  5. * BufferedReader:
  6. * public String readLine():一次读取一行数据
  7. * 包含该行内容的字符串,不包含任何行终止符,如果已到达流末尾,则返回 null
  8. */
  9. public class BufferDemo {
  10. public static void main(String[] args) throws IOException {
  11. //write();
  12. read();
  13. }
  14. public static void write() throws IOException {
  15. BufferedWriter bw=new BufferedWriter(new FileWriter("demo1.txt"));
  16. for (int x = 0; x < 3; x++) {
  17. bw.write("我爱林青霞");
  18. bw.newLine();
  19. bw.flush();
  20. }
  21. bw.write("重要的事情说三遍");
  22. bw.close();
  23. }
  24. public static void read() throws IOException {
  25. BufferedReader br=new BufferedReader(new FileReader("demo1.txt"));
  26. String line=null;
  27. while((line=br.readLine())!=null){
  28. System.out.println(line);
  29. }
  30. br.close();
  31. }
  32. }

字符转换流

  1. public class TransferDemo {
  2. public static void main(String[] args) throws IOException {
  3. // 封装数据源
  4. BufferedReader br = new BufferedReader(new FileReader("demo1.txt"));
  5. // 封装目的地
  6. BufferedWriter bw = new BufferedWriter(new FileWriter("demo4.txt"));
  7. // 两种方式其中的一种一次读写一个字符数组
  8. char[] chs = new char[1024];
  9. int len = 0;
  10. while ((len = br.read(chs)) != -1) {
  11. bw.write(chs, 0, len);
  12. bw.flush();
  13. }
  14. // 释放资源
  15. bw.close();
  16. br.close();
  17. }
  18. }
  1. public class TransferDemo2 {
  2. public static void main(String[] args) throws IOException {
  3. // 封装数据源
  4. BufferedReader br = new BufferedReader(new FileReader("demo1.txt"));
  5. // 封装目的地
  6. BufferedWriter bw = new BufferedWriter(new FileWriter("demo4.txt"));
  7. String line=null;
  8. while((line=br.readLine())!=null){
  9. bw.write(line);
  10. bw.newLine();
  11. bw.flush();
  12. }
  13. // 释放资源
  14. bw.close();
  15. br.close();
  16. }
  17. }

复制文本文件

  1. public class CopyFile {
  2. public static void main(String[] args) throws IOException{
  3. String srcFile="pride-and-prejudice.txt";
  4. String destFile="demo5.txt";
  5. long start=System.currentTimeMillis();
  6. //method(srcFile,destFile);//共耗时191毫秒
  7. //method2(srcFile,destFile);//共耗时59毫秒
  8. //method3(srcFile,destFile);//共耗时68毫秒
  9. //method4(srcFile,destFile);//共耗时38毫秒
  10. method5(srcFile,destFile);//共耗时177毫秒
  11. long end=System.currentTimeMillis();
  12. System.out.println("共耗时"+(end-start)+"毫秒");
  13. }
  14. //基本字符流一次读写一个字符
  15. public static void method(String srcFile,String destFile) throws IOException{
  16. FileReader fr=new FileReader(srcFile);
  17. FileWriter fw=new FileWriter(destFile);
  18. int ch=0;
  19. while((ch=fr.read())!=-1){
  20. fw.write(ch);
  21. }
  22. fr.close();
  23. fw.close();
  24. }
  25. //基本字符流一次读写一个字符数组
  26. public static void method2(String srcFile,String destFile) throws IOException{
  27. FileReader fr=new FileReader(srcFile);
  28. FileWriter fw=new FileWriter(destFile);
  29. int len=0;
  30. char[] chs=new char[1024];
  31. while((len=fr.read(chs))!=-1){
  32. fw.write(chs,0,len);
  33. }
  34. fr.close();
  35. fw.close();
  36. }
  37. //字符缓冲流一次读写一个字符
  38. public static void method3(String srcFile,String destFile) throws IOException{
  39. BufferedReader br=new BufferedReader(new FileReader(srcFile));
  40. BufferedWriter bw=new BufferedWriter(new FileWriter(destFile));
  41. int ch=0;
  42. while((ch=br.read())!=-1){
  43. bw.write(ch);
  44. }
  45. br.close();
  46. bw.close();
  47. }
  48. //字符缓冲流一次读写一个字符数组
  49. public static void method4(String srcFile,String destFile) throws IOException{
  50. BufferedReader br=new BufferedReader(new FileReader(srcFile));
  51. BufferedWriter bw=new BufferedWriter(new FileWriter(destFile));
  52. int len=0;
  53. char[] chs=new char[1024];
  54. while((len=br.read(chs))!=-1){
  55. bw.write(chs,0,len);
  56. }
  57. br.close();
  58. bw.close();
  59. }
  60. public static void method5(String srcFile,String destFile) throws IOException{
  61. BufferedReader br=new BufferedReader(new FileReader(srcFile));
  62. BufferedWriter bw=new BufferedWriter(new FileWriter(destFile));
  63. String line=null;
  64. while((line=br.readLine())!=null){
  65. bw.write(line);
  66. bw.newLine();
  67. bw.flush();
  68. }
  69. br.close();
  70. bw.close();
  71. }
  72. }

复制多级文件夹

  1. public class CopyFolders {
  2. public static void main(String[] args) throws IOException {
  3. File srcFile=new File("demo2");
  4. File destFile=new File("demo3");
  5. copyFolder(srcFile,destFile);
  6. }
  7. //复制多级文件夹
  8. public static void copyFolder(File srcFile,File destFile) throws IOException {
  9. if(srcFile.isDirectory()){
  10. //在destFile下创建新文件夹
  11. File newDirectory=new File(destFile,srcFile.getName());
  12. newDirectory.mkdir();
  13. // 获取该File对象下的所有文件或者文件夹
  14. File[] files=srcFile.listFiles();
  15. if(files!=null){
  16. //将srcFile下的所有文件都复制到新destFile
  17. for(File file:files){
  18. copyFolder(file,newDirectory);
  19. }
  20. }
  21. }else{
  22. File newFile=new File(destFile,srcFile.getName());
  23. copyFile(srcFile,newFile);
  24. }
  25. }
  26. //复制文件
  27. public static void copyFile(File srcFile,File destFile) throws IOException {
  28. BufferedInputStream bis=new BufferedInputStream(new FileInputStream(srcFile));
  29. BufferedOutputStream bos=new BufferedOutputStream(new FileOutputStream(destFile));
  30. int len=0;
  31. byte[] bys=new byte[1024];
  32. while((len=bis.read(bys))!=-1){
  33. bos.write(bys,0,len);
  34. }
  35. bos.close();
  36. bis.close();
  37. }
  38. }