java的Io流机制的学习

时间:2019-08-18
本文章向大家介绍java的Io流机制的学习,主要包括java的Io流机制的学习使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

IO流机制

File类的使用

File类的构造方法

File(URI uri)

File(String pathname)

File(File parent, String child)

File(String parent, String child)

File类的常用方法

boolean exists():判断文件是否存在

boolean createNewFile() :创建一个新文件,只能创建一个文件,不能创建目录(文件夹),创建时先判断文件 是否存在 ,不存在则创建并返回true, 存在则返回false.

File getAbsoluteFile() File 类型的绝对路径File文件形式,相当于 File file =new File(绝对名称),

String getAbsolutePath() String类型的绝对路径的字符串形式

String getName() : 获取的是路径的最后一级的名称

String getPath() //获取的是相对路径的名称:打File对象事调用的就是getPath()方法

String Parent() //返回当前文件的上一级目录,若没有上一级目录返回null

bolean isDirectory() //判断是否是文件

bolean isFile() //判断是否是文件夹

bolean delete()//删除一个文件或文件夹

File[] listFiles()// 只能获取当前文件夹下的的所有文件

Long length() //获取点前文件夹的长度

public class Test01_01 {
    public static void main(String[] args) throws IOException {
        // 创建文件对象:绝对路径
        File f1 = new File("d:/a.txt");
        // 创建文件对象:相对路径
        File f2 = new File("a.txt");
        boolean b1 = f1.createNewFile();
        boolean b2 = f2.createNewFile();

        String path1 = f1.getAbsolutePath(); //d:\a.txt
        String path2 =  f2.getAbsolutePath(); //C:\idea_workspace\basic-java\a.txt
        System.out.println(path1+" "+path2);
        File fpath1 = f1.getAbsoluteFile(); //d:\a.txt
        File fpath2 = f2.getAbsoluteFile(); //C:\idea_workspace\basic-java\a.txt
        System.out.println(fpath1+" "+fpath2);

        String name1 = f1.getName(); //a.txt
        String name2 = f2.getName(); //a.txt
        System.out.println(name1+" "+name2);
        /*
        获取的是相对路径
         */
        String  p1 = f1.getPath();//d:\a.txt
        String p2 = f2.getPath(); //d:\a.txt
        System.out.println(p1+" "+p2);
        File f3 = new File("d:/aaa/aa");
        System.out.println(f3.getParent()+"f3....");
        System.out.println(f2.getParent()+"f2....");
        Boolean b = f3.mkdirs();
        System.out.println(b);
        File f4 = new File(f3,"a"); //在 aaa/aa文件夹下再创建一个文件目录a
        String parent = f4.getParent();
        System.out.println(parent);
        f4.mkdir();
        File f5 = new File("d:/aaa/aa/a","a.txt");
        System.out.println(f5.getParent());
        System.out.println(f4.getName()); //a
        f5.createNewFile();//在d:/aaa/aa/a下创建一个a.txt文件
        System.out.println(f5); //d:\aaa\aa\a\a.txt,实际是复写了toStirng()方法,在toStirng中调用了getPath()
        //删除文件或文件夹delete()
        f2.delete();
        f1.delete();

    }


}


    public static void main(String[] args) {
        // 创建文件对象
        File f = new File("d:/aaa/b.txt");
        // 获得文件名
        String filename = f.getName();
        // 获得文件大小
        long filesize = f.length();
        // 获得文件的绝对路径
        String path = f.getAbsolutePath();
        // 获得父文件夹路径,返回字符串
        String parentPath = f.getParent();
        // 获得父文件夹路径,返回文件对象
        File parentFile = f.getParentFile();
        // 输出信息
        System.out.println("文件名:" + filename);
        System.out.println("文件大小:" + filesize);
        System.out.println("文件路径:" + path);
        System.out.println("文件父路径:" + parentPath);
        System.out.println("文件父路径:" + parentFile);
    }

运行结果

判断是否是文件或文件夹

    public static void main(String[] args) {
        // 创建文件对象
        File f1 = new File("d:/b.txt");
        // 判断是否是一个文件
        if(f1.isFile()) {
            System.out.println(f1.getName()+"是一个文件");
        }  else {
            System.out.println(f1.getName()+"不是一个文件");
        }
        // 创建文件对象
        File f2 = new File("d:/aaaa");
        // 判断是否是一个文件夹
        if(f2.isDirectory()) {
            System.out.println(f2.getName()+"是一个文件夹");
        }  else {
            System.out.println(f2.getName()+"不是一个文件夹");
        }
    }

