java对文本文档进行操作(拷贝、显示)出现乱码一般来说,可以从两个方面入手。

1、文本文件本身的编码格式。

2、java代码中处理文本文件的编码格式。

这里要注意的一点是,我们可以看出copyFileByByte方法和copyFileByChar1方法都是没有办法设置目的文件的编码格式的,并且处理不好都可能出现乱码,但是需要明确一点的是,copyFileByByte方法拷贝的文件即便出现乱码也可以通过另存为其他格式来调整消除乱码,同样的操作在copyFileByChar1方法拷贝生成的源文件是不能消除乱码的。

假设我们以字节流格式来读取一份utf-8编码格式的txt文档:

package com.audi;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream; public class ReadFile
{
File fileName = new File("C:/Users/Mike/Desktop/为什么使用接口.txt");
public void readFileByByte()
{
InputStream inputStream =null;
try
{
inputStream = new FileInputStream(fileName);
byte[] temp = new byte[2048];
StringBuilder buf = new StringBuilder(); //非线程安全,不过这里是单线程,无所谓线程安全与否
int length = 0;
while (-1!=(length=inputStream.read(temp)))
{
// 注意下面的代码使用utf-8作为编码格式
buf.append(new String(temp,0,length,"utf-8"));
}
System.out.println(buf.toString()); } catch (FileNotFoundException e)
{
e.printStackTrace();
System.out.println("文件C:/Users/Mike/Desktop/为什么使用接口.txt不存在");
}
catch (IOException e) {
e.printStackTrace();
}
finally {
try
{
if (inputStream!=null)
{
inputStream.close();
}
} catch (Exception e2)
{
e2.printStackTrace();
} } } public void readFileByChar()
{ }
}

文本文件本来的内容为:

测试代码如下:

package com.audi;

public class Test
{
public static void main(String[] args)
{
ReadFile readFile = new ReadFile();
readFile.readFileByByte();
}
}

运行程序读取文本文件得到的控制台输出:

可以看出中文没有乱码,一切正常,如果我们在在上面的代码中不设置解码格式,运行程序,依然正常,这是为什么?

这是因为我的java文件的默认编码格式就是utf-8,所以java代码在编译的时候默认就取了这个格式作为解码格式。

如果换成GBK,那么一样会出现乱码:

那么如果我们要使用字节流拷贝一份文本文件呢?

java代码如下:

public void copyFileByByte()
{
InputStream inputStream = null;
OutputStream outputStream = null;
File destName = new File("copyFileByByte.txt");
try
{
inputStream = new FileInputStream(fileName);
outputStream = new FileOutputStream(destName);
int length =0;
byte[] temp = new byte[2048];
while (-1!=(length=inputStream.read(temp)))
{
outputStream.write(temp, 0, length);
}
outputStream.flush();
} catch (Exception e)
{
e.printStackTrace();
}
finally {
try
{
if (inputStream!=null)
{
inputStream.close();
}
if (outputStream!=null)
{
outputStream.close();
}
} catch (Exception e2)
{
e2.printStackTrace();
}
}
}

测试代码如下:

package com.audi;

public class Test
{
public static void main(String[] args)
{
ReadFile readFile = new ReadFile();
// readFile.readFileByByte();
readFile.copyFileByByte();
}
}

我的实际测试结果是,拷贝后的文件的编码格式是由源文件格式决定的,它会和源文件的格式保持一致。一般不会出现乱码。

下面是使用字符流来操作文本文件。

public void readFileByChar()
{
Reader reader = null;
try
{
reader = new BufferedReader(new FileReader(fileName));
int length =0;
char[] temp = new char[2048];
StringBuilder buf = new StringBuilder();
while (-1!=(length=reader.read(temp)))
{
buf.append(new String(temp, 0, length));
}
System.out.println(buf.toString());
} catch (FileNotFoundException e)
{
e.printStackTrace();
System.out.println("文件C:/Users/Mike/Desktop/为什么使用接口.txt不存在");
}
catch (IOException e) {
e.printStackTrace();
}
}

上面的代码和字节流读取文件不同的是:

1、缓冲数组改成了char[]类型

2、new string构造函数中不能再设定格式,这将会导致它直接使用测试的java文件格式来解码读取的字符流。如下图所示,因为我的txt文档也是utf-8格式的,所以不会出现乱码错误。

测试代码:

package com.audi;

public class Test
{
public static void main(String[] args)
{
ReadFile readFile = new ReadFile();
// readFile.readFileByByte();
// readFile.copyFileByByte();
readFile.readFileByChar();
}
}

