1.1,什么是io流?

  流是一组有顺序的,有起点和终点的字节集合,是对数据传输的总称或抽象。即数据在两设备间的传输称为流,流的本质是数据传输,根据数据传输特性将流抽象为各种类,方便更直观的进行数据操作。

1.2,流按类型分为两种:

 * 字节流 : 字节流可以操作任何数据,因为在计算机中任何数据都是以字节的形式存储的
 * 字符流 : 字符流只能操作纯字符数据,比较方便。

1.2.1,io流的结构图:

  

1.3,IO流常用父类:

  * 字节流的抽象父类:            * 字符流的抽象父类:
    * InputStream               * Reader 
    * OutputStream               * Writer  

1.4,IO程序书写:

* 使用前,导入IO包中的类
* 使用时,进行IO异常处理
* 使用后,释放资源

1.5,IO流(FileInputStream)~(1.5~2.0讲的是字节流):

* read()一次读取一个字节

* read()一次读取一个字节

         FileInputStream fis = new FileInputStream("aaa.txt");    //创建一个文件输入流对象,并关联aaa.txt
int b; //定义变量,记录每次读到的字节
while((b = fis.read()) != -) { //将每次读到的字节赋值给b并判断是否是-1
System.out.println(b); //打印每一个字节
} fis.close(); //关闭流释放资源

read()方法读取的是一个字节,为什么返回是int,而不是byte?
  因为字节输入流可以操作任意类型的文件,比如图片音频等,这些文件底层都是以二进制形式的存储的,如果每次读取都返回byte,有可能在读到中间的时候遇到111111111
  那么这11111111是byte类型的-1,我们的程序是遇到-1就会停止不读了,后面的数据就读不到了,所以在读取的时候用int类型接收,如果11111111会在其前面补上
  24个0凑足4个字节,那么byte类型的-1就变成int类型的255了这样可以保证整个数据读完,而结束标记的-1就是int类型

1.6,IO流(FileOutputStream):

  * write()一次写出一个字节

         FileOutputStream fos = new FileOutputStream("bbb.txt");    //如果没有bbb.txt,会创建出一个
//fos.write(97); //虽然写出的是一个int数,但是在写出的时候会将前面的24个0去掉,所以写出的一个byte
fos.write();
fos.write();
fos.close();

1.7,第一种方法:IO之字节流流的基本读写(FileInputStream读取,FileOutputStream写出):

         //创建输入流对象,C:\\Users\\十年饮冰,难凉热血\\Desktop\\日常一趣\\1970年的来历.rtf
FileInputStream fis = new FileInputStream("C:\\Users\\十年饮冰,难凉热血\\Desktop\\日常一趣\\1970年的来历.rtf");
//创建输出流对象,关联copt.txt
FileOutputStream fos = new FileOutputStream("C:\\Users\\十年饮冰,难凉热血\\Desktop\\日常一趣\\copt.rtf");
int b;
while((b = fis.read())!=-1){ //在不断的读取每一个字节
fos.write(b); //将每一个字节写出
}
fis.close(); //关闭释放
fos.close();
}

第二种读写方法:

         //创建输入流对象
FileInputStream fis = new FileInputStream("C:\\Users\\十年饮冰,难凉热血\\Desktop\\日常一趣\\由浅入深学Java—基础、进阶与必做260题.pdf");
//创建输出流对象
FileOutputStream fos = new FileOutputStream("C:\\Users\\十年饮冰,难凉热血\\Desktop\\日常一趣\\Jy.pdf");
byte[] arr = new byte[1024*8];
int len;
while((len =fis.read(arr))!=-1){
fos.write(arr,0,len);  
}

