IO流(二)————字符流

来源:互联网 发布:centos nginx默认目录 编辑:程序博客网 时间:2024/06/07 18:18
1. 字符流FileReader
* 1.字符流是什么
    * 字符流是可以直接读写字符的IO流
    * 字符流读取字符, 就要先读取到字节数据, 然后转为字符. 如果要写出字符, 需要把字符转为字节再写出.    
* 2.FileReader
    * FileReader类的read()方法可以按照字符大小读取

FileReader fr = new FileReader("aaa.txt");//创建输入流对象,关联aaa.txtint ch;while((ch = fr.read()) != -1) {//将读到的字符赋值给ch//通过项目默认的码表一次读取一个字符System.out.println((char)ch);//将读到的字符强转后打印}fr.close();//关流 
2. 字符流FileWriter
* FileWriter类的write()方法可以自动把字符转为字节写出
FileWriter fw = new FileWriter("aaa.txt");fw.write("aaa");fw.close();
3. 字符流的拷贝
FileReader fr = new FileReader("a.txt");FileWriter fw = new FileWriter("b.txt");int ch;while((ch = fr.read()) != -1) {fw.write(ch);}fr.close();fw.close();//Writer类中有一个2k的小缓冲区,如果不关流,就会将内容写到缓冲区里,关流会将缓冲区内容刷新,再关闭
4. 什么情况下使用字符流
* 字符流也可以拷贝文本文件, 但不推荐使用. 因为读取时会把字节转为字符, 写出时还要把字符转回字节.
* 程序需要读取一段文本, 或者需要写出一段文本的时候可以使用字符流
* 读取的时候是按照字符的大小读取的,不会出现半个中文
* 写出的时候可以直接将字符串写出,不用转换为字节数组

5. 字符流是否可以拷贝非纯文本的文件
* 不可以拷贝非纯文本的文件
* 因为在读的时候会将字节转换为字符,在转换过程中,可能找不到对应的字符,就会用?代替,写出的时候会将字符转换成字节写出去
* 如果是?,直接写出,这样写出之后的文件就乱了,看不了了  
//字符流不能拷贝纯文本的文件
FileReader fr = new FileReader("双元.jpg");FileWriter fw = new FileWriter("copy.jpg");int c;while((c = fr.read()) != -1) {fw.write(c);}fr.close();fw.close();
6. 自定义字符数组的拷贝
FileReader fr = new FileReader("aaa.txt");//创建字符输入流,关联aaa.txtFileWriter fw = new FileWriter("bbb.txt");//创建字符输出流,关联bbb.txtint len;char[] arr = new char[1024*8];//创建字符数组while((len = fr.read(arr)) != -1) {//将数据读到字符数组中fw.write(arr, 0, len);//从字符数组将数据写到文件上}fr.close();//关流释放资源fw.close();
7. 带缓冲的字符流
* BufferedReader的read()方法读取字符时会一次读取若干字符到缓冲区, 然后逐个返回给程序, 降低读取文件的次数, 提高效率
* BufferedWriter的write()方法写出字符时会先写到缓冲区, 缓冲区写满时才会写到文件, 降低写文件的次数, 提高效率
BufferedReader br = new BufferedReader(new FileReader("aaa.txt"));//创建字符输入流对象,关联aaa.txtBufferedWriter bw = new BufferedWriter(new FileWriter("bbb.txt"));//创建字符输出流对象,关联bbb.txtint ch;while((ch = br.read()) != -1) {//read一次,会先将缓冲区读满,从缓冲去中一个一个的返给临时变量chbw.write(ch);//write一次,是将数据装到字符数组,装满后再一起写出去}br.close();//关流bw.close();  
8. readLine()和newLine()方法
* BufferedReader的readLine()方法可以读取一行字符(不包含换行符号)
* BufferedWriter的newLine()可以输出一个跨平台的换行符号"\r\n"

