java——IO流

一. File

  • File类可以使用文件路径字符串来创建File实例,该文件路径可以是绝对路径或相对路径
  • File类的list()方法中可以接收一个FilenameFilter参数,通过该参数可以只列出符合条件的文件

二. 流(Stream)

  • Stream是从起源(source)到接收(sink)的有序数据
  • 按照流向分可以分为输入流和输出流 
    • 输入流:只能从中读取数据,不能写入数据(基类是InputStream和Reader)
    • 输出流:只能向其中写入数据,不能读取数据(基类是OutputStream和Writer)
  • 按照操作的数据单元分为字节流和字符流 
    • 字节流:操作的数据单元是8位的字节(基类是InputStream和OutputStream)
    • 字符流:操作的数据单元是16位的字节(基类时Reader和Writer)
  • 按照角色可以分为节点流和处理流 
    • 节点流:可以从/向一个特定的IO设备中读/写数据的流,也被称为低级流
    • 处理流:用于对一个已存在的流进行连接或封装来实现读/写功能,也称为高级流或包装流

三、   数据流的基本概念

1.数据流

在Java中把不同的数据源与程序之间的数据传输都抽象表述为“流”(stream),以实现相对统一和简单的输入/输出操作方式。传输中的数据就像流水一样,也称为数据流。

2 .I/O数据流的分类方式

