J2ME中多線程網(wǎng)絡(luò)連接編程的分析
介紹并分析了J2ME的通用網(wǎng)絡(luò)連接框架(GCF),并以此為基礎(chǔ),指出了在J2ME網(wǎng)絡(luò)連接編程中存在的問(wèn)題,并提出了通過(guò)構(gòu)建多線程的兩種解決方法。
本文引用地址:http://2s4d.com/article/149138.htm關(guān)鍵詞J2MEGCF多線程網(wǎng)絡(luò)連接
引言
J2ME(Java2MicroEdition)是Java2的一個(gè)組成部分,它與J2SE、J2EE并稱。J2ME是一種高度優(yōu)化的Java運(yùn)行環(huán)境,主要針對(duì)消費(fèi)類電子設(shè)備的,例如蜂窩電話、可視電話、數(shù)字機(jī)頂盒和汽車(chē)導(dǎo)航系統(tǒng)等等。即J2ME是為消費(fèi)電子產(chǎn)品和手持設(shè)備量身定制的Java專用版本。
J2ME的出現(xiàn)使開(kāi)發(fā)跨平臺(tái)的消費(fèi)類電子產(chǎn)品的應(yīng)用軟件成為可能。Java語(yǔ)言的與平臺(tái)無(wú)關(guān)的特性移植到小型電子設(shè)備上,允許移動(dòng)無(wú)線設(shè)備之間共享應(yīng)用程序。它提供了基于HTTP的高級(jí)Internet協(xié)議,使移動(dòng)電話能以Client/Server方式直接訪問(wèn)Internet的全部信息,從而使得不同的Client訪問(wèn)不同的資源。
在將來(lái)的無(wú)線通信時(shí)代中,大量的網(wǎng)絡(luò)應(yīng)用程序?qū)⒈婚_(kāi)發(fā)出來(lái)去滿足無(wú)線移動(dòng)通訊的要求,而要充分的發(fā)揮無(wú)線移動(dòng)通訊設(shè)備的通信能力,J2ME網(wǎng)絡(luò)編程就變得尤為重要。那么為了高效地進(jìn)行網(wǎng)絡(luò)編程,就要利用Java語(yǔ)言的多線程編程機(jī)制。
1、J2ME的網(wǎng)絡(luò)連接框架(GCF)
通用連接框架(GenericConnectionFramework,GCF)為資源有限的設(shè)備提供了一個(gè)可擴(kuò)展的、通用的I/O框架。GCF是一組在javax.microedition.io包中定義的接口。圖1顯示了GCF的類層次結(jié)構(gòu)。
圖1GCF的類層次結(jié)構(gòu)
在GCF中共定義了七個(gè)接口,其中Connection是最基本的連接類型。且同時(shí)提供了對(duì)數(shù)據(jù)包和流連接的支持。沿著層次結(jié)構(gòu)向下派生出提供更多功能的接口。例如,StreamConnection接口支持輸入流和輸出流,ContentConnection接口又?jǐn)U展了StreamConnection接口以支持對(duì)流的內(nèi)容類型、數(shù)據(jù)長(zhǎng)度和編碼格式的確定,HttpConnection接口又?jǐn)U展了ContentConnection接口以支持對(duì)于標(biāo)準(zhǔn)的HTTP請(qǐng)求。如在框架層規(guī)定的適用于手機(jī)或雙向?qū)ず魴C(jī)的移動(dòng)信息設(shè)備框架MIDP(MobileInformationDeviceProfile)在其MIDP1.0規(guī)范只要求設(shè)備支持HTTP連接協(xié)議,而更新的MIDP2.0規(guī)范要求同時(shí)支持HTTP和HTTPS,后者提供了對(duì)更安全的網(wǎng)絡(luò)連接的支持。
2、網(wǎng)絡(luò)編程中的多線程
由于目標(biāo)設(shè)備具有內(nèi)存小,計(jì)算能力弱和電池供電等特點(diǎn),所以如何使應(yīng)用程序高效的運(yùn)行就成為開(kāi)發(fā)中的一個(gè)大問(wèn)題.尤其針對(duì)手機(jī)等移動(dòng)信息設(shè)備時(shí),無(wú)線通訊的特點(diǎn)又對(duì)我們的程序提出了更高的要求.從代碼優(yōu)化的角度,在網(wǎng)絡(luò)編程中引入多線程就顯得十分重要。
當(dāng)程序運(yùn)行的時(shí)候,ApplicationManagementSoftware(應(yīng)用管理軟件)首先初始化一個(gè)MIDlet,然后調(diào)用它的startApp()方法使得MIDlet進(jìn)入active狀態(tài),這條程序分支就是主線程,它執(zhí)行其他的方法后都會(huì)返回到這個(gè)分支上來(lái)繼續(xù)執(zhí)行。然而網(wǎng)絡(luò)連接是個(gè)可能堵塞的操作,意味著它可能長(zhǎng)時(shí)間都不返回。
在SUN公司的無(wú)線開(kāi)發(fā)包WTK中模擬一段網(wǎng)絡(luò)連接程序運(yùn)行時(shí),WTK會(huì)提示網(wǎng)絡(luò)連接工作可能會(huì)堵塞用戶輸入,需要?jiǎng)?chuàng)建另外一個(gè)線程去進(jìn)行聯(lián)網(wǎng)操作。針對(duì)以上情況,引入多線程的處理機(jī)制。
2.1利用Thread類與Runnable接口
編寫(xiě)J2ME網(wǎng)絡(luò)連接應(yīng)用程序的時(shí)候往往借助Command顯示部件,調(diào)用其事件處理函數(shù)完成網(wǎng)絡(luò)的連接工作,代碼框架如下:
publicvoidcommandAction(Commandc,Displayables){
if(c==sendCommand){
requestConnect();//連接方法
}
elseif(
c==backCommand){
display.setCurrent(mainForm);}
else{
destroyApp(false);
notifyDestroyed();}
}
//獲取一個(gè)HTTP的連接
privatevoidrequestConnect(){
Stringurl=URL.URLString
HttpConnectionhpc=null;
try{
hpc=(HttpConnection)Connector.open(url);
intstatus=hpc.getResponseCode();
if(status!=HttpConnection.HTTP_OK)
content=聯(lián)機(jī)失敗!;
else
content=已聯(lián)機(jī)!;
}
catch(IOExceptione){System.out.println(content);}
try{
if(hpc!=null)hpc.close();
}
catch(IOExceptione2){}}
上面的程序工作原理可用圖2的工作原理圖a來(lái)表示。
圖2工作原理圖a
圖2工作原理圖a
分析圖2可以得出,如果這樣的網(wǎng)絡(luò)連接程序在手機(jī)上運(yùn)行,那么將可能長(zhǎng)時(shí)間得不到響應(yīng)。因?yàn)檫B接工作只有一個(gè)主線程,所有的應(yīng)用都是在這個(gè)主線程當(dāng)中進(jìn)行的,如果此主線程不返回,那么就不能進(jìn)行后面的行為,用戶也不能進(jìn)行任何操作。
下面改進(jìn)一下程序,創(chuàng)建一個(gè)實(shí)現(xiàn)Runnable接口的ConnectPipe類來(lái)創(chuàng)建多線程。代碼如下
//實(shí)現(xiàn)Runnable接口
classConnectPipeimplementRunnable{
……
publicvoidrun(){
requestConnect();}
}
修改commandAction函數(shù):
publicvoidcommandAction(Commandc,Displayables){
if(c==sendCommand){
//創(chuàng)建新線程
newThread(newConnectPipe()).start();
}
elseif(c==backCommand){
……
}
}
修改之后程序能夠較為順利的運(yùn)行,當(dāng)處理網(wǎng)絡(luò)連接的時(shí)候,啟動(dòng)一個(gè)線程后主線程會(huì)立刻返回,兩個(gè)線程并行,不會(huì)引發(fā)在此地堵塞。其工作原理可用圖3的工作原理圖b來(lái)表示。
圖3工作原理圖b
詳細(xì)分析圖3,又發(fā)現(xiàn)盡管程序可以正常工作,但是每次用戶按下按鈕的時(shí)候都會(huì)有新的線程產(chǎn)生,這樣顯然不夠高效,同時(shí),異步的行為又有可能使兩個(gè)線程間產(chǎn)生死鎖。幸好java中提供了wait()和notify()/notifyAll()來(lái)進(jìn)行線程間的通訊,協(xié)調(diào)同步問(wèn)題。那么對(duì)應(yīng)本程序中的線程同步問(wèn)題,設(shè)計(jì)思想如下:?jiǎn)?dòng)線程后,讓其進(jìn)入等待的狀態(tài),當(dāng)用戶激活Command事件的時(shí)候喚醒線程,才讓其繼續(xù)運(yùn)行。代碼類似如下:
publicsynchronizedvoidrun(){
while(dealing){
try{wait();}//線程等待
catch(InterruptedExceptionie){}
if(dealing)requestConnect();
}
}
publicsynchronizedvoiddeal(){
notify();//喚醒線程
}
其中dealing變量用于定義一個(gè)鎖,當(dāng)其為true時(shí),當(dāng)前線程等待,直到用戶激活Command事件之后,調(diào)用deal()方法中的notify()喚醒當(dāng)前線程繼續(xù)運(yùn)行。這樣程序就顯得相當(dāng)?shù)母咝?,也在很大程度上避免了線程間的死鎖問(wèn)題。其工作原理可用圖4的工作原理圖c來(lái)表示。
圖4工作原理圖c
2.2利用系統(tǒng)類Timer和TimerTask
系統(tǒng)類Timer類是一個(gè)計(jì)時(shí)器,和TimerTask類結(jié)合可以來(lái)實(shí)現(xiàn)在MIDlet中定時(shí)執(zhí)行特定任務(wù)。需要說(shuō)明的是每一個(gè)Timer對(duì)象實(shí)際上都是一個(gè)后臺(tái)運(yùn)行的獨(dú)立的線程。這是因?yàn)檎{(diào)度一次的任務(wù)都是由TimerTask類的實(shí)現(xiàn)對(duì)象負(fù)責(zé),TimerTask類是一個(gè)抽象類,它的主要特點(diǎn)是實(shí)現(xiàn)了Runnable接口,因此擴(kuò)展了必須實(shí)現(xiàn)的publicvoidrun()方法。
所以,在J2ME的網(wǎng)絡(luò)編程中,我們可以利用Timer類和TimerTask類來(lái)建立線程,完成網(wǎng)絡(luò)連接等工作。設(shè)計(jì)思想如下:創(chuàng)建一個(gè)Timer類計(jì)時(shí)器,一個(gè)完成網(wǎng)絡(luò)連接功能的TimerTask類,在系統(tǒng)空閑時(shí),反復(fù)調(diào)度任務(wù)要求連接,直到連接成功,再調(diào)用TimerTask類的cancel()可以停止一個(gè)具體的調(diào)度任務(wù)。核心代碼類似如下:
classConnectTimerimplementTimerTask{
ConnectTimer(){
m_Timer=newTimer();//定義Timer
m_Timer.schedule(this,500,5000);//調(diào)度任務(wù)
}
……
publicsynchronizedvoidrun(){
requestConnect();//連接方法
cancel();//取消任務(wù)
}
}
publicvoidcommandAction(Commandc,Displayables){
if(c==sendCommand){
newConnectTimer;}
elseif(c==backCommand){
……
}
}
3、結(jié)束語(yǔ)
綜上所述,在J2ME的應(yīng)用開(kāi)發(fā)中網(wǎng)絡(luò)程序的設(shè)計(jì)具有重要的地位,而編程的關(guān)鍵又在于編寫(xiě)高效友好的J2ME網(wǎng)絡(luò)連接程序。通過(guò)Java語(yǔ)言內(nèi)置的多線程處理機(jī)制,利用線程進(jìn)行同步并行處理,解決了網(wǎng)絡(luò)連接時(shí)的阻塞問(wèn)題,達(dá)到了程序高效運(yùn)行的目的
評(píng)論