运行结果:

获取当前文件夹下的所有文件列表

public class Test01_08 {
    public static void main(String[] args) {
        // 创建文件对象
        File f = new File("d:/aaa");
        // 获得文件夹下所有文件
        File[] files = f.listFiles();
        // 遍历文件数组
        for (File file : files) {
            // 将文件的名字打印到控制台
            System.out.println(file.getName());
        }
    }
}

listFiles(FileFilter filter)、File listFiles(FilenameFilter filter)

、listFile的重写方法

File[] listFiles(FileFilter filter) //返回该文件夹下满足指定过滤器条件的文件和目录。

File listFiles(FilenameFilter filter)

FileFilter 和 FilenameFilter 是一个接口,接口中只定义了accept()一个方法,要使用该接口就要实现该方法

boolean accept(File pathname): FileFilter的方法,File pathname的File文件既有文件类型的也有目录类型的

boolean accept(File dir, String name) : FilenameFilter的方法,

注:File dir:的所有文件都被封装成了一个目录

File[] file = f.listFile(FileFilter filter)方法的执行步骤:

1.获取文件夹下的所有文件,将所有的文件都封装成File类型

2.遍历获取的文件,调用FileFilter的accept(File pathname)方法,将文件挨个传入,根据accept()中定义的规则筛选文件,判断方法的返回值, 若方法的返回值为true则把当前遍历的File文件存放在File[]数组中`

注: 调用boolean accept(File dir, String name)的原理大致相同,只是还多了一个功能,能接收的当前文件 夹的文件名,String name:接收的是文件名称,不是路径名称

public class Test01_09 {
    public static void main(String[] args) {
        File f = new File("d:/aaa/aa/a");
        /**
         *    d:/aaa/aaa/a下有两个文件:一个_a文件夹,一个a.txt文件
         *    要求:只获取d:/aaa/aaa/a下的文件,不获取文件夹
         */
         File[] file = f.listFiles(new FilenameFilter() {
             @Override
             public boolean accept(File pathname, String name) {
                 System.out.println(name); //a.txt /_a
                 if(name.endsWith(".txt")){
                     return true;
                 }
                 return false;
             }
         });
        System.out.println(file[0]);
    }

}

**注: 使用FilenameFilter filter,所有文件都被封装成目录即文案夹

流的分类

按流向分:

  • 输入流:从硬盘读取数据到内存
  • 输出流:从内存中读取数据到硬盘

按读取的类型分:

  • 字节流:读取时一次读取一个字节
    OutputStream: 输出字节流
    InPutStream: 输入字节流
  • 字符流:读取时一次读取一个字符
    Reader: 输入字符流
    Writer:输出字节流

InputStream&OutPutStram

InputStream: 字节输入流

FileInputStream:

从硬盘读取文件内容到内存的的字节输入流

构造方法:

FileInputStream(File file)

FileInputStream(String name)

常用方法:

int read() //返回的是对应的字节数据,当没有数据时返回-1

int read(byte[] b) //返回值为读取的字节个数,

一个一个字节读取到byte[]数组中,再将byte[]数组的内容传递给应用程序

int read(byte[] b, int off, int len) //返回值为从内存中读取的字节个数

一个一个字节读取,将字节数据存放在指定的数组位置

public class Test03 {
    public static void main(String[] args) throws IOException {
        // 创建字节输入流对象并关联文件
        FileInputStream fis = new FileInputStream("d:/c.txt");
        // 定义变量接收读取的字节
        int len = -1;
        byte[] bytes = new byte[6];
        // 循环从流中读取数据
        while((len = fis.read(bytes))>0) {
            System.out.print(new String(bytes,0,len));
        }
        // 关闭流
        fis.close();
    }
}

程序读取流的流程:

java程序 -----》找到jvm虚拟机-------->通知Os操作系统-----》由OS去硬盘读取文件

os拿到文件的数据------》通知OS操作系统----->找到JVM虚拟机------》将数据返还给java程序

OutPutStram:字节输出流

FileOutputStream:从内存中读取数据到硬盘的输出流,读取的时候一个字节一个字节读取,输出的时候也是一个字节一个字节的输出

构造方法:

  • FileOutputStream(File file) //创建一个输出流,将流关联到指定的输出目的地
  • FileOutputStream(String name) //创建一个输出流,将流关联到指定的输出目的地
  • FileOutputStream(File file, boolean append) //创建一个输出流,将流关联到指定的输出目的地,并指定是否将数据添加到文件末尾
    append = true:当流对象再与目的文件关联时,将数据添加到原文件的末尾
    append = false :当流对象再与目的文件关联时,覆盖原文件
  • FileOutputStream(String name, boolean append) //创建一个输出流,将流关联到指定的输出目的地,并指定是否将数据添加到文件末尾 append = true:将数据添加到原文件的末尾 append = false :覆盖原文件

常用方法:

  • void write(int b) ://将一个字节写到指定目的地,数字>=255,写出的数据内容为空
  • void write(byte[] b) // 将一个字节数组的数据写到目的地,写出的形式也是一个字节一个字节的写出到硬盘中
  • void write(byte[] b, int off, int len) 将指定位置的字节数组的数据写到硬盘
  • void close() //关闭流对象

    FileOutputStream fos = new FileInputStream("d/a.txt")的语义:

1.首先创建了一个输出流对象

2.输出流对象指向输出的目的地址d:/a.txt

fos.write(97)的语义:

流对象获取字节数据,将数据输出到指定的目的文件中

注:write(97)的97 代表的是一个字节,当写如文件中,文件展示是会根据编码表对数据进行解码,97在编码表中对应的是小写字母a

public class Test01 {
    public static void main(String[] args) throws IOException {
        // 1.创建字节输出流FileOutputStream对象并指定文件路径。
        FileOutputStream fos = new FileOutputStream("d:/a.txt");
        // 2.调用字节输出流的write(int byte)方法写出数据
        fos.write(254); //
        byte[] bytes = {97,98};
        fos.write(bytes);
        // 3.关闭流
        fos.close();
    }
}

文件的换行:

public class Test02 {
    public static void main(String[] args) throws IOException {
        // 1.创建字节输出流FileOutputStream对象并指定文件路径,并追加方式
        FileOutputStream fos = new FileOutputStream("d:/c.txt",true);
        // 2.调用字节输出流的write方法写出数据
        // 2.1 要输出的字符串
        String content = "i love java \r\n";
        for (int i = 0; i< 5; i++) {
            fos.write(content.getBytes());
        }
        // 3.关闭流
        fos.close();
    }
}

程序写出流的流程:

java程序 -----》找到jvm虚拟机-------->通知Os操作系统-----》由OS去内存读取文件

os拿到文件的数据------》通知OS操作系统----->找到JVM虚拟机------》将数据写入到硬盘

Reader&Writer

Reader:字符输入流

FileReader: 操作文件的字符输入流

构造方法:

FileReader(File file)

FileReader(String fileName)

常用方法:

int read()

int read(char[] cbuf) //读取数据时是按字符读取的,单底层又将字符转为字节对象,因此,返回值是字节的个数 l

例如:读取好aaahhh,

返回的int值为8,一个中文字符占2个字节

int read(char[] cbuf, int off, int len)

Writer:字符输出流

FileWriter:操作文件的字符输出流

构造方法:

  • FileWriter(File file)
  • FileWriter(String fileName)
  • FileWriter(File file, boolean append)
  • FileWriter(String fileName, boolean append)

常用方法:

  • void write(int c)
  • void write(char[] cbuf)
  • void write(char[] cbuf, int off, int len)
  • void write(String str)
  • void write(String str, int off, int len)
  • void flush() //将内存缓冲区的数据刷到目的文件中
  • FileWriter的write()方法实现机制

首选,将要写出的字符数据转为字节数据,放到内存缓冲区中,使用flush()将缓冲区中的数据写入到文件中,因此文件的接收的还是字节数据,再根据文件的编码参照对应的编码表将字节转为对应的字符

Properties(属性集)的使用

Properties的作用:

  • load(InputStream instream): //读取键值对文件到Properties集合
  • store(OutputStream out, String comments)://写出集合中的数据,形成一个键值对文件

Properties:是一个双列集合

properties extends hashmap ,hashmap implements map

properties:是一个唯一与Io流结合的集合

构造方法:

Properties() //默认范型为<String,String>

常用方法:

String getProperty(String key) ://底层调用map 的 get()方法

Object setProperty(String key, String value) //底层调用map 的 put()方法

set stringPropertyNames() //返回列表的键值,相当于map的keySet()方法

void load(InputStream inStream) //读取外部的键值对文件,将键值对的数据放在Properties集合中,InputStream关联键值对文件的地址

读取的键值对文件的形式一般都是 键 对应 值 ,键值对的连接一般使用 = 空格 或其他连接

void load(Reader reader)

void store(OutputStream out, String comments) //读取Properties集合,将集合中的数据已键值对文件形式存放在OutputStream关联的路径中

void store(Writer writer, String comments)

public class Test06 {
    public static void main(String[] args) throws IOException {
//1:创建一个空的集合
        Properties prop = new Properties();
//2:读取数据到集合中
        prop.load(new FileInputStream("d:/score.txt"));
//3:遍历集合,获取到每一个key
        Set<String> keys = prop.stringPropertyNames();
//获取到每一个key
        for (String key : keys) {
//4:判断当前的key 是否为 "lisi"
            if ("lisi".equals(key)) {
//把"lisi"的值设置为100
                prop.setProperty(key, "100");
            }
        }
//把集合中所有的信息,重新存储到文件中
        prop.store(new FileOutputStream("d://score.txt"), "haha");
    }

}

字节缓冲流

缓冲流的作用

在基本流的对象上创建,用于增强基本流的读写效率

四种基本流:

1.字节输入流

2.字节输出流

3.字符输入流

4.字符输出流

BufferedInpuStream: 字节缓冲输入流

构造方法:

  • BufferedInputStream(InputStream in)
  • BufferedInputStream(InputStream in, int size) //int size:指定了缓冲区的大小

常用方法:

继承了InputStream,有Inputstream的常用方法

BufferedInputStream(FileInputStream in) 的实现原理:

通过`BufferedInputStream创建了一个缓冲区,将缓冲区与流对象关联,FileInputStream in读取数据时不用频繁的来往于硬盘和内存之间,先将数据写入缓冲区中,最后将数据一次性的写入内存