1.8,缓冲流思想:

  缓冲思想:

    * 字节流一次读写一个数组的速度明显比一次读写一个字节的速度快很多,这是加入了数组这样的缓冲区效果,java本身在设计的时候, 也考虑到了这样的设计思想(装饰设计模式后面讲解),所以提供了字节缓冲区流。
    * BufferedInputStream
      BufferedInputStream内置了一个缓冲区(数组),从BufferedInputStream中读取一个字节时,* BufferedInputStream会一次性从文件中读取8192个, 存在缓冲区中, 返回给程序一个,程序再次读取时, 就不用找文件了, 直接从缓冲区中获取,直到缓冲区中所有的都被使用过, 才重新从文件中读取8192个。
    * BufferedOutputStream
      BufferedOutputStream也内置了一个缓冲区(数组),程序向流中写出字节时, 不会直接写到文件, 先写到缓冲区中,直到缓冲区写满, BufferedOutputStream才会把缓冲区中的数据一次性写到文件里。

        FileInputStream fis = new FileInputStream("致青春.mp3");            //创建文件输入流对象,关联致青春.mp3
BufferedInputStream bis = new BufferedInputStream(fis); //创建缓冲区对fis装饰
FileOutputStream fos = new FileOutputStream("copy.mp3"); //创建输出流对象,关联copy.mp3
BufferedOutputStream bos = new BufferedOutputStream(fos); //创建缓冲区对fos装饰 int b;
while((b = bis.read()) != -1) {
bos.write(b);
} bis.close(); //只关装饰后的对象即可
bos.close();

1.9,IO流(flush和close方法的区别):

  * flush()方法                * close()方法
   * 用来刷新缓冲区的,刷新后可以再次写出        * 用来关闭流释放资源的的,如果是带缓冲区的流对象的close()方法,不但会关闭流,还会再关闭流之前刷新缓冲区,关闭后不能再写出

2.0,IO流(字节流读写中文)

  * 字节流读取中文的问题
    * 字节流在读中文的时候有可能会读到半个中文,造成乱码
  * 字节流写出中文的问题
    * 字节流直接操作的字节,所以写出中文必须将字符串转换成字节数组,写出回车换行 write("\r\n".getBytes());

       //字节流读取中文会造成乱码的解决方法
FileInputStream fis = new FileInputStream("xxx.txt");
byte[] arr = new byte[5];
int len;
while((len=fis.read(arr))!=-1){
System.out.println(new String(arr,0,len));
}
fis.close();
}
         //字节流写出中文的问题
FileOutputStream fos = new FileOutputStream("zzz.txt");
fos.write("我读书少你,不要骗我".getBytes());
fos.write("\r\n".getBytes());
fos.close();

2.1,字符流是什么(2.1~3.1讲的是字符流知识):

  * 字符流是可以直接读写字符的IO流
  * 字符流读取字符, 就要先读取到字节数据, 然后转为字符. 如果要写出字符, 需要把字符转为字节再写出。

2.2,FileReader类的read()方法可以按照字符大小读取

         FileReader fr = new FileReader("aaa.txt");                //创建输入流对象,关联aaa.txt
int ch;
while((ch = fr.read()) != -1) { //将读到的字符赋值给ch
System.out.println((char)ch); //将读到的字符强转后打印
}
fr.close(); //关流

2.3,FileWriter类的write()方法可以自动把字符转为字节写出:

      public static void main(String[] args){
FileWriter fw = new FileWriter("ccc.txt",true);  //true是对数据进行追加的
fw.write(97);
fw.close();
}

2.4,字符流的拷贝:

         //创建字符输入流
FileReader fr = new FileReader("ccc.txt");
//创建字符输出流
FileWriter fw = new FileWriter("xxx.txt");
int len;
//while中的条件是fr.read()的字符长度等于len假如fr.read的值等于-1的时候就停止跳转
while((len=fr.read())!=-1){
fw.write(len);
}
fr.close();
fw.close();//writer类中有一个2k的小缓冲区,如果不关流,就会将内容写到缓冲区里,关流回将缓冲区内容刷新,再关闭
}

2.5,(什么情况下使用字符流):

  * 字符流也可以拷贝文本文件, 但不推荐使用. 因为读取时会把字节转为字符, 写出时还要把字符转回字节.
  * 程序需要读取一段文本, 或者需要写出一段文本的时候可以使用字符流
  * 读取的时候是按照字符的大小读取的,不会出现半个中文
  * 写出的时候可以直接将字符串写出,不用转换为字节数组

