资讯专栏INFORMATION COLUMN

Java IO学习五

gaomysion / 1388人阅读

摘要:不建议对这两个对象尝试使用单个线程,因为这样可能会造成该线程死锁。构造函数创建尚未连接到管道输入流的管道输出流。通常,数据由某个线程从对象读取,并由其他线程将其写入到相应的。

管道流

管道流的主要作用是可以进行两个线程间的通讯,分为管道输出流(PipedOutputStream)、管道输入流(PipedInputStream),如果想要进行管道输出,则必须要把输出流连在输入流之上,在PipedOutputStream类上有如下的一个方法用于连接管道:
public void connect(PipedInputStream snk)throws IOException

通常是创建两个多带带的线程来实现通信,如果是单个线程的话容易出现线程堵塞,因为输出流最多只能向缓冲区写入1024个字节的数据,如果超出就会出现线程堵塞,因此必须创建多个线程实现缓冲区的释放和存储

PipedOutputStream

管道输出流是管道的发送端,可以将管道输出流连接到管道输入流来创建一个通信管道,通常,数据由某个线程写入 PipedOutputStream 对象,并由其他线程从连接的 PipedInputStream 读取。不建议对这两个对象尝试使用单个线程,因为这样可能会造成该线程死锁。如果某个线程正从连接的管道输入流中读取数据字节,但该线程不再处于活动状态,则该管道被视为处于 毁坏 状态。

构造函数

PipedOutputStream() 创建尚未连接到管道输入流的管道输出流。

PipedOutputStream(PipedInputStream snk) 创建连接到指定管道输入流的管道输出流。

常用函数

close() 关闭

void connect(PipedInputStream snk) 将此管道输出流连接到接收者。

void flush() 刷新此输出流并强制写出所有缓冲的输出字节。

void write(byte[] b, int off, int len)len 字节从初始偏移量为 off 的指定 byte 数组写入该管道输出流。

void write(int b) 将指定 byte 写入传送的输出流。

PipedInputStream

管道输入流应该连接到管道输出流;管道输入流提供要写入管道输出流的所有数据字节。通常,数据由某个线程从 PipedInputStream 对象读取,并由其他线程将其写入到相应的 PipedOutputStream。不建议对这两个对象尝试使用单个线程,因为这样可能死锁线程。管道输入流包含一个缓冲区,可在缓冲区限定的范围内将读操作和写操作分离开。 如果向连接管道输出流提供数据字节的线程不再存在,则认为该管道已损坏。

构造函数

PipedInputStream() 创建尚未连接的 PipedInputStream

PipedInputStream(PipedOutputStream src) 创建 PipedInputStream,使其连接到管道输出流 src

常用函数

int available() 返回可以不受阻塞地从此输入流中读取的字节数。

void close() 关闭此管道输入流并释放与该流相关的所有系统资源。

void connect(PipedOutputStream src) 使此管道输入流连接到管道输出流 src

int read() 读取此管道输入流中的下一个数据字节。

int read(byte[] b, int off, int len) 将最多 len 个数据字节从此管道输入流读入 byte 数组。

protected void receive(int b) 接收数据字节。

实例
package IO;

import java.io.IOException;
import java.io.PipedInputStream;
import java.io.PipedOutputStream;

/**
 * Created by chenjiabing on 17-5-25.
 */

/**
 * 注意的问题:
 * 1.写线程正在往缓冲区写数据的时候,但是此时的读线程的管道结束,那么此时的写线程的管道就会发生IOException异常
 * 2.读线程正在从缓冲区读数据的时候,但是此时的写线程的管道已经结束了,此时就会引起读线程的管道发生IOException异常
 * 3.必须是启用多线程才能实现管道之间的读写,否则会出现堵塞现象,因为这里的PipeOutputStream每次向缓冲区写入的字节数最大是1024,如果不及时的减少缓冲区的数据量就会出现堵塞
 */

public class demo7 {
    public static PipedOutputStream outputStream = new PipedOutputStream();
    public static PipedInputStream inputStream = new PipedInputStream();

    /**
     * 创建一个写入数据进程,使用的是PipeOutStream,将数据写入管道中
     */
    public static void send() {
        new Thread(new Runnable() {
            @Override
            public void run() {
                byte[] bytes = new byte[2000];     //创建一个2000字节的数组
                while (true) {
                    try {
                        outputStream.write(bytes, 0, 2000);  //写入管道,但是这里的缓冲区最多写入1024个字节的数据,因此这个是一次没有写完
                        System.out.println("写入成功");
                    } catch (IOException e) {
                        System.out.println("写入失败");
                        System.exit(1);
                    }
                }
            }
        }).start();
    }

