民工心事 发表于 2023-5-23 16:53:16

java IO流

Java IO流

什么是流?

概念:内存和存储设备之间传输数据的通道。
数据借助流传输。
流分类:

[*]按照方向:输入流(将存储设备中的内容读入到内存中)和输出流(将内存中的内容写入到存储设备中)
[*]按照单位:字节流(以字节为单位,可以读写所有数据)和字符流(以字符为单位,只能读取文本数据)
[*]按照功能:节点流(具有实际传输数据的读写功能)和过滤流(在节点流的基础之上增强功能)
字节流

字节流的父类(抽象类):

[*]InputStream:字节输入流:提供一些如read(),close()的方法
[*]OutputStream:字节输出流:提供一些如write(),close()的方法
https://img2023.cnblogs.com/blog/2844064/202305/2844064-20230523163114943-181742637.png
文件字节流


[*]FileInputStream:文件字节输入流,继承字节输入流
[*]FileOutputStream:文件字节输出流,继承字节输入流
https://img2023.cnblogs.com/blog/2844064/202305/2844064-20230523163146079-707682639.png
文件字节输入流操作:
//创建FileInputStream并且指定文件路径
FileInputStream fis = new FileInputStream("d:\\aa.txt");
int data = 0;
//read返回的int,当等于-1表示读完了,如果是字符返回对应的asc码
//一次读取一个字节
while ((data=fis.read())!=-1){
    System.out.println((char) data);
}
/*
a
b
c什么为一次读取一个字节的情况
当需要一次读取多个字节
//创建FileInputStream并且指定文件路径
FileInputStream fis = new FileInputStream("d:\\aa.txt");
//一次读取多个个字节
byte[] buf = new byte;
//本处read返回实际读取个数,读完返回-1,并且将读取到的存储在buf中
//因为数组大小一次读取三个
System.out.println(fis.read(buf));
System.out.println(new String(buf));
fis.close();
/*
3
abc
*/

//或者

FileInputStream fis = new FileInputStream("d:\\aa.txt");
      byte[] buf = new byte;
      int count=0;
      while ((count=fis.read(buf))!=-1){
         System.out.println(new String(buf,0,count));
      }
      fis.close();