2.6,IO流(字符流是否可以拷贝非纯文本的文件):

  * 不可以拷贝非纯文本的文件
  * 因为在读的时候会将字节转换为字符,在转换过程中,可能找不到对应的字符,就会用?代替,写出的时候会将字符转换成字节写出去
  * 如果是?,直接写出,这样写出之后的文件就乱了,看不了

2.7,IO流(自定义字符数组的拷贝):

          FileReader fr = new FileReader("aaa.txt");            //创建字符输入流,关联aaa.txt
FileWriter fw = new FileWriter("bbb.txt"); //创建字符输出流,关联bbb.txt
int len;
char[] arr = new char[1024*8]; //创建字符数组
while((len = fr.read(arr)) != -1) { //将数据读到字符数组中
fw.write(arr, 0, len); //从字符数组将数据写到文件上
} fr.close(); //关流释放资源
fw.close();

2.8,IO流(带缓冲的字符流) :

* BufferedReader的read()方法读取字符时会一次读取若干字符到缓冲区, 然后逐个返回给程序, 降低读取文件的次数, 提高效率
* BufferedWriter的write()方法写出字符时会先写到缓冲区, 缓冲区写满时才会写到文件, 降低写文件的次数, 提高效率

         BufferedReader br = new BufferedReader(new FileReader("aaa.txt"));    //创建字符输入流对象,关联aaa.txt
BufferedWriter bw = new BufferedWriter(new FileWriter("bbb.txt")); //创建字符输出流对象,关联bbb.txt int ch;
while((ch = br.read()) != -1) { //read一次,会先将缓冲区读满,从缓冲去中一个一个的返给临时变量ch
bw.write(ch); //write一次,是将数据装到字符数组,装满后再一起写出去
} br.close(); //关流
bw.close();

2.9,IO流(readLine()和newLine()方法):

  * BufferedReader的readLine()方法可以读取一行字符(不包含换行符号)
  * BufferedWriter的newLine()可以输出一个跨平台的换行符号"\r\n"

         BufferedReader br = new BufferedReader(new FileReader("aaa.txt"));
BufferedWriter bw = new BufferedWriter(new FileWriter("bbb.txt"));
String line;
while((line = br.readLine()) != null) {
bw.write(line);
//bw.write("\r\n"); //只支持windows系统
bw.newLine(); //跨平台的
} br.close();
bw.close();

3.0,IO流(LineNumberReader):

  * LineNumberReader是BufferedReader的子类, 具有相同的功能, 并且可以统计行号
  * 调用getLineNumber()方法可以获取当前行号
  * 调用setLineNumber()方法可以设置当前行号

         LineNumberReader lnr = new LineNumberReader(new FileReader("aaa.txt"));
String line;
lnr.setLineNumber(100); //设置行号
while((line = lnr.readLine()) != null) {
System.out.println(lnr.getLineNumber() + ":" + line);//获取行号
} lnr.close();

3.1,IO流(使用指定的码表读写字符)(掌握):

  * FileReader是使用默认码表读取文件, 如果需要使用指定码表读取, 那么可以使用InputStreamReader(字节流,编码表)
  * FileWriter是使用默认码表写出文件, 如果需要使用指定码表写出, 那么可以使用OutputStreamWriter(字节流,编码表)

         BufferedReader br =                                     //高效的用指定的编码表读
new BufferedReader(new InputStreamReader(new FileInputStream("UTF-8.txt"), "UTF-8"));
BufferedWriter bw = //高效的用指定的编码表写
new BufferedWriter(new OutputStreamWriter(new FileOutputStream("GBK.txt"), "GBK"));
int ch;
while((ch = br.read()) != -1) {
bw.write(ch);
} br.close();
bw.close();

除却字节流与字符流本博客还将总结一些课外的流,开发中也会遇见,为了丰富知识的同学们可以看看。

