javaIO流深入研究

java IO流

1、文件

文件流

文件在程序中是以流的方式来操作的。

image-20220926141452345

:数据在数据源(文件)和程序(内存)之间经历的路径

输入流:数据从数据源(文件)到程序(内存)的路径

输出流:数据从程序(内存)到数据源(文件)的路径

常用文件操作

创建文件

类名:File

类图:

image-20220926144847845

构造方法:

image-20220926142041046

代码:

package IO;
import java.io.File;
import java.io.IOException;

public class Filenew {
    public static void main(String[] args) {
        newf();
        newf1();
        newf2();
    }
//    方式一
    public static void newf(){
        String url="C:\\Users\\lenovo\\Desktop\\ok.txt";
        File file = new File(url);
        try {
            file.createNewFile();
            System.out.println("创建成功");
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    //    方式二
    public static void newf1(){
        File pfile=new File("C:\\Users\\lenovo\\Desktop\\");
        String filename="ok1.txt";
        File file = new File(pfile,filename);
        try {
            file.createNewFile();
            System.out.println("创建成功");
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    //    方式三
    public static void newf2(){
        String pname="C:\\Users\\lenovo\\Desktop\\";
        String filename="ok2.txt";
        File file = new File(pname,filename);
        try {
            file.createNewFile();
            System.out.println("创建成功");
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}
获取文件相关信息

代码:

package IO;

import java.io.File;
import java.io.IOException;

public class FileInfo {
    public static void main(String[] args) throws IOException {
        info();
    }
    public static void info() throws IOException {
        File file=new File("C:\\Users\\lenovo\\Desktop\\hello.txt");
        file.createNewFile();
//        1、获取类名
        System.out.println(file.getName());
//        2、获取绝对路径
        System.out.println(file.getAbsolutePath());
//        3、获取文件父级目录
        System.out.println(file.getParent());
//        4、获取文件大小(字节)
        System.out.println(file.length());
//        5、文件是否存在
        System.out.println(file.exists());
//        6、文件是否属于一个父级目录
        System.out.println(file.isDirectory());
    }
}
目录的操作和文件的删除

mkdir创建一级目录、mkdirs创建多级目录、delete删除空目录或文件

image-20220926150046017

代码:

package IO;

import java.io.File;

public class FileD {
    public static void main(String[] args) {
//        m1();
        m2();
    }
//    文件存在就删除,文件不可为目录
    public static void m1(){
        File file =new File("C:\\Users\\lenovo\\Desktop\\hello.txt");
        if (file.exists()){
            if (file.delete()){
                System.out.println("删除成功");
            }else {
                System.out.println("文件属于目录");
            }
        }else {
            System.out.println("文件不存在");
        }
    }
//    目录存在提示,不存在创建
    public static void m2(){
        File file =new File("C:\\Users\\lenovo\\Desktop\\hello");
        if (file.exists()){
            System.out.println("目录已存在");
        }else {
            System.out.println("开始创建目录");
            if (file.mkdir()){
                System.out.println("创建成功");
            }else {
                System.out.println("创建失败");
            }
        }
    }
}

2、IO流原理及流的分类

原理:

1.I/O是Input/Output的缩写,I/O技术是非常实用的技术,用于处理数据传输。
如读/写文件,网络通讯等。

2.Java程序中,对于数据的输入/输出操作以”流(stream)”的方式进行。

3.java.io包下提供了各种“流”类和接口,用以获取不同种类的数据,并通过方法输入或输出数据。

4.输入input:读取外部数据(磁盘、光盘等存储设备的数据)到程序(内存)中。

5.输出output:将程序(内存)数据输出到磁盘、光盘等存储设备中

流的分类

按操作数据单位不同分为: 字节流(8 bit),字符流(按字符)

按数据流的流向不同分为: 输入流,输出流

按流的角色的不同分为: 节点流,处理流/包装流

image-20220926152253320

四个基础抽象类

详细分类:

image-20220926152748914

常用类

InputStream常用的子类:

  1. FilelnputStream:文件输入流
  2. BufferedlnputStream:缓冲字节输入流
  3. ObjectlnputStream:对象字节输入流

InputStream类图:

image-20220926153738976
FileInputStream使用:
package IO;

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;

public class InputStreamIO {
    public static void main(String[] args) throws IOException {
        readfile();
        readfile1();
    }
//    1.0 单个字节读取
    public static void readfile() throws IOException {
        String filename = "C:\\Users\\lenovo\\Desktop\\hello.txt";
        int readData = 0;
        char[] cs = new char[20];
        FileInputStream finput = null;
        try {
            finput = new FileInputStream(filename);
            int i = 0;
            while ((readData = finput.read()) != -1) {
                cs[i] = (char) readData;
                i++;
            }
            System.out.println(String.valueOf(cs));
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
//            关闭流
            finput.close();
        }
    }
//    2.0 字节数组读取
    public static void readfile1() throws IOException {
        String filename = "C:\\Users\\lenovo\\Desktop\\hello.txt";
        int readlength = 0;
        byte[] buf=new byte[8];
        FileInputStream finput = null;
        try {
            finput = new FileInputStream(filename);
            while ((readlength = finput.read(buf)) != -1) {
                System.out.print(new String(buf));
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
    //            关闭流
            finput.close();
        }
    }
}
FileOutputSream使用:
package IO;
import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.charset.StandardCharsets;

public class OutputStreamIO {
    public static void main(String[] args) throws IOException {
        outf();
    }
    public static void outf() throws IOException {
        String filename = "C:\\Users\\lenovo\\Desktop\\hello.txt";
        FileOutputStream foutput = null;
        String str="ok";
        try {
//            foutput=new FileOutputStream(filename);//覆盖原内容的写入方式
            foutput=new FileOutputStream(filename,true);//原内容基础上追加的写入方式
            foutput.write(str.getBytes(StandardCharsets.UTF_8));
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            foutput.close();
        }
    }
}
文件拷贝:
package IO;

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;

public class Filecopy {
    public static void main(String[] args) throws IOException {
        copy();
    }
    public static void copy() throws IOException {
        String filenm1="D:\\study.png";
        String filenm2="D:\\study1.png";
        FileInputStream finput=null;
        FileOutputStream foutput=null;

        try {
            finput=new FileInputStream(filenm1);
            foutput=new FileOutputStream(filenm2);
            int length=0;
            byte[] buf=new byte[1024];
            while ((length = finput.read(buf))!=-1){
                foutput.write(buf,0,length);//一定使用此方法
            }
            System.out.println("成功");
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }finally {
            finput.close();
            foutput.close();
        }
    }
}

FileReader和 FileWriter是字符流,即按照字符来操作io

FileReader

FileReader相关方法:

new FileReader(File/String)

read:每次读取单个字符,返回该字符,如果到文件末尾返回-1

read(char[]):批量读取多个字符到数组,返回读取到的字符数,如果到文件末尾返回-1相关API:

new String(char[]):将char[]转换成String

new String(char[],off,len):将char[]的指定部分转换成String

代码:

package IO;

import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;

public class Filereaders {
    public static void main(String[] args) throws IOException {
        read1();
    }
//    方式一单个字符读取
    public static void read() throws IOException {
        String filename="C:\\Users\\lenovo\\Desktop\\hello.txt";
        FileReader reader=null;
        int a=0;
        try {
            reader=new FileReader(filename);
            while ((a=reader.read())!=-1){
                System.out.print((char)a);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            reader.close();
        }
    }
    //    方式二字符数组读取
    public static void read1() throws IOException {
        String filename="C:\\Users\\lenovo\\Desktop\\hello.txt";
        FileReader reader=null;
        char[] buf=new char[8];
        int length=0;
        try {
            reader=new FileReader(filename);
            while ((length = reader.read(buf))!=-1){
                System.out.print(new String(buf,0,length));
            }
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            reader.close();
        }
    }
}
FileWriter

FileWriter常用方法:

new FileWriter(File/String):覆盖模式,相当于流的指针在首端

new FileWriter(File/String,true):追加模式,相当于流的指针在尾端write(int):写入单个字符

write(char[]):写入指定数组

write(char[],off,len):写入指定数组的指定部分write (string):写入整个字符串

write(string,off,len):写入字符串的指定部分相关APl: String类:toCharArray:将String转换成char[]

FileWriter使用后,必须要关闭(close)或刷新(flush),否则写入不到指定的文件!

代码:

package IO;

import java.io.FileWriter;
import java.io.IOException;

public class Filewriters {
    public static void main(String[] args) throws IOException {
        writer();
    }
    public static void writer() throws IOException {
        String filename="C:\\Users\\lenovo\\Desktop\\hello.txt";
        FileWriter writer=null;
        String str="成功写入\n";
        char[] chars=str.toCharArray();
        try {
            writer=new FileWriter(filename,true);
            writer.write(chars);
            writer.write(chars,0,2);
            writer.write(str);
            writer.write(str,0,2);
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            writer.close();
        }
    }
}

3、节点流和处理流

基本介绍:

1.节点流可以从一个特定的数据源读写数据,如FileReader、FileWriter [源码]

2.处理流(也叫包装流)是“连接”在已存在的流(节点流或处理流)之上,为程序提供更为强大的读写功能,如BufferedReader、BufferedWriter [源码]

节点流和处理流一览图

image-20220927155011935

节点流和处理流的区别和联系

1、节点流是底层流/低级流,直接跟数据源相接。

2、处理流(包装流)包装节点流,既可以消除不同节点流的实现差异,也可以提供更方便的方法来完成输入输出。[源码理解]

3、处理流(也叫包装流)对节点流进行包装,使用了修饰器设计模式,不会直接与数据源相连【模拟修饰器设计模式]

处理流的功能主要体现在以下两个方面:

1、性能的提高:主要以增加缓冲的方式来提高输入输出的效率。

2、操作的便捷:处理流可能提供了一系列便捷的方法来一次输入输出大批量的数据,使用更加灵活方便

***扩展:修饰器设计模式

常用处理流

字符流:BufferedReader和BufferedWriter

BufferedReader:

package IO;

import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;

public class BufferR {
    public static void main(String[] args) throws IOException {
        String filename="C:\\Users\\lenovo\\Desktop\\hello.txt";
        BufferedReader bufR=new BufferedReader(new FileReader(filename));
        String str="";
        while ((str = bufR.readLine())!=null){
            System.out.println(str);
        }
//         bufR.readLine();按行读取 性能效率高
        bufR.close();
    }
}

BufferedWriter:

package IO;

import java.io.BufferedWriter;
import java.io.FileWriter;
import java.io.IOException;

public class BufferW {
    public static void main(String[] args) throws IOException {
        String filename="C:\\Users\\lenovo\\Desktop\\hello1.txt";
        BufferedWriter bufW=new BufferedWriter(new FileWriter(filename,true));
        String str="你好";
        bufW.write(str);
        bufW.newLine();//插入换行
        bufW.write("io流");

        bufW.close();
    }
}

不要去操作二进制文件,会产生损坏的文件

常见二进制文件:声音、视频、doc、PDF

字节流:BufferedInputStream和BufferedOutputStream

代码:

package IO;

import java.io.*;

public class BufferedIN_OUT_copy {
    public static void main(String[] args) throws IOException {
        String filename1="C:\\Users\\lenovo\\Desktop\\XXXX.jpg";
        String filename2="C:\\Users\\lenovo\\Desktop\\XXXX1.jpg";
        BufferedInputStream bis=null;
        BufferedOutputStream bos=null;
        try {
            bis=new BufferedInputStream(new FileInputStream(filename1));
            bos=new BufferedOutputStream(new FileOutputStream(filename2));
            byte[] buf=new byte[1024];
            int length=0;
            while ((length = bis.read(buf))!=-1){
                bos.write(buf,0,length);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            bis.close();
            bos.close();
        }
    }
}
对象流:ObjectlnputStream和ObjectOutputStream

序列化和反序列化:
1、序列化就是在保存数据时,保存数据的值和数据类型

2、反序列化就是在恢复数据时,恢复数据的值和数据类型

3、需要让某个对象支持序列化机制,则必须让其类是可序列化的,为了让某个类是可序列化的,该类必须实现如下两个接口之一:

Serializable //这是一个标记接口,没有方法,一般使用此接口

Externalizable

注意事项:

1、读写顺序要一致

2、要求实现序列化或反序列化对象,需要实现 Serializable

3、序列化的类中建议添加SerialVersionUID.为了提高版本的兼容性

4、序列化对象时,默认将里面所有属性都进行序列化,但除了static或**transient(作用就是保证数据不被序列化)**修饰的成员

5、序列化对象时,要求里面属性的类型也需要实现序列化接口

6、序列化具备可继承性,也就是如果某类已经实现了序列化,则它的所有子类也已经默认实现了序列化

ObjectlnputStream:

package IO;

import java.io.FileInputStream;
import java.io.ObjectInputStream;

/*
* 反序列化读取数据
* */
public class ObjectIN {
    public static void main(String[] args) throws Exception {
        String filename="C:\\Users\\lenovo\\Desktop\\data.dat";
        ObjectInputStream oos=new ObjectInputStream(new FileInputStream(filename));
        System.out.println(oos.readInt());
        System.out.println(oos.readBoolean());
        System.out.println(oos.readChar());
        System.out.println(oos.readDouble());
        System.out.println(oos.readUTF());
        Object dog=oos.readObject();
        System.out.println(dog.toString());
        oos.close();
    }
}

ObjectOutputStream:

package IO;
import java.io.FileOutputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
/*
* 序列化存数据
* */
public class ObjectOUT {
    public static void main(String[] args) throws Exception {
//        dat文件不是纯文本
        String filename="C:\\Users\\lenovo\\Desktop\\data.dat";
        ObjectOutputStream ois=new ObjectOutputStream(new FileOutputStream(filename));
//        存储序列化数据 ,转化为包装类,包装类均实现Serializable接口
        ois.writeInt(100);//int->Integer
        ois.writeBoolean(true);//boolean->Boolean
        ois.writeChar('h');//char->Character
        ois.writeDouble(9.4);//double->Double
        ois.writeUTF("你好");//String
        ois.writeObject(new Dog("旺财",3));
        ois.close();
        System.out.println("保存完毕");
    }
}
class Dog implements Serializable {
    private String name;
    private int age;

    public Dog(String name, int age) {
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return "Dog{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}
转换流InputStreamReader和 OutputStreamWriter

介绍

1.InputStreamReader:Reader的子类,可以将InputStream(字节流)包装成Reader(字符流)。

2.OutputStreamWriter:Writer的子类,实现将OutputStream(字节流)包装成Writer(字符流)。

3.当处理纯文本数据时,如果使用字符流效率更高,并且可以有效解决中文问题,所以建议将字节流转换成字符流。

4.可以在使用时指定编码格式(比如utf-8, gbk , gb2312, ISO8859-1等)

处理编码格式不统一

写入:

package IO;

import java.io.*;

public class OutputStreamW {
    public static void main(String[] args) throws IOException {
        String filename="C:\\Users\\lenovo\\Desktop\\hello2.txt";
        BufferedWriter bw=new BufferedWriter(new OutputStreamWriter(new FileOutputStream(filename),"gbk"));
        bw.write("怎么说,我是gbk");
        bw.close();
    }
}

读取:

package IO;

import java.io.BufferedReader;
import java.io.FileInputStream;
import java.io.InputStreamReader;

public class InputStreamR {
    public static void main(String[] args) throws Exception {
        String filename="C:\\Users\\lenovo\\Desktop\\hello1.txt";
//        InputStreamReader转换流指定编码gbk
        InputStreamReader isr = new InputStreamReader(new FileInputStream(filename), "gbk");
        BufferedReader br=new BufferedReader(isr);//使用处理流提高效率
        String s=br.readLine();
        System.out.println(s);
        br.close();
    }
}
打印流 PrintStream,PrintWriter
package IO;

import java.io.IOException;
import java.io.PrintStream;
import java.nio.charset.StandardCharsets;

public class PrintSm {
    public static void main(String[] args) throws IOException {
        PrintStream ps=System.out;
        ps.println("成功");
//        等效方法
        ps.write("也成功".getBytes(StandardCharsets.UTF_8));
        ps.close();
//        修改打印位置
        System.setOut(new PrintStream("C:\\a.txt"));
        System.out.println("hhh");//将写入文件中
    }
}
package IO;

import java.io.FileWriter;
import java.io.IOException;
import java.io.PrintWriter;

public class PrintW {
    public static void main(String[] args) throws IOException {
        PrintWriter pw=new PrintWriter(System.out);//获取系统打印
        PrintWriter pw1=new PrintWriter(new FileWriter("XXXX"));//文件打印
    }
}

4、Properties类

Properties的常见方法:

load:加载配置文件的键值对到Properties对象

list:将数据显示到指定设备

getProperty(key):根据键获取值

setProperty(key,value):设置键值对到Properties对象

store:将Properties中的键值对存储到配置文件,在idea中,保存信息到配置文件,如果含有中文,会存储为unicode码

读取需求文件:

image-20220928153211323

传统方法

package IO;

import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;

public class PropertiesR {
    public static void main(String[] args) throws IOException {
        BufferedReader br=new BufferedReader(new FileReader("src\\mysql.properties"));
        String line="";
        while ((line=br.readLine())!=null){
           String[] S= line.split("=");
           System.out.println(S[1].replace(" ",""));
        }
        br.close();
    }
}

使用Properties类实现:

package IO;

import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.util.Properties;

public class PropertiesR1 {
    public static void main(String[] args) throws IOException {
//        1、创建properties类
        Properties p=new Properties();
//        2、加载指定的配置文件
        p.load(new FileReader("src\\mysql.properties"));
//        3、list方法显示键值对,传参PrintStream流
        p.list(System.out);
//        4、通过key值获取value
        System.out.println(p.getProperty("ip"));
//        5、传入键值对,添加数据
        p.setProperty("psw1","12345678");
//        6、将键值对添加到文件中
        p.store(new FileWriter("src\\mysql.properties"),"此参数为添加的注解");//覆盖式添加
        p.list(System.out);
    }
}
  System.out.println(S[1].replace(" ",""));
    }
    br.close();
}

}


###  使用Properties类实现:

```java
package IO;

import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.util.Properties;

public class PropertiesR1 {
    public static void main(String[] args) throws IOException {
//        1、创建properties类
        Properties p=new Properties();
//        2、加载指定的配置文件
        p.load(new FileReader("src\\mysql.properties"));
//        3、list方法显示键值对,传参PrintStream流
        p.list(System.out);
//        4、通过key值获取value
        System.out.println(p.getProperty("ip"));
//        5、传入键值对,添加数据
        p.setProperty("psw1","12345678");
//        6、将键值对添加到文件中
        p.store(new FileWriter("src\\mysql.properties"),"此参数为添加的注解");//覆盖式添加
        p.list(System.out);
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

爱打辅助的小可爱

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值