1.JDK提供的InputStream分为两类:

  • 直接提供数据的InputStream

    * FileInputStream:从文件读取

    * ServletInputStream:从HTTP请求读取数据

    * Socket.getInputStream():从TCP连接读取数据
  • 提供额外附加功能的FilterInputStream

    * 如果要给FileInputStream添加缓冲功能:

    • BufferedFileInputStream extends FileInputStream
*    如果要给FileInputStream添加计算机签名的功能:
* DigestFileInputStream extends FileInputStream
* 如果要给FileInputStream添加加密/解密功能:
* CipherFileInputStream extends FileInputStream
  • 组合功能而非继承的设计模式称为Filter模式(或者Decorator模式)
  • 通过少量的类实现了各种功能的组合
//演示代码
InputStream input = new GZIPInputStream(//直接读取解压缩包的内容
new BufferedInputStream(//提供缓冲的功能
new FileInputStream("test.gz")));

廖雪峰示例中的CountInputStream有错误,当读取完毕后,返回-1,而count再读取完毕后,会减1,导致结果不一样。运行结果如下。只需要将count的初始值设置为1即可。
```#java
public class CountInputStream extends FilterInputStream {
int count=0;
public CountInputStream(InputStream in) {
super(in);
}
//重写read方法,使用count计数
public int read(byte[] b,int off,int len) throws IOException {
int n = super.read(b,off,len);
count += n;
System.out.println(count);
return n;//最后返回-1,count-1
}
}

```#java
public class Main {
static void printCount1(List<Integer> list) throws IOException{
try(InputStream input = new GZIPInputStream(
new BufferedInputStream(
new FileInputStream("./src/main/java/com/testList/test.gz")))){
byte[] buffer = new byte[1024];//创建竹筒
int count=0;
int n ;
while((n=input.read(buffer))!=-1){
count += n;
list.add(n);
}
System.out.println("直接读取的和:"+count);
System.out.println("直接读取得到的集合的和"+getSum(list));
}
}
static void printCount2(List<Integer> list) throws IOException{
try(CountInputStream input = new CountInputStream(
new GZIPInputStream(
new BufferedInputStream(
new FileInputStream("./src/main/java/com/testList/test.gz"))))){
byte[] buffer = new byte[1024];
int n;
while((n=input.read(buffer))!=-1){
list.add(n);
// System.out.println();
}
System.out.println("通过CountInputStream获取的和:"+input.count);
System.out.println("通过CountInputStream获取的集合的和:"+getSum(list));
}
}
static Integer getSum(List<Integer> list){
int sum = 0;
for(int i=0;i<list.size();i++){
sum += list.get(i);
}
return sum;
}
public static void main(String[] args) throws IOException {
List<Integer> list1 = new LinkedList<>();
List<Integer> list2 = new LinkedList<>();
printCount2(list2);
printCount1(list1);
//从结果
System.out.println(list1);
}
}

2.总结

  • Java IO使用Filter模式为InputStream/OutputStream增加功能
  • 可以把一个InputStream和任意FilterInputStream组合
  • 可以把一个OutputStream和任意FilterOutputStream组合
  • Filter模式可以在运行期动态增加功能(又成Decorator模式)