3.2,什么是序列流(了解):

  * 序列流可以把多个字节输入流整合成一个, 从序列流中读取数据时, 将从被整合的第一个流开始读, 读完一个之后继续读第二个, 以此类推。

         //整合多个字节输入流: SequenceInputStream(Enumeration)
FileInputStream fis1 = new FileInputStream("a.txt"); //创建输入流对象,关联a.txt
FileInputStream fis2 = new FileInputStream("b.txt"); //创建输入流对象,关联b.txt
FileInputStream fis3 = new FileInputStream("c.txt"); //创建输入流对象,关联c.txt
Vector<InputStream> v = new Vector<>(); //创建vector集合对象
v.add(fis1); //将流对象添加
v.add(fis2);
v.add(fis3);
Enumeration<InputStream> en = v.elements(); //获取枚举引用
SequenceInputStream sis = new SequenceInputStream(en); //传递给SequenceInputStream构造
FileOutputStream fos = new FileOutputStream("d.txt");
int b;
while((b = sis.read()) != -1) {
fos.write(b);
} sis.close();
fos.close();

3.3,什么是内存输出流(了解):

  * 该输出流可以向内存中写数据, 把内存当作一个缓冲区, 写出之后可以一次性获取出所有数据

            FileInputStream fis = new FileInputStream("a.txt");
ByteArrayOutputStream baos = new ByteArrayOutputStream();
int b;
while((b = fis.read()) != -1) {
baos.write(b);
} //byte[] newArr = baos.toByteArray(); //将内存缓冲区中所有的字节存储在newArr中
//System.out.println(new String(newArr));
System.out.println(baos);
fis.close();

3.4,什么是对象操作流(了解):

  * 该流可以将一个对象写出, 或者读取一个对象到程序中. 也就是执行了序列化和反序列化的操作.

3.4.1,使用方式:

  * 写出: new ObjectOutputStream(OutputStream), writeObject()

                 public static void main(String[] args) throws IOException {
Person p1 = new Person("张三", 23);
Person p2 = new Person("李四", 24);
// FileOutputStream fos = new FileOutputStream("e.txt");
// fos.write(p1);
// FileWriter fw = new FileWriter("e.txt");
// fw.write(p1);
//无论是字节输出流,还是字符输出流都不能直接写出对象
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("e.txt"));//创建对象输出流
oos.writeObject(p1);
oos.writeObject(p2);
oos.close();
}

3.4.2,使用方式:

    * 读取: new ObjectInputStream(InputStream), readObject()

                //读取对象,反序列化
public static void main(String[] args) throws IOException, ClassNotFoundException {
ObjectInputStream ois = new ObjectInputStream(new FileInputStream("e.txt"));
Person p1 = (Person) ois.readObject();
Person p2 = (Person) ois.readObject();
System.out.println(p1);
System.out.println(p2);
ois.close();
}

3.4.3,IO流(对象操作流优化)(写入与读取写一起了)(了解):

  * 将对象存储在集合中写出

     Person p1 = new Person("张三", 23);
Person p2 = new Person("李四", 24);
Person p3 = new Person("马哥", 18);
Person p4 = new Person("辉哥", 20); ArrayList<Person> list = new ArrayList<>();
list.add(p1);
list.add(p2);
list.add(p3);
list.add(p4); ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("f.txt"));
oos.writeObject(list); //写出集合对象
oos.close();
    //读取到的是一个集合对象
ObjectInputStream ois = new ObjectInputStream(new FileInputStream("f.txt"));
ArrayList<Person> list = (ArrayList<Person>)ois.readObject(); //泛型在运行期会被擦除,索引运行期相当于没有泛型
//想去掉黄色可以加注解 @SuppressWarnings("unchecked")
for (Person person : list) {
System.out.println(person);
}
ois.close();

3.5,什么是打印流(了解):

  * 该流可以很方便的将对象的toString()结果输出, 并且自动加上换行, 而且可以使用自动刷出的模式
  * System.out就是一个PrintStream, 其默认向控制台输出信息

           PrintStream ps = System.out;