实际运行效果;

如果把测试java文件的编码格式改为gbk,那么就会出现乱码

下面以字符流方式拷贝文件,同样无法手动设置文件的编码格式:

public void copyFileByChar1()
{
FileReader fReader = null;
FileWriter fWriter = null;
File destName = new File("copyFileByChar1.txt");
try
{
fReader = new FileReader(fileName);
fWriter = new FileWriter(destName);
int length =0;
char[] temp = new char[2048];
while (-1!=(length=fReader.read(temp)))
{
fWriter.write(temp,0,length);
}
} catch (FileNotFoundException e)
{
e.printStackTrace();
System.out.println("文件C:/Users/Mike/Desktop/为什么使用接口.txt不存在");
}
catch (IOException e) {
e.printStackTrace();
}
finally
{
try
{
if (fReader!=null)
{
fReader.close();
}
if (fWriter!=null)
{
fWriter.close();
}
System.out.println("copy succeed");
} catch (Exception e2)
{
e2.printStackTrace();
}
}
}

测试代码,拷贝后文件的格式依然会和测试代码的java文件的编码格式保持一致。

package com.audi;

public class Test
{
public static void main(String[] args)
{
ReadFile readFile = new ReadFile();
// readFile.readFileByByte();
// readFile.copyFileByByte();
// readFile.readFileByChar();
readFile.copyFileByChar1();
}
}

这里要注意的一点是,我们可以看出copyFileByByte方法和copyFileByChar1方法都是没有办法设置目的文件的编码格式的,并且处理不好都可能出现乱码,但是需要明确一点的是,copyFileByByte方法拷贝的文件即便出现乱码也可以通过另存为其他格式来调整消除乱码,同样的操作在copyFileByChar1方法拷贝生成的源文件是不能消除乱码的。

下面介绍另外一种方法来拷贝文件,使用的是InputStreamReader和OutputStreamWriter:

public void copyFileByChar2()
{
InputStreamReader inputStreamReader =null;
OutputStreamWriter outputStreamWriter = null;
File destName = new File("copyFileByChar2.txt"); try
{
/*其实只有InputStreamReader和OutputStreamWriter才可以设置编码格式
*
* */
inputStreamReader = new InputStreamReader(new java.io.FileInputStream(fileName),"utf-8");
outputStreamWriter = new OutputStreamWriter(new java.io.FileOutputStream(destName),"utf-8");
int length =0;
char[] temp = new char[2048];
while (-1!=(length=inputStreamReader.read(temp)))
{
outputStreamWriter.write(temp,0,length);
}
} catch (UnsupportedEncodingException e1)
{
e1.printStackTrace();
} catch (FileNotFoundException e1)
{
e1.printStackTrace();
} catch (IOException e)
{
e.printStackTrace();
}
finally
{
try
{
outputStreamWriter.flush();
if (inputStreamReader!=null)
{
inputStreamReader.close();
}
if (outputStreamWriter!=null)
{
outputStreamWriter.close();
}
} catch (Exception e2)
{
e2.printStackTrace();
}
System.out.println("拷贝结束了");
}
}

这个时候拷贝文件的格式完全可控,再也不会依赖测试文件的格式了。

此时,设置源文件UTF-8格式,测试java文件GBK格式:

运行测试代码:

package com.audi;

public class Test
{
public static void main(String[] args)
{
ReadFile readFile = new ReadFile();
// readFile.readFileByByte();
readFile.copyFileByByte();
// readFile.readFileByChar();
readFile.copyFileByChar1();
readFile.copyFileByChar2();
}
}

拷贝后的文件,没有出现乱码:

最后,其实我们上面一直在强调都是针对文本文件的操作,那如果是非文本文件呢?

非文本文件☞什么?

严格来说,文件只有两种类型:文本文件(也叫ASCII文件)和二进制文件。

ASCII码:  00110101 00110110 00110111 00111000
            ↓      ↓    ↓   ↓
十进制码:           5     6    7    8 共占用4个字节。

二进制文件是按二进制的编码方式来存放文件的。 例如, 数5678(五千六百七十八,十进制)的存储形式为:

00010110 00101110(二进制)只占二个字节。

非文本文件我们只能通过字节流来操作,因为字节流虽然也不能设置编解码格式,但是它是一个一个字节读取的,源文件是什么样,拷贝出的文件也是什么样,特定的解码器再去解码就可以了。

