java 线程之间如何通信

以前都没有注意这方面的问题,今天看面试题的时候看到了,写一下发表一下看法。

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

  1.管道的创建与使用

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

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

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

  pipeout=new pipedyoutstream();

  pipein= newpipedputsteam(pipepout);

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

2.演示程序: pipeapp

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

  主线程(pipeapp.Java

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

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

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

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

  inputstreamylnpipe =pipeapp.changetoy(xfileln);

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

  inputstreamzlnpipe = pipeapp.changetoz(ylnpipe);

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

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

  datainputstreaminputstream = new datainputstream(zlnpiepe);

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

  String str=inputstream.readline();

  While(str!=null){

    system.out.println(str);

    str=inputstream.readline();

  }

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

  inputstream.close();

  changetoy()方法

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

  datainputstreamxfileln =new datainutstream(inputstream)

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

  pipeoutstreampipeout = new pipeoutputstream();

  pipeinputstreampipeln = new pipedinputsteam(pipeout);

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

  printstreamprintstream = new printstream(pipeout);

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

  ythread ythread=new thread(xfileln,printstream);

  之后,程序启动线程:

  changetoz()方法

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

  datainputstreamyfileln= new datainputstream(inputstream);

  程序创建一个新的管道:

  pipedoutstreampipeout2 = new pipedoutputstream();

  pipedinputstreampipeln2 = new pipedinputsream(pipeout2);

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

源程序如下:

//

//pipeapp.Java-pipeapp的主应用程序

//

importJava.io.*

classpipeapp{

publicstatic void main(string[] args){

pipeapppipeapp=new pipeapp();

try{

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

inputstreamylnpipe = pipeapp.changetoy(xfileln);

inputstreamzlnpipe=pipeapp.changetoz(ylnpipe);

system.out.println();

system.out.println("hereare the results");

system.out.pringln();

datainputstreaminputstream = nes datainputstream(zlnpipe);

stringstr = inputstream.readline();

while(str!=null){

system.out.println(str);

str=inputstream.readline();

}

inputstream.close();

}

catch(exceptione){

system.out.println(e.tostring());

}

}

publicinputstream changetoy(inputstream inputstream){

try{

datainputstreampipeout = new datainputsteam(inputstream);

pipedoutstreampipeout = new pipedoutputstream();

pipedlnsteampipeln = new pipedlnputstream(pipeout);

printstreamprintstream = new printstream(pipeout);

ythreadythread = new ythread(xfileln,printstream);

ythread.start();

returnpipeln;

}

catch(exeptione){

system.out.println(x.tostring());

}

returnnull;

}

publicinputstream changetoz(inputstream inputsteam){

try{

datainputstreamyfileln = new datainputstream(inputstream);

pipeoutputstreampipeln2 = new pipedinputstream(pipeout2);

printrstreamprintstream2 = new printsteam(pipeout2);

zthreadzthread = new zthread(yfileln,printstream2);

zthread.start();

returnpipeln2;

}

catch(exceptione){

system.out.println(e.tostring());

}

returnnull;

}

}

Ythread类和Zthread

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

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

  Ythread(datainputstreamxfileln,pringstream printstream){

    this.xfileln= xfileln;

    this.printstream= printstream;

  }

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

stringxstring = xfileln.readline();

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

  stringystring = xstring.replace('x','y');

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

  prinstream.prinrln(ystring);

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

  pringstram.flush();

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

  pringstram.close();

ythread类的源程序如下:

//

//ythread.Java

//

importJava.io.*;

classythread exteads thread{

datainputstreamxfileln;

pringstreamprintstream;

ythread(datainputstreamxfileln,pringstream.printstream){

this.xfileln= xfileln;

this.printstream= printstream;

}

publicvoid run(){

try{

stringxstring = xfileln.readline();

while(xstring!=null){

stringystring= xstring.replace('x','y');

printstream.pringln(ystring);

printstream.flush();

xstring=xfileln.readline();

}

printstream.close();

}

catch{ioexceptione}{

system.out.println(e.tostring());

}

}

}

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值