ps.println(97); //其实底层用的是Integer.toString(x),将x转换为数字字符串打印
ps.println("xxx");
ps.println(new Person("张三", 23));
Person p = null;
ps.println(p); //如果是null,就返回null,如果不是null,就调用对象的toString()

3.6,什么是标准输入输出流(掌握):

  * System.in是InputStream, 标准输入流, 默认可以从键盘输入读取字节数据
  * System.out是PrintStream, 标准输出流, 默认可以向Console中输出字符和字节数据

3.6.1,修改标准输入输出流(了解):

 修改输入流: System.setIn(InputStream)    * 修改输出流: System.setOut(PrintStream)

             System.setIn(new FileInputStream("a.txt"));                //修改标准输入流
System.setOut(new PrintStream("b.txt")); //修改标准输出流 InputStream in = System.in; //获取标准输入流
PrintStream ps = System.out; //获取标准输出流
int b;
while((b = in.read()) != -1) { //从a.txt上读取数据
ps.write(b); //将数据写到b.txt上
} in.close();
ps.close();

3.6.2,(修改标准输入输出流拷贝图片)(了解):

         System.setIn(new FileInputStream("IO图片.png"));        //改变标准输入流
System.setOut(new PrintStream("copy.png")); //改变标准输出流 InputStream is = System.in; //获取标准输入流
PrintStream ps = System.out; //获取标准输出流 int len;
byte[] arr = new byte[1024 * 8]; while((len = is.read(arr)) != -1) {
ps.write(arr, 0, len);
} is.close();
ps.close();

3.7,(数据输入输出流)(读写一起写了)(了解):

  * 1.什么是数据输入输出流
    * DataInputStream, DataOutputStream可以按照基本数据类型大小读写数据
    * 例如按Long大小写出一个数字, 写出时该数据占8字节. 读取的时候也可以按照Long类型读取, 一次读取8个字节.

     * DataOutputStream(OutputStream), writeInt(), writeLong() 

             DataOutputStream dos = new DataOutputStream(new FileOutputStream("b.txt"));
dos.writeInt(997);
dos.writeInt(998);
dos.writeInt(999); dos.close();
* DataInputStream(InputStream), readInt(), readLong() DataInputStream dis = new DataInputStream(new FileInputStream("b.txt"));
int x = dis.readInt();
int y = dis.readInt();
int z = dis.readInt();
System.out.println(x);
System.out.println(y);
System.out.println(z);
dis.close();

3.8,(Properties的概述和作为Map集合的使用)(了解)

  * A:Properties的概述                  * A:Properties的特殊功能
  * Properties 类表示了一个持久的属性集。          * public Object setProperty(String key,String value)
  * Properties 可保存在流中或从流中加载。          * public String getProperty(String key)
  * 属性列表中每个键及其对应值都是一个字符串。       * public Enumeration<String> stringPropertyNames() 

         Properties prop = new Properties();
prop.load(new FileInputStream("config.properties")); //将文件读取到集合中
prop.setProperty("tel", "18972345678");
prop.store(new FileOutputStream("config.properties"), null); //第二个参数是对列表参数的描述,可以给值,也可以给null
System.out.println(prop);

