21-java学习-字符流出现的原因、String中的编解码问题、转换流&案例代码
目录:
字符流出现的原因
String中的编解码问题
转换流
1.字符流出现的原因及编码表概述和常见编码表
A:字符流出现的原因:由于字节流操作中文不是特别方便,所以,java就提供了字符流。
B:码表
C:字符流: 字符流 = 字节流 + 编码表
2.String类中的编码和解码问题
编码: 就是把字符串转换成字节数组
- 把一个字符串转换成一个字节数组
- public byte[] getBytes();使用平台的默认字符集将此 String编码为 byte 序列,并将结果存储到一个新的 byte 数组中。
- public byte[] getBytes(String charsetName) 使用指定的字符集将此 String 编码为 byte 序列,并将结果存储到一个新的 byte 数组中。
- 解码: 把字节数组转换成字符串
- public String(byte[] bytes): 通过使用平台的默认字符集解码指定的 byte 数组,构造一个新的 String。
- public String(byte[] bytes, String charsetName) 通过使用指定的 charset 解码指定的 byte 数组,构造一个新的 String。
- 使用什么字符集进行编码,那么就是使用什么字符集进行解码
- 老地方 ----- 十进制 ---- 二进制 ---- 发出去
- 接收 ---- 二进制 ---- 十进制 --- 老地方
String(byte[] bytes, String charsetName):通过指定的字符集解码字节数组
byte[] getBytes(String charsetName):使用指定的字符集合把字符串编码为字节数组
编码:把看得懂的变成看不懂的: String -- byte[]
解码:把看不懂的变成看得懂的: byte[] -- String
###21.03_IO流(转换流OutputStreamWriter的使用)(掌握)
A:OutputStreamWriter的构造方法
OutputStreamWriter(OutputStream out):根据默认编码(GBK)把字节流的数据转换为字符流
OutputStreamWriter(OutputStream out,String charsetName):根据指定编码把字节流数据转换为字符流
B:案例演示: OutputStreamWriter写出数据
3.字符流的5种写数据的方式
A: 方法概述
public void write(int c) 写一个字符
public void write(char[] cbuf) 写一个字符数组
public void write(char[] cbuf,int off,int len) 写一个字符数组的 一部分
public void write(String str) 写一个字符串
public void write(String str,int off,int len) 写一个字符串的一部分
4.转换流InputStreamReader的使用
A:InputStreamReader的构造方法
InputStreamReader(InputStream is):用默认的编码(GBK)读取数据
InputStreamReader(InputStream is,String charsetName):用指定的编码读取数据
5.字符流的2种读数据的方式
A: 方法概述
public int read() 一次读取一个字符
public int read(char[] cbuf) 一次读取一个字符数组 如果没有读到 返回-1
A:案例演示: 字符流的2种读数据的方式
1. int ch = 0 ; // 记录读取到的字符
while((ch = isr.read()) != -1){
System.out.print((char)ch);
}
// 释放资源
isr.close() ;
2. char[] chs = new char[1024] ;
int len = 0 ; // 作用: 记录读取到有效的字符个数
while((len = isr.read(chs)) != -1){
//System.out.print(String.valueOf(chs, 0, len));
System.out.print(new String(chs , 0 , len));
}
// 释放资源
isr.close() ;
6.字符流复制文本文件
A:案例演示: 字符流复制文本文件
// 频繁的读写操作
// 一次读取一个字符数组复制文件
char[] chs = new char[1024];
int len = 0 ;
while((len = isr.read(chs)) != -1){
osw.write(chs, 0, len) ;
}
7.FileWriter和FileReader复制文本文件
A:FileReader和FileWriter的出现
转换流的名字比较长,而我们常见的操作都是按照本地默认编码实现的,
所以,为了简化我们的书写,转换流提供了对应的子类。
FileWriter
FileReader
字符流便捷类: 因为转换流的名字太长了,并且在一般情况下我们不需要制定字符集,
于是java就给我们提供转换流对应的便捷类
-
- 转换流 便捷类
- OutputStreamWriter ------- FileWriter
- InputStreamReader ------- FileReader
例子:
BufferedReader br = new BufferedReader(new FileReader("OutputStreamWriterDemo.java")) ;
BufferedWriter bw = new BufferedWriter(new FileWriter("c.txt")) ;
// 频繁的读写操作
// 一次读取一个字符复制文件
// 一次读取一个字符数组复制文件
char[] chs = new char[1024] ;
int len = 0 ;
while((len = br.read(chs)) != -1){
bw.write(chs, 0, len) ;
}
// 释放资源
bw.close() ;
br.close() ;
8.字符缓冲流的基本使用
高效的字符流
高效的字符输出流: BufferedWriter
构造方法: public BufferedWriter(Writer w)
高效的字符输入流: BufferedReader
构造方法: public BufferedReader(Reader e)
9.字符缓冲流复制文本文件
A:案例演示: 字符缓冲流复制文本文件
// 创建对象
BufferedReader br = new BufferedReader(new FileReader("OutputStreamWriterDemo.java")) ;
BufferedWriter bw = new BufferedWriter(new FileWriter("c.txt")) ;
// 频繁的读写操作
// 一次读取一个字符复制文件
// 一次读取一个字符数组复制文件
char[] chs = new char[1024] ;
int len = 0 ;
while((len = br.read(chs)) != -1){
bw.write(chs, 0, len) ;
}
// 释放资源
bw.close() ;
br.close() ;
10.字符缓冲流的特殊功能
A:字符缓冲流的特殊功能
BufferedWriter: public void newLine():根据系统来决定换行符 具有系统兼容性的换行符
BufferedReader: public String readLine():一次读取一行数据 是以换行符为标记的 读到换行符就换行 没读到数据返回null
包含该行内容的字符串,不包含任何行终止符,如果已到达流末尾,则返回 null
B:案例演示: 字符缓冲流的特殊功能
11.字符缓冲流的特殊功能复制文本文件
A:案例演示: 字符缓冲流的特殊功能复制文本文件
// 创建对象
BufferedReader br = new BufferedReader(new FileReader("OutputStreamWriterDemo.java")) ;
BufferedWriter bw = new BufferedWriter(new FileWriter("copyFile2.txt")) ;
// 频繁的读写操作
String line = null ; // 作用: 记录读取到的行数据
while((line = br.readLine()) != null){
bw.write(line) ;
bw.newLine() ;
bw.flush() ;
}
// 释放资源
bw.close() ;
br.close() ;
12.把集合中的数据存储到文本文件
A:案例演示: 需求:把ArrayList集合中的字符串数据存储到文本文件
分析:
- a: 创建一个ArrayList集合
- b: 添加元素
- c: 创建一个高效的字符输出流对象
- d: 遍历集合,获取每一个元素,把这个元素通过高效的输出流写到文本文件中
- e: 释放资源
public class MyTest4 {
public static void main(String[] args) throws IOException {
/*A:
案例演示:
需求:把ArrayList集合中的字符串数据存储到文本文件*/
ArrayList<String> list = new ArrayList<>();
list.add("赵四");
list.add("刘能");
list.add("王大拿");
list.add("谢大脚");
//遍历集合中的数据,通过输出流,写入到文本文件中保存
FileWriter fileWriter = new FileWriter("names.txt");
for (String name : list) {
// System.out.println(name);
fileWriter.write(name);
fileWriter.write("\r\n");
fileWriter.flush();
}
fileWriter.close();
}
}
13.把文本文件中的数据存储到集合中
A:案例演示: 需求:从文本文件中读取数据(每一行为一个字符串数据)到集合中,并遍历集合
分析:
* a: 创建高效的字符输入流对象
* b: 创建一个集合对象
* c: 读取数据(一次读取一行)
* d: 把读取到的数据添加到集合中
* e: 遍历集合
* f: 释放资源
public class MyTest5 {
public static void main(String[] args) throws IOException {
//把文本文件中的数据读取到集合中取
ArrayList<String> list = new ArrayList<>();
//读取文本文件,把文件中的数据读取到集合
//采用高效的字符输入流,一次读取一行
BufferedReader bfr = new BufferedReader(new InputStreamReader(new FileInputStream("names.txt")));
String name=null;
while ((name=bfr.readLine())!=null){
list.add(name);
}
System.out.println(list);
bfr.close();
}
}
14.随机获取文本文件中的姓名
A:案例演示: 需求:我有一个文本文件,每一行是一个学生的名字,请写一个程序,每次允许随机获取一个学生名称
* 分析:
* a: 创建一个高效的字符输入流对象
* b: 创建集合对象
* c: 读取数据,把数据存储到集合中
* d: 产生一个随机数,这个随机数的范围是 0 - 集合的长度 . 作为: 集合的随机索引
* e: 根据索引获取指定的元素
* f: 输出
* g: 释放资源
public class MyTest6 {
public static void main(String[] args) throws IOException {
/* 案例演示:
需求:我有一个文本文件,每一行是一个学生的名字,请写一个程序,每次允许随机获取一个学生名称*/
ArrayList<String> list = new ArrayList<>();
//读取文本文件,把文件中的数据读取到集合
//采用高效的字符输入流,一次读取一行
BufferedReader bfr = new BufferedReader(new InputStreamReader(new FileInputStream("names.txt")));
String name = null;
while ((name = bfr.readLine()) != null) {
list.add(name);
}
//System.out.println(list);
bfr.close();
//随机从集合中抽取一个人
//生成一个随机索引
Random random = new Random();
int index = random.nextInt(list.size());
String s = list.get(index);
System.out.println(s+"====来回答问题");
//把数据和程序解耦
}
}
15.复制单级文件夹
A:案例演示: 需求: 复制D:\\course这文件夹到E:\\course
- 分析:
- a: 封装D:\\course为一个File对象
- b: 封装E:\\course为一个File对象,然后判断是否存在,如果不存在就是创建一个目录
- c: 获取a中的File对应的路径下所有的文件对应的File数组
- d: 遍历数组,获取每一个元素,进行复制
- e: 释放资源
// 获取a中的File对应的路径下所有的文件对应的File数组
File[] files = srcFolder.listFiles() ;
// 遍历
for(File f : files){
// 获取当前文件对应的名称
String currentFileName = f.getName() ;
// 构建目标文件
File destFile = new File(destFolder , currentFileName) ;
// 复制
copyFile(f , destFile) ;
}
}
/**
* 复制文件
* @throws IOException
*/
public static void copyFile(File f, File destFile) throws IOException {
// 创建高效的字节流对象
BufferedInputStream bis = new BufferedInputStream(new FileInputStream(f)) ;
BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(destFile)) ;
// 频繁的读写操作
// 一次读取一个字节数组复制文件
byte[] bytes = new byte[1024] ;
int len = 0 ;
while((len = bis.read(bytes)) != -1){
bos.write(bytes, 0, len) ;
}
// 释放资源
bos.close() ;
bis.close() ;
}
16.复制指定目录下指定后缀名的文件并修改名称
A:案例演示: * 需求: 复制D:\\demo目录下所有以.java结尾的文件到E:\\demo .并且将其后缀名更改文.jad
- 分析:
- a: 复制D:\\demo文件夹到E:\\
- b: 对E:\\demo文件夹下的文件进行重命名
// 封装D:\\demo为一个File对象
File srcFolder = new File("D:\\demo") ;
// 封装E:\\demo为一个File对象
File destFolder = new File("E:\\demo") ;
// 判断是否存在
if(!destFolder.exists()){
// 创建
destFolder.mkdir() ;
}
// 复制文件夹
copyFolder(srcFolder , destFolder) ;
// ---------------------上面的代码是复制文件夹,下面的代码是更改名称-----------------------------------------
// 获取E:\\demo下的所有的文件对应File数组
File[] listFiles = destFolder.listFiles() ;
// 遍历
for(File file : listFiles){
// 构建目标文件
String currentFileName = file.getName() ;
currentFileName = currentFileName.replace(".java", ".jad") ;
File destFile = new File(destFolder , currentFileName) ;
// 重命名
file.renameTo(destFile) ;
}
}
/**
* 复制文件夹
* @throws IOException
*/
private static void copyFolder(File srcFolder, File destFolder) throws IOException {
// 获取
File[] files = srcFolder.listFiles(new FilenameFilter() {
@Override
public boolean accept(File dir, String name) {
return new File(dir, name).isFile() && name.endsWith(".java");
}
}) ;
// 遍历
for(File file : files){
// 获取名称
String currentFileName = file.getName() ;
// 构建目标文件
File destFile = new File(destFolder , currentFileName) ;
// 复制
StreamTest4.copyFile(file, destFile) ;
}
}
17.键盘录入学生信息按照总分排序并写入文本文件
A:案例演示: 需求:键盘录入3个学生信息(姓名,语文成绩(chineseScore),数学成绩(mathScore),英语成绩(englishScore)),
- 按照总分从高到低存入文本文件
-
- 分析:
- a: 创建一个学生类: 姓名,语文成绩(chineseScore),数学成绩(mathScore),英语成绩(englishScore)
- b: 因为要排序,所以需要选择TreeSet进行存储学生对象
- c: 键盘录入学生信息,把学生信息封装成一个学生对象,在把学生对象添加到集合中
- d: 创建一个高效的字符输出流对象
- e: 遍历集合,把学生的信息写入到指定的文本文件中
- f: 释放资源
//因为要排序,所以需要选择TreeSet进行存储学生对象
TreeSet<Student> students = new TreeSet<Student>(new Comparator<Student>() {
@Override
public int compare(Student s1, Student s2) {
// 比较总分
int num = s2.getTotal() - s1.getTotal() ;
// 比较姓名
int num2 = (num == 0 ) ? s2.getName().compareTo(s1.getName()) : num ;
// 返回
return num2;
}
}) ;
// 键盘录入学生信息,把学生信息封装成一个学生对象,在把学生对象添加到集合中
for(int x = 0 ; x < 3 ; x++){
int i = x + 1 ;
System.out.println("第" + i + "个学生信息录入开始................................");
// 创建一个Scanner对象
Scanner sc = new Scanner(System.in) ;
System.out.println("请您输入第" + i + "个学生的姓名: ");
String name = sc.nextLine() ;
System.out.println("请您输入第" + i + "个学生的语文成绩: ");
String chineseScoreStr = sc.nextLine() ;
System.out.println("请您输入第" + i + "个学生的数学成绩: ");
String mathScoreStr = sc.nextLine() ;
System.out.println("请您输入第" + i + "个学生的英语成绩: ");
String englishScoreStr = sc.nextLine() ;
// 创建一个学生对象
Student s = new Student() ;
s.setName(name) ;
s.setChineseScore(Integer.parseInt(chineseScoreStr)) ;
s.setMathScore(Integer.parseInt(mathScoreStr)) ;
s.setEnglishScore(Integer.parseInt(englishScoreStr)) ;
// 把学生对象添加到集合中
students.add(s) ;
}
// 创建一个高效的字符输出流对象
BufferedWriter bw = new BufferedWriter(new FileWriter("studentInfo.config")) ;
bw.write("姓名\t\t总分\t\t语文成绩\t\t数学成绩\t\t英语成绩") ;
bw.newLine() ;
bw.flush() ;
// 遍历
for(Student s : students){
bw.write(s.getName() + "\t\t" + s.getTotal() + "\t\t" + s.getChineseScore() + "\t\t" + s.getMathScore() + "\t\t" + s.getEnglishScore()) ;
bw.newLine() ;
bw.flush();
}
// 释放资源
bw.close() ;
//提示
System.out.println("学生信息存储完毕");