`
kiddwyl
  • 浏览: 402686 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Java 中利用管道实现线程间的通讯

阅读更多
 在Java 语言中,提供了各种各样的输入输出流(stream),使我们能够很方便的对数据进行操作,其中,管道(pipe)流是一种特殊的流,用于在不同线程(threads)间直接传送数据。一个线程发送数据到输出管道,另一个线程从输入管道中读数据。通过使用管道,实现不同线程间的通讯。 无需求助于类似临时文件之类的东西。本文在简要介绍管道的基本概念后,将以一个具体的实例pipeapp加以详细说明。

  1.管道的创建与使用

  Java提供了两个特殊的专门的类专门用于处理管道,它们就是pipedinputstream类和pipeoutputstream类。

  Pipedinputstream代表了数据在管道中的输出端,也就是线程向管道读数据的一端;pipeoutputstream代表了数据在管道中的输入端,也就是线程向管道写数据的一端,这两个类一起使用可以提供数据的管道流。

  为了创建一个管道流,我们必须首先创建一个pipeoutstream对象,然后,创建pipeinputstream对象,实例如下:

  pipeout= new pipedyoutstream();
  pipein= new pipedputsteam(pipepout);

  一旦创建了一个管道后,就可以象操作文件一样对管道进行数据的读写。

  2.演示程序: pipeapp

  应用程序由三个程序组成:主线程(pipeapp.Java)及由主线程启动的两个二级线程(ythread.Java和zthread.Java),它们使用管道来处理数据。程序从一个内容为一行一行"x"字母的"input.txt"文件中读取数据,使用管道传输数据,第一次是利用线程ythread将数据"x"转换为"y",最后利用线程zthread将"y"转换为"z",之后,程序在屏幕上显示修改后的数据。

  主线程 (pipeapp.Java)

  在main()方法中,程序首先创建一个应用对象:pipeapp pipeapp=new pipeapp();

由于程序中流操作都需要使用IOException异常处理,所以设置了一个try块。在try中,为了从源文件中读取数据,程序为"input.txt"文件创建了一个输入流Xfileln,:

  fileinputstream xfileln= new fileinputstream("input.txt");

  新的输入流传递给changetoy()方法,让线程ythread能读取该文件:

  inputstream ylnpipe =pipeapp.changetoy(xfileln);

  changetoy()方法创建将输入数据"x"改变到"y"的线程ythread,并返回该线程的输入管道:

  inputstream zlnpipe = pipeapp.changetoz(ylnpipe);

  changetoz()方法启动将数据从"y"改变到"z"的线程zehread,主程序将使用从changetoz()返回的输入管道。得到以修改的数据。

  然后,程序将管道输入流定位到datainputstream对象,使程序能够使用readline()方法读取数据:
  datainputstream inputstream = new datainputstream(zlnpiepe);

  创建了输入流以后,程序就可以以行一行的读取数据病显示在屏幕上。

  String str= inputstream.readline();
  While(str!=null)
  {
    system.out.println(str);
    str=inputstream.readline();
  }

  显示完成之后,程序关闭输入流:

  inputstream.close();
  changetoy()方法

  changetoy()方法首先通过传递一个参数inputstream给datainputstream对象来定位资源的输入流,使程序能使用readline()方法从流中读取数据:

  datainputstream xfileln =new datainutstream(inputstream);

  然后,changetoy()创建输出管道和输入管道:

  pipeoutstream pipeout = new pipeoutputstream();
  pipeinputstream pipeln = new pipedinputsteam(pipeout);

  为了能够使用println()方法输出修改的后的文本行到管道,程序将输出管道定位到printstream对象:

  printstream printstream = new printstream(pipeout);

  现在,程序可以创建将数据从x改变到y的线程,该线程是ythread类的一个对象,他传递两个参数:输入文件(xfileln)和输出管道(调用printstream)

  ythread ythread =new thread(xfileln,printstream);

  之后,程序启动线程:

  changetoz()方法

  changetoz()方法与changetoy()方法很相似,他从changetoy()返回的输入流开始:

  datainputstream yfileln= new datainputstream(inputstream);

  程序创建一个新的管道:

  pipedoutstream pipeout2 = new pipedoutputstream();
  pipedinputstream pipeln2 = new pipedinputsream(pipeout2);

  该线程通过这个新的管道发出修改后的数据(输入流pipeln2)给主程序。

源程序如下:

//
//pipeapp.Java-pipeapp的主应用程序
//
import Java.io.*
class pipeapp
{
public static void main(string[] args)
{
pipeapp pipeapp=new pipeapp();
try
{
fileinputstream xfile =new fileinputstream("input.txt");
inputstream ylnpipe = pipeapp.changetoy(xfileln);
inputstream zlnpipe=pipeapp.changetoz(ylnpipe);
system.out.println();
system.out.println("here are the results");
system.out.pringln();
datainputstream inputstream = nes datainputstream(zlnpipe);
string str = inputstream.readline();
while (str!=null)
{
system.out.println(str);
str=inputstream.readline();
}
inputstream.close();
}
catch(exception e)
{
system.out.println(e.tostring());
}
}
public inputstream changetoy(inputstream inputstream)
{
try
{
datainputstream pipeout = new datainputsteam(inputstream);
pipedoutstream pipeout = new pipedoutputstream();
pipedlnsteam pipeln = new pipedlnputstream(pipeout);
printstream printstream = new printstream(pipeout);
ythread ythread = new ythread(xfileln,printstream);
ythread.start();
return pipeln;
}
catch(exeption e)
{
system.out.println(x.tostring());
}
return null;
}
public inputstream changetoz(inputstream inputsteam)
{
try
{
datainputstream yfileln = new datainputstream(inputstream);
pipeoutputstream pipeln2 = new pipedinputstream(pipeout2);
printrstream printstream2 = new printsteam(pipeout2);
zthread zthread = new zthread(yfileln,printstream2);
zthread.start();
return pipeln2;
}
catch(exception e)
{
system.out.println(e.tostring());
}
return null;
}
}

  Ythread类和Zthread类

  由于ythread类与zthread类基本一样,在此仅以ythread为例加以说明。

  Ythread的构造器接收两个参数:输入的文件和第一个管道的输出端,构造器存储这两个参数作为类的数据成员:

  Ythread(datainputstream xfileln,pringstream printstream)
  {
    this.xfileln = xfileln;
    this.printstream = printstream;
  }

  线程通过run()方法来处理数据。首先读取一行数据,确保xstring不为空的情况下循环执行:
string xstring = xfileln.readline();

  每读一行数据,完成一次转换

  string ystring = xstring.replace('x','y');

  然后将修改后的数据输出到管道的输出端:

  prinstream.prinrln(ystring);

  为了确保所有缓冲区的数据完全进入管道的输出端:

  pringstram.flush();

  循环完成后,线程关闭管道输出流:

  pringstram.close();

ythread类的源程序如下:

//
//ythread.Java
//
import Java.io.*;
class ythread exteads thread
{
datainputstream xfileln;
pringstream printstream;
ythread(datainputstream xfileln,pringstream.printstream)
{
this.xfileln = xfileln;
this.printstream = printstream;
}
public void run()
{
try
{
string xstring = xfileln.readline();
while(xstring!=null)
{
string ystring= xstring.replace('x','y');
printstream.pringln(ystring);
printstream.flush();
xstring= xfileln.readline();
}
printstream.close();
}
catch{ioexception e}
{
system.out.println(e.tostring());
}
}
}

  pipeapp应用程序使用microsoft visual j++1.1编译

分享到:
评论

相关推荐

    Java中利用管道实现线程间的通讯

    在Java中,管道(Pipe)流提供了一种有效的方法,允许数据从一个线程传递到另一个线程,从而实现线程间的通信。管道通常由两个流组成:一个输出流用于写入数据,另一个输入流用于读取数据。这种方式使得数据可以在不...

    Java多线程文章系列.pdf

    #### 十、Java中利用管道实现线程间的通讯 - **管道创建与使用**: - 创建管道对象,如`Pipe`。 - 通过`sink()`和`source()`方法实现数据传输。 #### 十一、实战Java多线程编程精要之高级支持 - **线程组**: - ...

    多线程快速入门.zip

    在“0002-蚂蚁课堂(每特学院)-多线程快速入门”这个资源中,你将学习如何创建线程,理解线程的状态转换,如何使用同步机制避免线程安全问题,以及如何有效地实现线程间的通信。通过实践案例,你将掌握如何在实际项目...

    通过AndroirdAccessory实现两个手机的USB通讯Demo

    这个Demo项目“通过AndroidAccessory实现两个手机的USB通讯Demo”正是展示了如何利用这种模式来实现在两台Android设备间的数据传输。下面将详细讲解Android Accessory模式的工作原理、实现步骤以及如何在这个Demo中...

    JAVA NIO 异步通信模板服务端

    4. **Pipe(管道)**:用于两个线程间的数据传输。 ### NIO服务端模板结构 1. **初始化ServerSocketChannel**:首先创建ServerSocketChannel,并绑定到指定的端口,然后将其注册到Selector上,监听ACCEPT事件。 2...

    唯品会java方向实习笔试题

    2. **进程间的通讯方式及区别** 进程间通信(IPC)是操作系统中的一个重要概念,它允许不同的进程之间交换数据或信息。常用的IPC机制包括但不限于: - **管道(Pipe)**:主要用于具有亲缘关系的进程之间,即父子...

    netty QQ聊天室代码案例

    在实际应用中,要实现100万级别的流量处理,需要考虑的关键技术包括连接池管理、线程模型优化、内存池使用、零拷贝机制等。开发者需要理解Netty的ByteBuf、ChannelHandler、EventLoopGroup等核心组件,以及如何利用...

    Netty 17道面试题和答案.docx

    - **即时通讯**:Netty可用于创建类似微信的聊天应用,实现用户间的实时通信。 - **消息推送系统**:很多消息推送平台基于Netty构建,确保消息能够及时、可靠地送达客户端。 - **其他**:Netty还可用于实现...

    守护进程(互相监听)

    在"双守护进程"的场景中,两个守护进程可能通过管道、套接字或共享内存等机制进行通信。他们可能各自负责不同的系统服务,或者一个作为主守护进程,负责启动、监控和恢复另一个。这种设计能够提供更好的容错性和自愈...

    目前最详尽的openfire介绍

    MINA通过Java NIO提供跨不同传输类型(如TCP/IP、UDP/IP)的统一API,同时支持串口通讯和虚拟机内部管道通讯。它还提供了过滤器作为扩展机制,类似于Servlet过滤器,以及高级和低级API接口,满足不同层次的开发需求...

    【BAT必备】netty面试题

    - **微服务架构下的服务通信:**Netty可以作为底层通信库支持微服务间的交互。 #### 四、Netty进阶话题 **4.1 如何理解Netty的异步非阻塞特性?** Netty的异步非阻塞特性意味着在处理网络事件时不会阻塞当前线程...

Global site tag (gtag.js) - Google Analytics