Java 你往我来的 IO 流,适合小白或傻傻分不清【输入、输出流】的伙计们 —— 第 04 篇 Java 的 字符输出流:Writer 字符输入流:Reader

Java 你往我来的 IO 流【第 03 篇】 Java 的 字符输出、输入流

字符输出流:Writer

字符输入流:Reader

        关于 Java 常用的 IO 流,整理了 07 篇文章,这是第 04 篇。如果是小白,或者一直没弄清楚 IO 流,请依次把这几篇文章过一遍。当然,高手就不用来我的博客浪费青春了。

        小二,上代码:


import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.Reader;
import java.io.Writer;

/*
 * 这个类的知识点(字符流)
 * 1、使用 字符输出流 Writer(抽象父类)-> FileWriter(子类)把数据写入文件
 * 2、使用 字符输入流 Reader(抽象父类)-> FileReader(子类)把数据读到内存
 */
public class CharStream
{
    public static void main(String[] args)
    {
        // 输出流:将数据从内存写入硬盘文件等,简称:写入输出外界(Writer、OutputStream)
        // 输入流:将数据从硬盘文件等写入内存,简称:读取输入内存(Reader、InputStream)
        // 流分两种:字节流和字符流
        // 字节流适用场景:主要用来处理字节或二进制对象。如:音频文件、图片、歌曲
        // 字符流适用场景:主要用来处理字符或字符串。如:关系到中文(文本)
        
        File file = new File("D:" + File.separator + "temp" + File.separator + "test.txt"); // 在 Windows 下的路径分隔符和 Linux 下的路径分隔符是不一样的,如 Windows 的路径:D:\test.txt,而 Linux 则是:D:/test.txt,所以如果考虑跨平台,最好是用 File.separator 去实现分隔符,它能兼容二者
        
        // 创建目录(这里如果不执行创建,那么必须保证 D 盘上有 temp 目录)
        //file.getParentFile().mkdirs();
        // 创建文件。这里不用手动创建也可以,因为下面在创建 FileWriter 实例时会自动创建
        //file.createNewFile();
        
        // 调用字符输出流方法(把内存数据写入文件)
        writeToFile(file);
        
        // 调用字符输入流方法(把文件数据读取到内存)
        readFromFile(file);
    }
    
    // 字符输出流(把内存数据写入文件)
    public static void writeToFile(File file)
    {
        Writer writer = null; // 为什么不放到 try 里面声明?如果把它放 try 里面声明,那么它的作用域仅限于 try,即只能在 try 里面使用它
        try
        {
            writer = new FileWriter(file); // FileWriter 是 Writer(抽象类,不能实例化)的子类
            writer.write("向 text.txt 文件输入数据\r\n换行再输入数据");
            // 字符流使用了缓冲区,将数据写入文件时,数据并没有直接写入文件,而是保存在缓冲区,缓冲区满了之后再写入文件。flush() 方法强制将数据立即写入文件并清空缓冲区
            // flush() 不是必须的,但一定要调用 close() 方法
            // 如果 flush() 和 close() 方法都不调用,那么数据不能写入文件
            writer.flush();
        } catch (IOException e)
        {
            e.printStackTrace();
        } finally // finally 作用:无论是否抛出异常,finally 代码块总是会被执行,这里用来释放资源
        {
            try
            {
                if(writer != null)
                {
                    writer.close(); // 关闭流(如果不关闭流,那么相应的垃圾回收机制就不进行回收,导致垃圾越来越多,浪费资源)
                }
            } catch (IOException e)
            {
                e.printStackTrace();
            }
        }
    }
    
    // 字符输入流(把文件数据读取到内存)
    public static void readFromFile(File file)
    {
        Reader reader = null; // 为什么不放到 try 里面声明?如果把它放 try 里面声明,那么它的作用域仅限于 try,即只能在 try 里面使用它
        try
        {
            reader = new FileReader(file); // FileReader 是Reader(抽象类,不能实例化)的子类
            char[] c = new char[1024];
            int len; // 每次读取的字符数
            // 一个文件可能很大,每次读取 1024 个字符数据并放到字符变量 c 里面,用 while 循环判断是否读完,如果没有读完,这里将继续读后面的数据,直到全部读完返回 -1 结束循环
            while((len = reader.read(c)) != -1) // 如果达到流的末尾返回 -1(把文件读完之后返回 -1)
            {
                System.out.print(new String(c,0,len)); // new String(b) 将字节转换为字符串。0 和 len 代表从第 0 个到第 len 个,len 是每次读取数据的长度
            }
        } catch (FileNotFoundException e)
        {
            e.printStackTrace();
        } catch (IOException e)
        {
            e.printStackTrace();
        } finally // finally 作用:无论是否抛出异常,finally 代码块总是会被执行,这里用来释放资源
        {
            try
            {
                if(reader != null)
                {
                    reader.close(); // 关闭流(如果不关闭流,那么相应的垃圾回收机制就不进行回收,导致垃圾越来越多,浪费资源)                    
                }
            } catch (IOException e)
            {
                e.printStackTrace();
            }
        }
    }
}


你可能感兴趣的:(Writer,Reader)