java高级之Io流的更多相关文章

  1. java中的IO流

    Java中的IO流 在之前的时候我已经接触过C#中的IO流,也就是说集中数据固化的方式之一,那么我们今天来说一下java中的IO流. 首先,我们学习IO流就是要对文件或目录进行一系列的操作,那么怎样操 ...

  2. 第55节:Java当中的IO流-时间api(下)-上

    Java当中的IO流(下)-上 日期和时间 日期类:java.util.Date 系统时间: long time = System.currentTimeMillis(); public class ...

  3. 第54节:Java当中的IO流(中)

    Java当中的IO流(中) 删除目录 // 简书作者:达叔小生 import java.io.File; public class Demo{ public static void main(Stri ...

  4. 第53节:Java当中的IO流(上)

    Java当中的IO流 在Java中,字符串string可以用来操作文本数据内容,字符串缓冲区是什么呢?其实就是个容器,也是用来存储很多的数据类型的字符串,基本数据类型包装类的出现可以用来解决字符串和基 ...

  5. Java中的IO流总结

    Java中的IO流总结 1. 流的继承关系,以及字节流和字符流. 2. 节点流FileOutputStream和FileInputStream和处理流BufferedInputStream和Buffe ...

  6. Java中的IO流大体介绍

    由于Java中的IO流是在是知识点繁多,所以我大约花了1周的时间将其整理起来.但是整理起来后并不是将完事了,我还是要分字节流和字符流来讲述.然后字节流和字符流中还有是否带有缓冲流. 讲述完IO流后我将 ...

  7. Java中的IO流,Input和Output的用法,字节流和字符流的区别

    Java中的IO流:就是内存与设备之间的输入和输出操作就成为IO操作,也就是IO流.内存中的数据持久化到设备上-------->输出(Output).把 硬盘上的数据读取到内存中,这种操作 成为 ...

  8. Java中的IO流(五)

    上一篇<Java中的IO流(四)>记录了一下Properties类,此类不属于IO流,它属于集合框架.接下来说一下IO流中的其它流 一,打印流PrintStream PrintStream ...

  9. Java中的IO流(六)

    上一篇<Java中的IO流(五)>把流中的打印流PrintStream,PrintWriter,序列流SequenceInputStream以及结合之前所记录的知识点完成了文件的切割与文件 ...

随机推荐

  1. MySQL 高级 视图 事物 触发器 函数 索引优化

    视图 1.什么是视图 ​ 视图就是通过查询得到一张虚拟表,然后保存下来,下次直接使用即可 2.为什么要用视图 ​ 如果要频繁使用一张虚拟表,可以不用重复查询 3.如何用视图 create view t ...

  2. BLE各版本新功能总结

    文章转载自:http://www.sunyouqun.com/2017/04/ 协议发布时间 协议版本 2016/12 Bluetooth 5 2014/12 Bluetooth 4.2 2013/1 ...

  3. 7.控制计划任务crontab命令

    at 命令是针对仅运行一次的任务,循环运行的例行性计划任务,linux系统则是由 cron (crond) 这个系统服务来控制的Linux 系统上面原本就有非常多的计划 性工作,因此这个系统服务是默认 ...

  4. Mysql设置主库binlog文件自动清理

    mysql主库中设置了打开binlog模式后,会在datadir目录下生成大量的日志文件,mysql默认是不会自动清理的,我们来设置下mysql自动清理binlog文件 一.打开mysql [root ...

  5. Linux——grep binary file

    原创声明:本文系博主原创文章,转载或引用请注明出处. grep命令是linux下常用的文本查找命令.当grep检索的文件是二进制文件时,grep命令会提示: $grep pattern filenam ...

  6. 第一天Beta冲刺

    这个作业属于哪个课程 <课程的链接> 这个作业要求在哪里 <作业要求的链接> 团队名称 <做个一亿的小项目> 这个作业的目标 完成第一天Beta冲刺 作业正文 .. ...

  7. AutoFac控制反转

    一.AutoFac介绍 Autofac是.NET里IOC(Inversion of Control,控制反转)容器的一种,同类的框架还有Spring.NET,Unity,Castle等.可以通过NuG ...

  8. 原来你是这样的 jsonp(原理与具体实现细节)

    前言 原文地址 仓库地址 jsonp(JSON with padding)你一定不会陌生,前端向后端拿数据的方式之一,也是处理跨域请求的得利助手. 我们早已习惯,早已熟练了jQ或者zepto的ajax ...

  9. buuctf@ciscn_2019_n_1

    from pwn import * #io=process('./ciscn_2019_n_1') io=remote('node3.buuoj.cn',28216) io.sendline(0x38 ...

  10. nodejs+mysql 批量更新

    没办法,只能通过循环一次次更新: var updateMysql = function(){ data = excel[0].data; for(var i=1; i<data.length; ...