/** * @param args * 带缓冲区的流中的特殊方法 * readLine() * newLine(); *  * newLine()与\r\n的区别 * newLine()是跨平台的方法 * \r\n只支持的是windows系统 * @throws IOException  */BufferedReader br = new BufferedReader(new FileReader("aaa.txt"));BufferedWriter bw = new BufferedWriter(new FileWriter("bbb.txt"));String line;while((line = br.readLine()) != null) {bw.write(line);//bw.write("\r\n");//只支持windows系统bw.newLine();//跨平台的}br.close();bw.close(); 
9. 将文本反转
* 将一个文本文档上的文本反转,第一行和倒数第一行交换,第二行和倒数第二行交换
public class Test1 {/** * @param args * 将一个文本文档上的文本反转,第一行和倒数第一行交换,第二行和倒数第二行交换 *  * 分析: * 1,创建输入输出流对象 * 2,创建集合对象 * 3,将读到的数据存储在集合中 * 4,倒着遍历集合将数据写到文件上 * 5,关流 * @throws IOException  *  * 注意事项: * 流对象尽量晚开早关 */public static void main(String[] args) throws IOException {//改写后是尽量晚开早关// 1,创建输入输出流对象BufferedReader br = new BufferedReader(new FileReader("zzz.txt"));//2,创建集合对象ArrayList<String> list = new ArrayList<>();//3,将读到的数据存储在集合中String line;while((line = br.readLine()) != null) {list.add(line);}br.close();//关流//4,倒着遍历集合将数据写到文件上BufferedWriter bw = new BufferedWriter(new FileWriter("revzzz.txt"));for(int i = list.size() - 1; i >= 0; i--) {bw.write(list.get(i));bw.newLine();}//5,关流bw.close();}}
10. LineNumberReader
* LineNumberReader是BufferedReader的子类, 具有相同的功能, 并且可以统计行号
* 调用getLineNumber()方法可以获取当前行号
* 调用setLineNumber()方法可以设置当前行号

LineNumberReader lnr = new LineNumberReader(new FileReader("aaa.txt"));String line;lnr.setLineNumber(100);//设置行号while((line = lnr.readLine()) != null) {System.out.println(lnr.getLineNumber() + ":" + line);//获取行号}lnr.close(); 
11. 装饰设计模式 
public class Demo6_Wrap {/** * @param args * 装饰设计模式的好处是: * 耦合性不强,被装饰的类的变化与装饰类的变化无关 */public static void main(String[] args) {HeiMaStudent hms = new HeiMaStudent(new Student());hms.code();}}interface Coder {public void code();}class Student implements Coder {@Overridepublic void code() {System.out.println("javase");System.out.println("javaweb");}}class HeiMaStudent implements Coder {//1,获取被装饰类的引用private Student s;//获取学生引用//2,在构造方法中传入被装饰类的对象public HeiMaStudent(Student s) {this.s = s;}//3,对原有的功能进行升级@Overridepublic void code() {s.code();System.out.println("ssh");System.out.println("数据库");System.out.println("大数据");System.out.println("...");}}
12. 使用指定的码表读写字符
* FileReader是使用默认码表读取文件, 如果需要使用指定码表读取, 那么可以使用InputStreamReader(字节流,编码表)
* FileWriter是使用默认码表写出文件, 如果需要使用指定码表写出, 那么可以使用OutputStreamWriter(字节流,编码表)
public class Demo7_TransIO {/** * @param args * @throws IOException  */public static void main(String[] args) throws IOException {//demo1();//demo2();BufferedReader br = //更高效的读new BufferedReader(new InputStreamReader(new FileInputStream("utf-8.txt"), "utf-8"));BufferedWriter bw = //更高效的写new BufferedWriter(new OutputStreamWriter(new FileOutputStream("gbk.txt"), "gbk"));int c;while((c = br.read()) != -1) {bw.write(c);}br.close();bw.close();}public static void demo2() throws UnsupportedEncodingException,FileNotFoundException, IOException {InputStreamReader isr = new InputStreamReader(new FileInputStream("utf-8.txt"), "uTf-8");//指定码表读字符OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("gbk.txt"), "gbk");//指定码表写字符int c;while((c = isr.read()) != -1) {osw.write(c);}isr.close();osw.close();}public static void demo1() throws FileNotFoundException, IOException {//用默认编码表读写,出现乱码FileReader fr = new FileReader("utf-8.txt");FileWriter fw = new FileWriter("gbk.txt");int c;while((c = fr.read()) != -1) {fw.write(c);}fr.close();fw.close();}}
13. 转换流图解
* 画图分析转换流

14. 获取文本上字符出现的次数
* 获取一个文本上每个字符出现的次数,将结果写在times.txt上
public class Test3 {/** * 获取一个文本上每个字符出现的次数,将结果写在times.txt上 *  * 分析: * 1,创建带缓冲的输入流对象 * 2,创建双列集合对象TreeMap * 3,将读到的字符存储在双列集合中,存储的时候要做判断,如果不包含这个键,就将键和1存储,如果包含这个键,就将该键和值加1存储 * 4,关闭输入流 * 5,创建输出流对象 * 6,遍历集合将集合中的内容写到times.txt中 * 7,关闭输出流 * @throws IOException  */public static void main(String[] args) throws IOException {//1,创建带缓冲的输入流对象BufferedReader br = new BufferedReader(new FileReader("zzz.txt"));//2,创建双列集合对象TreeMapTreeMap<Character, Integer> tm = new TreeMap<>();//3,将读到的字符存储在双列集合中,存储的时候要做判断,如果不包含这个键,就将键和1存储,如果包含这个键,就将该键和值加1存储int ch;while((ch = br.read()) != -1) {char c = (char)ch;//强制类型转换/*if(!tm.containsKey(c)) {tm.put(c, 1);}else {tm.put(c, tm.get(c) + 1);}*/tm.put(c, !tm.containsKey(c) ? 1 : tm.get(c) + 1);}//4,关闭输入流br.close();//5,创建输出流对象BufferedWriter bw = new BufferedWriter(new FileWriter("times.txt"));//6,遍历集合将集合中的内容写到times.txt中for(Character key : tm.keySet()) {switch (key) {case '\t':bw.write("\\t" + "=" + tm.get(key)); break;case '\n':bw.write("\\n" + "=" + tm.get(key)); break;case '\r':bw.write("\\r" + "=" + tm.get(key)); break;default:bw.write(key + "=" + tm.get(key)); //写出键和值break;}bw.newLine();}//7,关闭输出流bw.close();}}

15. 试用版软件
* 当我们下载一个试用版软件,没有购买正版的时候,每执行一次就会提醒我们还有多少次使用机会用学过的IO流知识,模拟试用版软件,试用10次机会,执行一次就提示一次您还有几次机会,如果次数到了提示请购买正版
public class Test4 {/** *  当我们下载一个试用版软件,没有购买正版的时候,每执行一次就会提醒我们还有多少次使用机会用学过的IO流知识,模拟试用版软件, *  试用10次机会,执行一次就提示一次您还有几次机会,如果次数到了提示请购买正版 * @throws IOException  * 分析: * 1,创建带缓冲的输入流对象,因为要使用readLine方法,可以保证数据的原样性 * 2,将读到的字符串转换为int数 * 3,对int数进行判断,如果大于0,就将其--写回去,如果不大于0,就提示请购买正版 * 4,在if判断中要将--的结果打印,并将结果通过输出流写到文件上 */public static void main(String[] args) throws IOException {//1,创建带缓冲的输入流对象,因为要使用readLine方法,可以保证数据的原样性BufferedReader br = new BufferedReader(new FileReader("config.txt"));//2,将读到的字符串转换为int数String line = br.readLine();int times = Integer.parseInt(line);//将数字字符串转换为数字//3,对int数进行判断,如果大于0,就将其--写回去,如果不大于0,就提示请购买正版if(times > 0) {//4,在if判断中要将--的结果打印,并将结果通过输出流写到文件上System.out.println("您还有" + times-- + "次机会");FileWriter fw = new FileWriter("config.txt");fw.write(times + "");fw.close();}else {System.out.println("您的试用次数已到,请购买正版");}//关闭流br.close();}}