字符流就不一样了,它会根据设定的编解码格式的不同,一次读取两个或者三个字节来进行编解码,这样很可能会损坏源文件的编码方式,导致拷贝的文件出错。

java 字符流 字节流的更多相关文章

  1. Java 字符流实现文件读写操作(FileReader-FileWriter)

    Java 字符流实现文件读写操作(FileReader-FileWriter) 备注:字符流效率高,但是没有字节流底层 字节流地址:http://pengyan5945.iteye.com/blog/ ...

  2. java字符流操作flush()方法及其注意事项

    java字符流操作flush()方法及其注意事项   flush()方法介绍 查阅文档可以发现,IO流中每一个类都实现了Closeable接口,它们进行资源操作之后都需要执行close()方法将流关闭 ...

  3. Java字符流和字节流对文件操作

    记得当初自己刚开始学习Java的时候,对Java的IO流这一块特别不明白,所以写了这篇随笔希望能对刚开始学习Java的人有所帮助,也方便以后自己查询.Java的IO流分为字符流(Reader,Writ ...

  4. java IO之 字符流 (字符流 = 字节流 + 编码表) 装饰器模式

    字符流 计算机并不区分二进制文件与文本文件.所有的文件都是以二进制形式来存储的,因此, 从本质上说,所有的文件都是二进制文件.所以字符流是建立在字节流之上的,它能够提供字符 层次的编码和解码.列如,在 ...

  5. java字符流与字节流的区别是什么

    java中字符流与字节流的区别: 1.字节流操作的基本单元为字节:字符流操作的基本单元为Unicode码元. 2.字节流默认不使用缓冲区:字符流使用缓冲区. 3.字节流通常用于处理二进制数据,实际上它 ...

  6. java 字符流与字节流互转

    package test; import java.io.BufferedReader; import java.io.ByteArrayInputStream; import java.io.IOE ...

  7. Java 字符流操作

    上篇文章Java 字节流操作介绍了java中基本的字节流操作,但是我们常常对于字符操作,如果使用字节流来实现输入输出就显得麻烦,我们可以使用字符流来实现对我们看得见的字符char进行操作,主要内容如下 ...

  8. Java 字符流文件读写

    上篇文章,我们介绍了 Java 的文件字节流框架中的相关内容,而我们本篇文章将着重于文件字符流的相关内容. 首先需要明确一点的是,字节流处理文件的时候是基于字节的,而字符流处理文件则是基于一个个字符为 ...

  9. Java IO流-字节流

    2017-11-05 17:48:17 Java中的IO流按数据类型分类分为两种,一是字节流,二是字符流.字符流的出现是为了简化文本数据的读入和写出操作. 如果操作的文件是文本文件,那么使用字符流会大 ...

随机推荐

  1. python的编解码问题

    http://blog.chinaunix.net/uid-27838438-id-4227131.html

  2. htc使用方式(转)

    一.htc的两种使用方式: 关联行为(Attach Behavior): IE 5.0以上支持, htc 技术出现的初衷. 主要目的是把对象复杂的样式变化包装起来,替代 javascript + cs ...

  3. bzoj 3505 [Cqoi2014]数三角形——排列组合

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3505 好题!一定要经常回顾! 那个 一条斜线上的点的个数是其两端点横坐标之差和纵坐标之差的g ...

  4. Ubuntu下部署GitLab-——基于14.04系统

    搭建GitLab的目的: 方便公司开发管理代码 GitLab实现的功能: 1.关闭了gitlab的注册功能 2.修改了默认端口 3.汉化 0.前期准备 # 环境 Ubuntu 14.04 root@i ...

  5. webrtc 术语

    参考网址 https://webrtcglossary.com/ ORTC stands for Object-RTC.ORTC is an initiative involving Google, ...

  6. compile cef2526

    fetch --nohooks chromium cd /path/to/chromium/src# git checkout -b 51.0.2704.103 refs/tags/51.0.2704 ...

  7. GXT4.0 BorderLayoutContainer布局

    T字型布局. @Override public void onModuleLoad() { BorderLayoutContainer con = new BorderLayoutContainer( ...

  8. POJ3249(DAG上的dfs)

    Test for Job Time Limit: 5000MS   Memory Limit: 65536K Total Submissions: 10567   Accepted: 2482 Des ...

  9. POJ2777(线段树涂色问题)

    Count Color Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 42828   Accepted: 12973 Des ...

  10. python3 中 requests 框架

    原文的文件地址:http://blog.csdn.net/shanzhizi/article/details/50903748 一.安装 Requests 通过pip安装 pip install re ...