数据流分为输入流和输出流两类。输入流只能读取不能写。而输出流只能写不能读。(这里站在程序的角度来确定出入方向,即将数据从程序外部传送到程序中谓之“输入”数据,将程序中的数据传送到外部谓之“输出”数据。

3.缓冲流(Buffered Stream)

对数据流的每次操作都是以字节为单位进行的,既可以向输出流写入一个字节,也可从输入流中读取一个字节。显然效率太低,通常使用缓冲流,即为一个流配置一个缓冲区,一个缓冲区就是专门传送数据的一块内存。

4.数据源(Data Sourcc):是指那些能够提供数据的地方,包括键盘、磁盘文件、网络接口等。

5.数据宿(Data Sink):指能够接收数据的地方,可以是磁盘文件、网络接口以及显示器、打印机等外部设备。(数据宿也可认为是数据传输的目的地)

二、节点流和处理流

根据数据流所关联的是数据源还是其他数据流,可分为节点流(Node Stream)和处理流(Processing Stream)

节点流可以从/向一个特定的地方读/写数据。

处理流是对一个已存在的流的连接和封装,通过所封装的流的功能调用实现增强的数据读/写功能,处理流并不直接连接到数据源。

二、   字符流

1.字符输入流Reader

Reader类为所有面向字符的输入流的超类,声明为java.io中的抽象类。

public int read():读取一个字符,返回的是读到的那个字符。如果读到流的末尾,返回-1。

public int read(char[] cbuf):将读到的字符存入指定的数组中,返回的是实际读取的字符数。如果读到流的末尾,返回-1。

public abstract int read(char[] cbuf,int off,int len):将读到的字符存入数组的指定位置(off),每次最多读len个字符,返回实际读取的字符数。如果读到流的末尾,返回-1。

close():读取字符其实用的是window系统的功能,使用完毕后,进行资源的释放。

 

2.字符输出流writer

Weiter类为所有面向字符的输出流的超类,声明为java.io中的抽象类。

public void write(int c):将一个字符写入到流中。

public void write(char[]):将数组中的字符依次写出。

public abstract void write(char[] bcbuf,int off,int len):将数组中下标off开始的len个字符写出。

public void write(String):将一个字符串写入到流中。

public abstract void flush():刷新流,将流中的数据刷新到目的地中,流还存在。

public abstreact void close():关闭资源,关闭前会先调用flush,刷新流中的数据去目的地,然后流关闭。

3. FileWriter的使用

该类没有特有的方法。只有自己的构造数。

该类特点:

1.用于处理文本文件。

2.该类中有默认的编码表,

3.该类中有临时缓冲。

构造函数

publicFileWriter(Stringfilename);//调用系统资源,在指定位置,创建一个文件。注意:如果该文件已存在,将会被覆盖。

IO流的几个相关联系题:

eg1:编写Java程序:删除指定的文件或文件夹,如果文件夹中有文件或子文件夹也一并删除。

public void deleteFile(File file){
        //分析:
        //1.首先判断该路径file是否存在,如果不存在,即终止删除程序,如果存在,进行第2步:
        //2.判断路径file是文件还是文件夹,若为文件,则直接删除,若为文件夹,则进入第3布:
        //3.判断文件夹是否为空,若为空,直接删除该文件夹
        //4.若不为空,则记录文件夹下所有文件列表,利用for-each与递归算法删除所有内容,最后删除该文件夹本身
        if(!file.exists()){
            return;
        }
        if(file.isFile()){
            file.delete();
            System.out.println("正在删除文件:"+file.getAbsolutePath());
            return;
        }
        if(file.isDirectory()){
            if(file.list().length==0){
                file.delete();
                System.out.println("正在删除文件夹:"+file.getAbsolutePath());
                return;
            }
            File[] file1 = file.listFiles();
            for(File f:file1){
                deleteFile(f);
            }
            file.delete();
        }
    }

eg2:编写一个Java程序,要求每次运行该程序时打印出当前是第几次运行,即使计算机重启也能正常计数。
 

 1 public void getCountTest() throws IOException{
 2         //注意:两个流不能同时访问同一个文件!
 3         //DataOutputStream data1 = new DataOutputStream(new FileOutputStream(file));
 4         //该数据输出流每执行一次都会将原file路径下文件内容刷新一次(即覆盖)
 5         File file = new File("E:/count");
 6         if(!file.exists()){//判断file路径是否存在
 7             file.createNewFile();//创建file路径指定的文件count
 8         }
 9         DataInputStream data = new DataInputStream(new FileInputStream(file));
10         int x;
11         try {
12             x=data.readInt();
13             data.close();
14         } catch (EOFException e) { //若x=data.readInt();读不到数据(如果此输入流在读取这四个字节之前到达末尾),则会抛出EOFException异常
15             data.close();//在使用输出对象data1之前关闭输入管道,避免同时操作
16             DataOutputStream data1 = new DataOutputStream(new FileOutputStream(file));
17             data1.writeInt(1);
18             data1.flush();
19             data1.close();
20             System.out.println("程序第"+1+"次运行");
21             return;
22         }
23         DataOutputStream data1 = new DataOutputStream(new FileOutputStream(file));
24         data1.writeInt(++x);
25         System.out.println("程序第"+x+"次运行");
26         data1.flush();
27         data1.close();
28     }

 eg3:编写Java程序:实现文件夹的复制:

  |--该方法实现了将路径file2下文件拷贝至路径file1下文件

 1 @Test
 2     public void testCopy() throws IOException {
 3         File file1 = new File("E:\");
 4         File file2 = new File("E:\Desktop\饮水机项目_20161221_4");
 5         copyOne(file1,file2);
 6     }
 7     public void copyOne(File file1,File file2) throws IOException{
 8         if(!file2.exists()){
 9             return;
10         }
11         if(file2.isFile()){
12             InputStream in = new FileInputStream(file2);
13             OutputStream out = new FileOutputStream(new File(file1,file2.getName()));
14             byte[] b = new byte[1024*1024];
15             int len=-1;
18         while((len=in.read(b))!=-1){
19                 out.write(b,0,len);
20             }
21             in.close();
22             out.flush();
23             out.close();
24             return;
25         }
26         if(file2.isDirectory()){
27             File temp = new File(file1,file2.getName());
28             temp.mkdir();
29             if(file2.list().length==0){
30                 return;
31             }
32             File[] fi = file2.listFiles();
33             for(File f:fi){
34                 copyOne(temp,f);
35             }
36         }
37     }

 eg4:开发一个Java程序,实现将一个GBK编码的文件转为UTF-8编码的文件。

 1 @Test
 2     public void codeChange() throws Exception{
 3         //桥接流:字节流通向字符流的桥接流
 4         InputStreamReader in = new InputStreamReader(
 5                 new FileInputStream("E:/my.java"),"GBK");    //解码
 6         //桥接流:字符流通向字节流的桥接流
 7         OutputStreamWriter out = new OutputStreamWriter(
 8                 new FileOutputStream("E:/your.java"),"UTF-8");    //编码
 9         int k = -1;
10         while((k=in.read())!=-1){
11             out.write(k);
12         }
13         in.close();
14         out.flush();
15         out.close();
16     }
17     @Test //通过桥接字符流的便捷流(子类流)FileReader和FileWriter来快速实现
18     public void codeChange2() throws Exception{
19         FileReader in = new FileReader("E:/my.txt");    //解码
20         FileWriter out = new FileWriter("E:/your.txt");    //编码
21         int k = -1;
22         while((k=in.read())!=-1){
23             out.write(k);
24         }
25         in.close();
26         out.flush();
27         out.close();
28     }

 eg5:将从控制台输入的信息保存到一个文件中去。(中文不能乱码,建议使用System.in、Scanner和InputStreamReader等)
 

 1 //方法一:
 2     @Test
 3     public void scannerTest() throws Exception{
 4         Scanner scanner = new Scanner(System.in);
 5         PrintWriter out = new PrintWriter("E:/myWrite");
 6         String line = null;
 7         while(!(line=scanner.nextLine()).equals("exit")){
 8             out.println(line);
 9         }
10         scanner.close();
11         out.flush();
12         out.close();
13     }
14     //方法二:
15     @Test
16     public void bufferedReaderTest() throws Exception{
17         BufferedReader in = new BufferedReader(new InputStreamReader(System.in));
18         PrintWriter out = new PrintWriter("E:/myWrite");
19         String line = null;
20         while(!(line=in.readLine()).equals("exit")){
21             out.println(line);
22         }
23         in.close();
24         out.flush();
25         out.close();
26     }

 eg6:开发一个Java程序,统计每个java源文件的代码行数和所有java源文件的总行数(不含空行)

 1 private static int count;
 2     @Test
 3     public void testGetAllCount() throws Exception{
 4         File file = new File("D:/workspace/Test");
 5         getAllCount(file);
 6         System.out.println(count);
 7     }
 8     public void getAllCount(File file) throws Exception{
 9         if(!file.exists()){
10             return;
11         }
12         if(file.isFile()){
13             BufferedReader buffer = new BufferedReader(
14                     new InputStreamReader(new FileInputStream(file)));
15             String line=null;
16             while((line=buffer.readLine())!=null){
17                 if(line.trim().length()!=0){
18                     count++;
19                 }
20             }
21             buffer.close();
22         }
23         if(file.isDirectory()){
24             for(File f:file.listFiles()){
25                 getAllCount(f);
26             }
27         }
28     }
原文地址:https://www.cnblogs.com/FrankLei/p/6244915.html