廖雪峰Java6 IO编程-2input和output-4Filter模式的更多相关文章

  1. 廖雪峰Java6 IO编程-2input和output-7序列化

    1.序列化 序列化是指把一个Java对象变成二进制内容byte[] 序列化后可以把byte[]保存到文件中 序列化后可以把byte[]通过网络传输 一个Java对象要能序列化,必须实现Serializ ...

  2. 廖雪峰Java6 IO编程-2input和output-5操作zip

    1.ZipInputStream是一种FilterInputStream 可以直接读取zip的内容 InputStream->FilterInputStream->InflateInput ...

  3. 廖雪峰Java6 IO编程-2input和output-6classpath资源

    1.从classpath读取文件可以避免不同环境下文件路径不一致的问题. Windows和Linux关于路径的表示不一致 Windows:C:\conf\default.properties Linu ...

  4. 廖雪峰Java6 IO编程-3Reader和Writer-2Writer

    1.java.io.Writer和java.io.OutputStream的区别 OutputStream Writer 字节流,以byte为单位 字符流,以char为单位 写入字节(0-255):v ...

  5. 廖雪峰Java6 IO编程-3Reader和Writer-1Reader

    1.java.io.Reader和java.io.InputStream的区别 InputStream Reader 字节流,以byte为单位 字符流,以char为单位 读取字节(-1,0-255): ...

  6. 廖雪峰Java13网络编程-1Socket编程-2TCP编程

    1. Socket 在开发网络应用程序的时候,会遇到Socket这个概念. Socket是一个抽象概念,一个应用程序通过一个Socket来建立一个远程连接,而Socket内部通过TCP/IP协议把数据 ...

  7. 廖雪峰Java13网络编程-3其他-1HTTP编程

    1.HTTP协议: Hyper Text Transfer Protocol:超文本传输协议 基于TCP协议之上的请求/响应协议 目前使用最广泛的高级协议 * 使用浏览器浏览网页和服务器交互使用的就是 ...

  8. 廖雪峰Java13网络编程-1Socket编程-5UDP编程

    1. UDP编程: 不需要建立连接 可以直接发送和接收数据 1.1 客户端 DatagramSocket sock = new DatagramSocket(){} sock.connect(addr ...

  9. 廖雪峰Java13网络编程-1Socket编程-3TCP多线程编程

    TCP多线程编程 一个ServerSocket可以和多个客户端同时建立连接,所以一个Server可以同时与多个客户端建立好的Socket进行双向通信. 因此服务器端,当我们打开一个Socket以后,通 ...

随机推荐

  1. java中事件驱动

    在java语言中,事件不是由事件源自己来处理的,而是交给事件监听者来处理,要将事件源(如按钮)和对事件的具体处理分离开来.这就是所谓的事件委托处理模型. 事件委托处理模型由产生事件的事件源.封装事件相 ...

  2. 《DSP using MATLAB》Problem 6.14

    代码: %% ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ %% Output In ...

  3. SolrCore Initialization Failures - Max direct memory is likely too low

    org.apache.solr.common.SolrException:org.apache.solr.common.SolrException: The max direct memory is ...

  4. linux 条件

    1.文件状态测试-d 目录 -r 可读-f 常规文件 -w 可写-L 符号连接 -x 可执行-s 文件长度大于0,非空 -u 文件有suid位设置 示例: [ -s haison.c ] 0表示成功, ...

  5. nakadi-ui nakadi event broker 的可视化UI工具

    nakadi 是一款很不错的基于fafka 开发的event broker ,我们只需要使用http 请求就可以调用kafka 方便的发布订阅功能 环境准备 docker-compose 文件 ver ...

  6. Modern Data Lake with Minio : Part 2

    转自: https://blog.minio.io/modern-data-lake-with-minio-part-2-f24fb5f82424 In the first part of this ...

  7. C# to IL 18 Glossary(术语)

  8. 我发起了一个 用 C 语言 作为 中间语言 的 编译器 项目 VMBC

    大家好 ,  我发起了一个 用 C 语言 作为 中间语言 的 编译器 项目 VMBC . VMBC ,  全称是 Virtual Machine Base on C  . 有一种说法 ,  C 语言是 ...

  9. windows python监听文件触发脚本

    from watchdog.events import * class FileEventHandler(FileSystemEventHandler): def __init__(self): Fi ...

  10. Can't connect to local MySQL server through socket '/var/lib/mysql/mysql.sock' 解决办法

    启动mysql 报错: ERROR 2002 (HY000): Can’t connect to local MySQL server through socket ‘/var/lib/mysql/m ...