/*
abc
def
g本处只是例子,实际操作时缓冲区数组不需要创建得太小,可以是1024以上
文件字节输出流操作:
//创建文件字节输出流,为true代表文件已经存在时追加,false每次都会覆盖(默认)
FileOutputStream fos=new FileOutputStream("d://aa.txt",false);
fos.write(97);
fos.write('b');
fos.write('c');

//文件中:abc

//要写入字符串:转成byte数组
String s = "helloworld";
fos.write(s.getBytes());

//文件中:abchelloworld实现复制操作:
FileInputStream fis = new FileInputStream("d:\\aa.txt");
FileOutputStream fos = new FileOutputStream("d:\\bb.txt");
byte[] buf = new byte;
int count=0;
while ((count=fis.read(buf))!=-1){
    fos.write(buf,0,count);
}
fis.close();
fos.close();字节缓冲流


[*]缓冲流:BufferedInputStream/BufferedOutputStream
[*]继承了过滤流(FilterOutputStream/FilterInputStream),而过滤流又继承了文件字节流
[*]内置了缓存区(8k)
[*]作用:提高IO效率,减少访问磁盘次数;数据存储在缓存区,flush是将缓冲区的内容写入文件中,也可以直接close,后面实现缓冲区的类类似
字节输入缓冲流:
//      创建BufferedInputStream,需要传入一个底层流
      FileInputStream fis = new FileInputStream("d:\\aa.txt");
      BufferedInputStream bis = new BufferedInputStream(fis);
//      读取:类似,但是这个有缓冲区效率更高
      int data = 0;
      while ((data=bis.read())!=-1){
            System.out.print((char)data);
      }
      //当然也可以和文件字节流一样,直接再创建一个缓冲区实现,一模一样但是因为内部缓冲区的存在效率更高
      
//      关闭这一个就行
      bis.close();

//out:abchelloworld字节输出缓冲流:
FileOutputStream fos = new FileOutputStream("d://bb.txt");
BufferedOutputStream bos = new BufferedOutputStream(fos);
//转义符如换行等也可以写入也可以
String s = "hello\r\n";
bos.write(s.getBytes());//这是写入了缓冲区,没有写入文件
bos.flush();//需要刷新才能从缓冲区真正写入文件
bos.close();//当然close时自动flush对象流


[*]对象流:ObjectOutputStream/ObjectInputStream
[*]可以实现写入或者读取对象
[*]增强了缓冲区功能
[*]增强了读写8种基本数据类型和字符串的功能,不只是能读写int和byte
[*]readObject()从流中读取一个对象;writeObject(Object obj)向流中写入一个对象
[*]使用流传输对象的过程被称为序列化(写入)和反序列化(读取)
[*]继承文件字节流,需要基于其才能创建
[*]序列化和反序列化的对象类都必须实现Serializable接口(该接口本身没有任何方法仅仅标志该类可以序列化),而且其中关联的其他类也要实现该接口
[*]可以在类中自定义生成常量private static final long serialVersionUID ,该常量为序列化版本号用来标识该类,即使是同一个类如果UID不同依旧被认为不是同一个类无法反序列化不同UID的类
[*]使用transient修饰实现,表示该属性不需要序列化(例如:private transient int age )
[*]静态属性不能序列化
序列化:
public classStudentimplements Serializable {}



//main:
FileOutputStream fos = new FileOutputStream("d://aa.txt");
      ObjectOutputStream oos = new ObjectOutputStream(fos);
//      序列化
      Student student = new Student();
      oos.writeObject(student);
//      oos.writeBoolean();写入boolean
//         oos.writeBytes("hellow");//写入字符串
//      其他基本类型也有相应的办法写入
      oos.close();反序列化:
FileInputStream fis = new FileInputStream("d://aa.txt");
ObjectInputStream ois = new ObjectInputStream(fis);
//读取存入的第一个对象
Student s = (Student)ois.readObject();
ois.close();序列化、反序列化多个对象时可以使用集合:
//序列化
FileOutputStream fos = new FileOutputStream("d://aa.txt");
      ObjectOutputStream oos = new ObjectOutputStream(fos);
//      序列化
      Student student = new Student();
      Student student1 = new Student();
      ArrayList<Student> arrayList = new ArrayList<>();
      arrayList.add(student);
      arrayList.add(student1);
      oos.writeObject(arrayList);
      oos.close();

//反序列化
FileInputStream fis = new FileInputStream("d://aa.txt");
      ObjectInputStream ois = new ObjectInputStream(fis);
      ArrayList<Student> arrayList = (ArrayList<Student>) ois.readObject();
      ois.close();常见字符编码

https://img2023.cnblogs.com/blog/2844064/202305/2844064-20230523163229216-1358525436.png
字符流

问题的引出:
FileInputStream fis = new FileInputStream("d://aa.txt");
int data = 0;
while ((data=fis.read())!=-1){
    System.out.println((char)data);
}
fis.close();
//上述代码当我们读取的文件中为中文时得到乱码。
//原因:我们是运用字节流一个一个字节读取文件,而本处的文件编码方式是utf-8(中文常见的编码方式),每个中文字占3个字节。当一个一个字节读取时出现乱码这种情况下我们需要字符流
字符流的父类(抽象类):
Reader(字符输入流)和Writer(字符输出流)
只能用于文本文件
https://img2023.cnblogs.com/blog/2844064/202305/2844064-20230523163252901-1764464959.png
文件字符流


[*]FileReader和FileWriter
[*]父类是InputStreamReader和OutputStreamWriter,而这两者的父类又是Reader和Writer
[*]采用默认字符编码
https://img2023.cnblogs.com/blog/2844064/202305/2844064-20230523163320985-95406667.png
FileReader:
FileReader fr = new FileReader("d://aa.txt");
       //单个字符的读取
//      int data = 0;
//      while ((data=fr.read())!=-1){//读取一个字符
//            System.out.println((char)data);
//      }
//创建缓冲区读取
//缓冲区为2字符
      char[] buf = new char;
      int count = 0;
      while ((count = fr.read(buf))!=-1){
            System.out.println(new String(buf,0,count));
      }
      fr.close();
/*out:
你好
呀FileWriter:
FileWriter fw = new FileWriter("d://aa.txt");
fw.write("你好吗");
fw.flush();
fw.close();字符缓冲流


[*]缓冲流:BufferedReader/BufferedWriter
[*]高效读写
[*]支持输入换行符
[*]可以一次写一行、读一行
[*]可以指定缓冲区大小,也可以默认(8K)
[*]新建需要基础字符流(文件字符流)
[*]继承Reader和Writer
读取:
FileReader fr = new FileReader("d://aa.txt");
       BufferedReader br = new BufferedReader(fr);
//      和上面其他类类似的方法不再重写
//      特有的:读一行,末尾返回null
      String s = null;
      //一行一行的读取
      while ((s=br.readLine())!=null){
            System.out.println(s);
      }
      br.close();
//out:你好呀写入:
      FileWriter fw = new FileWriter("d:\\aa.txt");
      BufferedWriter bw = new BufferedWriter(fw);
//      写入
      bw.write("好好学习");
//      写入一个换行windows:\r\n linux:\n
      bw.newLine();
      bw.write(",天天向上");
      bw.close();
/*文件:
好好学习
,天天向上打印流


[*]PrintWriter:

[*]支持print()/println(),支持写入后换行
[*]支持数据原样打印
[*]继承Writer类
[*]类似PrintStream,但是PrintWriter只能打印字符流,而PrintStream可以打印字节流
PrintWriter pw = new PrintWriter("d://aa.txt");
//      打印并且换行
      pw.println(97);
      pw.println(true);
//打印
      pw.print("a");
                pw.print("b");
      pw.close();
/*文件中:
97
true
ab
转换流


[*]也叫桥转换流:InputStreamReader/OutputStreamWriter
[*]可以将字节流转换为字符流
[*]可设置字符的编码方式
[*]继承字符流Reader/Writer
[*]创建时需要基本的文件字符流
InputStreamReader:
FileInputStream fis = new FileInputStream("d://aa.txt");
InputStreamReader isr = new InputStreamReader(fis,"utf-8");
int data=0;
//一个一个字符读取
while ((data=isr.read())!=-1){
    System.out.print((char)data);
}
isr.close();OutputStreamWriter:
FileOutputStream fos = new FileOutputStream("d://aa.txt");
OutputStreamWriter osw = new OutputStreamWriter(fos,"gbk");
for(int i=0;i<10;i++){
    osw.write("我爱中国\r\n");
    osw.flush();
}
osw.close();Prorerties


[*]属性集合
[*]特点:

[*]存储属性名和属性值
[*]属性名和属性值都是字符串类型
[*]没有泛型
[*]继承Hashtable集合,具体看集合那一章笔记
[*]和流有关
//      分隔符
      System.out.println("路径分割符"+File.pathSeparator);
      System.out.println("名字分割符"+File.separator);

      File file = new File("d://cc.txt");
//      判断文件是否存在
      if(!file.exists()){
//            创建文件
            boolean b = file.createNewFile();
            System.out.println("创建结果:"+b);
      }
       if(file.exists()) {
//         直接删除
         System.out.println("删除结果" + file.delete());
         //使用jvm退出时删除
         file.deleteOnExit();
       }
//       获取文件信息
      System.out.println("绝对路径"+file.getAbsolutePath());
       //看你创建时传入的路径
      System.out.println("获取路径"+file.getPath());
//      获取名称
      System.out.println("获取名称"+file.getName());
//      获取父目录
      System.out.println("获取父目录"+file.getParent());
//      获取文件长度
      System.out.println("获取文件长度"+file.length());
//      文件创建时间
      System.out.println("文件创建时间"+new Date(file.lastModified()).toString());

//      判断文件是否可写
      System.out.println("是否可写"+file.canWrite());
//      是否是文件
      System.out.println("是否是文件"+file.isFile());
//      是否隐藏
      System.out.println("是否隐藏"+file.isHidden());

/*
路径分割符;
名字分割符\
创建结果:true
删除结果true
绝对路径d:\cc.txt
获取路径d:\cc.txt
获取名称cc.txt
获取父目录d:\
获取文件长度0
文件创建时间Thu Jan 01 08:00:00 CST 1970
是否可写false
是否是文件false
是否隐藏falseFile dir = new File("d://cc/aa");
//      判断文件是否存在
      if(!dir.exists()){
//            创建文件夹:只能创建单集目录
//            boolean b = dir.mkdir();
//            创建多级目录
            boolean b = dir.mkdirs();
            System.out.println("创建结果:"+b);
      }
       if(dir.exists()) {
//         直接删除:最下层且必须为空目录
         System.out.println("删除结果" + dir.delete());
         //使用jvm退出时删除,休眠结束后删除
         dir.deleteOnExit();
         Thread.sleep(5000);
       }
//       获取文件信息
      System.out.println("绝对路径"+dir.getAbsolutePath());
       //看你创建时传入的路径
      System.out.println("获取路径"+dir.getPath());
//      获取名称
      System.out.println("获取名称"+dir.getName());
//      获取父目录
      System.out.println("获取父目录"+dir.getParent());
//      文件创建时间
      System.out.println("文件创建时间"+new Date(dir.lastModified()).toString());


//      判断是否是文件夹
      System.out.println("是否可写"+dir.isDirectory());
//      是否隐藏
      System.out.println("是否隐藏"+dir.isHidden());
//      遍历文件夹
      String[] file = dir.list();
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
页: [1]
查看完整版本: java IO流