    /**
     * 使用PipeInputStream创建一个读取的线程
     */

    public static void receive() {
        new Thread(new Runnable() {
            @Override
            public void run() {
                byte[] bytes = new byte[100];  //一次性只读取100个字节
                int len = 0;
                try {
                    len = inputStream.read(bytes, 0, 100);  //读取
                    while (len != -1) {
                        System.out.println("已经读取了" + len + "个字节");
                        len = inputStream.read(bytes, 0, 100);
                    }


                } catch (IOException e) {
                    System.out.println("读取失败");
                    System.exit(1);
                }

            }
        }).start();

    }

    public static void main(String args[]) {
        try {
            inputStream.connect(outputStream);  //连接
        } catch (IOException e) {
            System.out.println("连接失败");
            System.exit(1);
        }
        send();
        receive();


    }

}

注意:从上面的运行结果可以看出,缓冲区最多可以写入1024个字节的数据,所以在缓冲区满了之后上面的send进程就会堵塞等待缓冲区空闲,如果recieve进程不继续读取数据了,那么就会一直出现堵塞

问题

写线程正在往缓冲区写数据的时候,但是此时的读线程的结束读取,那么此时的写线程的管道就会发生IOException异常,可以将上面receive进程中的while(true)去掉就可以清楚的看出

读线程正在从缓冲区读数据的时候,但是此时的写线程的管道已经结束了,此时就会引起读线程的管道发生IOException异常,将上面的send进程中的while(true)去掉就可以实现这个问题

必须是启用多线程才能实现管道之间的读写,否则会出现堵塞现象,因为这里的PipeOutputStream每次向缓冲区写入的字节数最大是1024,如果不及时的减少缓冲区的数据量就会出现堵塞

解决方法

后续更新中..........

参考文章

http://www.cnblogs.com/lich/archive/2011/12/11/2283928.html

http://ifeve.com/java-io-%E7%AE%A1%E9%81%93/

http://www.cnblogs.com/chinareny2k/archive/2010/03/24/1693878.html

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/67120.html

相关文章

  • 我的阿里之路+Java面经考点

    摘要:我的是忙碌的一年,从年初备战实习春招,年三十都在死磕源码,三月份经历了阿里五次面试,四月顺利收到实习。因为我心理很清楚,我的目标是阿里。所以在收到阿里之后的那晚,我重新规划了接下来的学习计划,将我的短期目标更新成拿下阿里转正。 我的2017是忙碌的一年,从年初备战实习春招,年三十都在死磕JDK源码,三月份经历了阿里五次面试,四月顺利收到实习offer。然后五月怀着忐忑的心情开始了蚂蚁金...

    姘搁『 评论0 收藏0
  • 好文章必读 - 收藏集 - 掘金

    摘要:方法即为收集器,它接收高阶函数和的后端掘金年的第一天,我坐在独墅湖边,写下这篇文章。正因如此,所以最全系列教程后端掘金是从版本开始引入的一个新的,可以替代标准的。 设计模式之单例模式 - 掘金前言 作为一个好学习的程序开发者,应该会去学习优秀的开源框架,当然学习的过程中不免会去阅读源码,这也是一个优秀程序员的必备素养,在学习的过程中很多人会遇到的障碍,那就是设计模式。很多优秀的框架会运...

    FrozenMap 评论0 收藏0
  • 关于Java IO与NIO知识都在这里

    摘要:从通道进行数据写入创建一个缓冲区,填充数据,并要求通道写入数据。三之通道主要内容通道介绍通常来说中的所有都是从通道开始的。从中选择选择器维护注册过的通道的集合,并且这种注册关系都被封装在当中停止选择的方法方法和方法。 由于内容比较多,我下面放的一部分是我更新在我的微信公众号上的链接,微信排版比较好看,更加利于阅读。每一篇文章下面我都把文章的主要内容给列出来了,便于大家学习与回顾。 Ja...

    Riddler 评论0 收藏0
  • Android IPC机制()用Socket实现跨进程聊天程序

    摘要:简介简介也称作套接字,是在应用层和传输层之间的一个抽象层,它把层复杂的操作抽象为几个简单的接口供应用层调用以实现进程在网络中通信。它分为流式套接字和数据包套接字,分别对应网络传输控制层的和协议。1.Socket简介 Socket也称作套接字,是在应用层和传输层之间的一个抽象层,它把TCP/IP层复杂的操作抽象为几个简单的接口供应用层调用以实现进程在网络中通信。它分为流式套接字和数据包套接字,...

    lufficc 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<