public class Test07 {
    public static void main(String[] args) throws IOException {
        // 创建字节输出流FileOutputStream对象并指定文件路径。
        FileOutputStream fos = new FileOutputStream("d:\\c.txt");
        // 利用字节输出流创建高效字节输出流对象
        BufferedOutputStream bos = new BufferedOutputStream(fos);
        // 调用高效字节输出流对象的write(int byte)方法写出一个字节数据
        bos.write(97);
        // 关闭流
        bos.close();
    }
}

BufferedOutputStream: 字节缓冲输出流

  • 构造方法:

BufferedOutputStream(OutputStream out)

BufferedOutputStream(OutputStream out, int size) //int size:指定了缓冲区的大小

常用方法:

继承了OutputStream,有Outputstream的常用方法

BufferedOutputStream(FileOutputStream out) 的实现原理:

通过`BufferedInputStream创建了一个缓冲区,将缓冲区与流对象关联,FileOutputStream out写出数据时不用频繁的来往于内存和硬盘之间,先将要写出的数据写入缓冲区中,最后将数据一次性的写出到硬盘

    public static void main(String[] args) throws IOException {
        // 创建字节输出流FileOutputStream对象并指定文件路径。
        FileOutputStream fos = new FileOutputStream("c:\\e.txt");
        // 利用字节输出流创建高效字节输出流对象
        BufferedOutputStream bos = new BufferedOutputStream(fos);
        // 调用高效字节输出流对象的write(byte[] buff)方法写出一个字节数据
        bos.write("i love java".getBytes());
        // 关闭流
        bos.close();
    }

字符缓冲流

BufferedReader:字符缓冲输入流

构造方法:

BufferedReader(Reader in)

BufferedReader(Reader in, int sz)

特有方法:

String readLine() //读取一个文本行

BufferedWriter:字符缓冲输出流

构造方法:

BufferedWriter(Writer out)

BufferedWriter(Writer out, int sz)

特有方法:

void newLine() //写入一个行分隔符,根据系统识别分隔符

转换流

使用转换流的原因:

因为对于字符来说有不同的编码方式,不同的编码方式对应着不同的编码表,字符的编码和字符的解码应该具有一一对应的关系

编码码的种类

1.GBK编码:一个字符占2个字节

2.Utf-8编码: 一个字符占3个字节

3.ASCII编码

因为编码不同会出现的问题:

电脑本地上一个.txt文件,默认编码格式为1个字符两个字节,当idea(默认编码为utf-8),读取数据时,将.txt传来的数据3个字节默认为一个字符,超找utf-8对应的码表,此时就会出现乱码的情况

InputStreamReader:转换输入流

InputStreamReader extend Reder

构造方法:

InputStreamReader(InputStream in)

InputStreamReader(InputStream in, Charset cs) //指定对读取数据的解码方式

转换输入流的实现原理:

FileInputStream fis = new FileInputStream("a.txt");

InputStreamReader isr = new InputStreamReader(fis,"gbk");

1.创建转换输入流对象,以字符的形式读取 InputStreamReader关联的源数据

2.FileInputStream将字符对象转为字节对象,并指定其解码方式

public static void main(String[] args) throws IOException{
        // 创建字节输入流对象并关联文件
        FileInputStream fis = new FileInputStream("a.txt");
        // 创建转换输入流对象
        InputStreamReader isr = new InputStreamReader(fis,"gbk");
        // 定义字符数组存放读取的内容
        char[] buffer = newchar[1024];
        // 定义变量接收读取的字符个数
        intlen = -1;
        while((len = isr.read(buffer)) != -1) {
            System.out.print(new String(buffer,0,len));
        }
        // 关闭流
        isr.close();
    }

OutputStreamWriter:转换输出流

OutputStreamWriter extend Writer

构造方法:

  • OutputStreamWriter(OutputStream out)
  • OutputStreamWriter(OutputStream out, Charset cs) //指定对输出数据的编码方式

转换输出流的原理:

FileOutputStream fos = new FileOutputStream("a.txt")

OutputStreamWriter osw = new OutputStreamWriter(fos, "gbk");

1.创建转换输出流对象,以字符的形式内存中的数据

2.通过FileOutputStream将字符对象转为字节对象,并指定其编码方式

public static void main(String[] args) throws IOException{
        // 要保存的字符串
        String content = "我爱Java";
        // 创建字节输出流对象
        FileOutputStream fos = new FileOutputStream("a.txt");
        // 创建转换输出流对象
        OutputStreamWriter osw = new OutputStreamWriter(fos, "gbk");
        // 调用方法写出数据
        osw.write(content);
        // 关闭流释放资源
        osw.close();
    }

序列化流&反序列化流

ObjectOutputStream:序列化流

序列化:将对象转化为流对象的过程

构造方法:

ObjectOutputStream(OutputStream out)

常用方法:

writeObject(Object obj) //将obj对象写到输出流关联的目的文件中

注:要对对象进行序列化,该对象对应的类必须实现Serializable接口,否则会抛异常

ObjectInputStream:反序列化流

反序列化:将流对象转为对象的过程

构造方法:

ObjectInputStream (InputStream in)

常用方法:

Object readObject() 将InputStream 关联的序列化流对象封装成一个Object对象

实现反序列化的原理:

当一个类实现Serializable接口,该类编译成class文件会自动生成一个serialVersionUID(序列化标识符),当对对象进行序列化时,该标识符也会写入到序列化流中,进行反序列化是就是通过这个标识符来找到对应的class文件,进而得到相应的反序列化对象

当一个class文件被修改时,此时又会生成一个新的serialVersionUID(序列化标识符),原来生成的序列化流就不能反序列成相应的对象,因此需要给类固定一个serialVersionUID(序列化标识符),否则会抛出异常

固定一个serialVersionUID(序列化标识符)的做法:

声明一个变量:private static final Long serialVersionUID = 自定义的一个值

注:被static 和transient(瞬态关键字)修饰的变量的值是不能被序列化的

序列化集合

在文件中保存多个对象,对象存储在集合中,对集合序列化和反序列化

步骤:

1.定义存储对象的集合

2.往集合中添加对象

3.创建序列化流

4.writeObject(集合)

打印流

PrintStream extend OutputStream

特点:只负责数据的输出,不负责读取

构造方法:

PrintStream(File file)

PrintStream(OutputStream out)

PrintStream(String fileName)

常用方法:可以打印任何类型的数据

print(可接收任何数据)

println(可接收任何数据)

PrintStream ps = new PrintStream(File f);

ps.print() //默认在控制条输出

使用System.setOut(ps) : 指定输出到 new PrintStream(File f)关联的文件

其他

原文地址:https://www.cnblogs.com/Auge/p/11372920.html