程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> J2EE >> Java中利用管道實現線程間的通訊

Java中利用管道實現線程間的通訊

編輯:J2EE

在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編譯


 

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved