日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當(dāng)前位置:首頁(yè) > > 充電吧
[導(dǎo)讀]前幾天學(xué)習(xí)了 Android 下 Socket 編程。學(xué)習(xí) Socket 編程是有目的的,需要完成在手機(jī)與 PC 之間的通訊。通訊的內(nèi)容是將手機(jī)上播放的 MP3 信息,通過(guò) Socket 傳輸?shù)?PC

前幾天學(xué)習(xí)了 Android 下 Socket 編程。

學(xué)習(xí) Socket 編程是有目的的,需要完成在手機(jī)與 PC 之間的通訊。通訊的內(nèi)容是將手機(jī)上播放的 MP3 信息,通過(guò) Socket 傳輸?shù)?PC 端。
在參考網(wǎng)上相關(guān) Socket 的文章后,基本上完成了 Socket 功能。所以就繼續(xù)學(xué)習(xí) Android 下音樂(lè)播放器的實(shí)現(xiàn)。在實(shí)現(xiàn)音樂(lè)播放器過(guò)程中,發(fā)現(xiàn)由于音樂(lè)播放器至少要有播放列表和正在播放兩個(gè) Activity,這樣問(wèn)題就來(lái)了:
(1). Socket 只是在第一個(gè) Activity 中實(shí)現(xiàn)了,當(dāng)這個(gè) Activity 活動(dòng)時(shí)沒(méi)有問(wèn)題。但此 Activity 非活動(dòng)時(shí),不能處理 Socket。
(2). 當(dāng)反復(fù)進(jìn)入第一個(gè) Activity 時(shí),會(huì)出現(xiàn) Socket 初始化報(bào)錯(cuò)的問(wèn)題。出現(xiàn)這樣的錯(cuò)誤,是由于 Sokcet 的初始化放在第一個(gè) Activity 的 onCreate 中。

由于在做音樂(lè)播放器時(shí)使用了 Service,所以想到用 Serivce 來(lái)處理 Socket 應(yīng)該沒(méi)有問(wèn)題。但是否有其它的方法呢?由于個(gè)人是剛剛接觸 Android 編程,就不能確定這個(gè)問(wèn)題了!
在論壇提問(wèn),得到的答案是:(1) Serivce; (2) 也可以更改activity的啟動(dòng)方式,讓串口不重復(fù)創(chuàng)建。顯然,第二種方法還沒(méi)有接觸過(guò)。采用第一種 Serivce 來(lái)實(shí)現(xiàn)更可靠一些。


首先,實(shí)現(xiàn) Socket Service。

package?com.jia.leozhengfirstapp;


import?java.io.IOException;
import?java.io.InputStream;
import?java.io.UnsupportedEncodingException;
import?java.net.ServerSocket;
import?java.net.Socket;


import?android.app.Service;
import?android.content.BroadcastReceiver;
import?android.content.Context;
import?android.content.Intent;
import?android.content.IntentFilter;
import?android.os.IBinder;
import?android.util.Log;


public?class?SocketService?extends?Service?{


??private?Socket?clientSocket?=?null;
??private?ServerSocket?mServerSocket?=?null;


??private?SocketAcceptThread?socketAcceptThread?=?null;
??private?SocketReceiveThread?socketReceiveThread?=?null;


??private?SocketReceiver?socketReceiver;


??public?static?final?String?SOCKER_ACTION?=?"com.jia.Socket.Control";
??public?static?final?String?SOCKER_RCV?=?"com.jia.Socket.ReceiveStr";


??private?boolean?stop?=?true;


??@Override
??public?IBinder?onBind(Intent?intent)?{
????//?TODO?Auto-generated?method?stub
????return?null;
??}
???@Override
??public?void?onCreate()?{
??????????super.onCreate();
??????????Log.d("service",?"socket?service?created");
??????????socketReceiver?=?new?SocketReceiver();
??????????IntentFilter?filter?=?new?IntentFilter();
??????????filter.addAction(SOCKER_ACTION);
??????????registerReceiver(socketReceiver,?filter);


??????????socketAcceptThread?=?new?SocketAcceptThread();
????????????//?開(kāi)啟?Socket?監(jiān)聽(tīng)線程
????????????socketAcceptThread.start();
???}


??@Override
??public?void?onStart(Intent?intent,?int?startId)?{
?????Log.d("service",?"socket?service?start");


??}


??@Override
??public?void?onDestroy()?{
??Log.d("service",?"socket?service?destroy!");


??}


??public?class?SocketReceiver?extends?BroadcastReceiver?{


????@Override
????public?void?onReceive(Context?context,?Intent?intent)?{
??????String?action?=?intent.getAction();
????????????if(action.equals(SOCKER_ACTION))?{
??????????????String?sub_action?=?intent.getExtras().getString("ACTION");
??????????????if(sub_action.equals("reconnect"))?{
????????????????Log.d("service",?"socket?service:?reconnect.");


?????????????????socketAcceptThread?=?new?SocketAcceptThread();
??????????????????//?開(kāi)啟?Socket?監(jiān)聽(tīng)線程
??????????????????socketAcceptThread.start();
??????????????}
????????????}
????}
??}


??private?class?SocketAcceptThread?extends?Thread
??{
???????@Override
???????public?void?run()
???????{
?????????Log.d("service",?"socket?service?-?SocketAcceptThread::run");
???????????try?{
???????????????//?實(shí)例化ServerSocket對(duì)象并設(shè)置端口號(hào)為?12589
???????????????mServerSocket?=?new?ServerSocket(12589);
???????????}?catch?(IOException?e)?{
???????????????//?TODO?Auto-generated?catch?block
???????????????e.printStackTrace();
???????????}


???????????try?{
???????????????//?等待客戶端的連接(阻塞)
???????????????clientSocket?=?mServerSocket.accept();
???????????}?catch?(IOException?e)?{
???????????????//?TODO?Auto-generated?catch?block
???????????????e.printStackTrace();
???????????}


???????????socketReceiveThread?=?new?SocketReceiveThread(clientSocket);
???????????stop?=?false;
???????????//?開(kāi)啟接收線程
???????????socketReceiveThread.start();


?????????????Intent?sendIntent?=?new?Intent(SOCKER_RCV);
?????????????sendIntent.putExtra("action",?"ClientIP");
?????????????sendIntent.putExtra("content",?clientSocket.getInetAddress().getHostAddress());
?????????????//?發(fā)送廣播,將被Activity組件中的BroadcastReceiver接收到
?????????????sendBroadcast(sendIntent);
???????}
???}


???private?class?SocketReceiveThread?extends?Thread
???{
???????private?InputStream?mInputStream?=?null;
???????private?byte[]?buf;
???????private?String?str?=?null;
???????Socket?sUsed;


???????SocketReceiveThread(Socket?s)
???????{
?????????Log.d("service",?"socket?service?-?SocketReceiveThread");
???????????try?{
???????????????//?獲得輸入流
???????????????this.mInputStream?=?s.getInputStream();
???????????????sUsed?=?s;
???????????}?catch?(IOException?e)?{
???????????????//?TODO?Auto-generated?catch?block
???????????????e.printStackTrace();
???????????}
???????}


???????@Override
???????public?void?run()
???????{
?????????Log.d("service",?"socket?service?-?SocketReceiveThread::run");
???????????while((!stop)?&&?(!mServerSocket.isClosed()))
???????????{
???????????????this.buf?=?new?byte[2048];


???????????????//?讀取輸入的數(shù)據(jù)(阻塞讀)
???????????????try?{
???????????????????this.mInputStream.read(buf);
???????????????}?catch?(IOException?e1)?{
???????????????????//?TODO?Auto-generated?catch?block
???????????????????e1.printStackTrace();
???????????????}


???????????????//?字符編碼轉(zhuǎn)換
???????????????try?{
???????????????????this.str?=?new?String(this.buf,?"GB2312").trim();
???????????????}?catch?(UnsupportedEncodingException?e)?{
???????????????????//?TODO?Auto-generated?catch?block
???????????????????e.printStackTrace();
???????????????}


???????????????Intent?sendIntent?=?new?Intent(SOCKER_RCV);
???????????????sendIntent.putExtra("action",?"RcvStr");
???????????????sendIntent.putExtra("content",?this.str);
???????????????//?發(fā)送廣播,將被Activity組件中的BroadcastReceiver接收到
???????????????sendBroadcast(sendIntent);
???????????}
???????}
???}
}



在每個(gè) Activity 中處理 SOCKER_RCV action,以響應(yīng) Socket 狀態(tài)的變化和接收到數(shù)據(jù)。
Service 與 Activity 之間通訊需要使用到廣播: Broadcast。
(1) 在 Activity 中定義全局的變量,如下:

public?static?final?String?SOCKER_ACTION?=?"com.jia.Socket.Control";
public?static?final?String?SOCKER_RCV?=?"com.jia.Socket.ReceiveStr";


SocketReceiver?socketReceiver;



(2) 在 Activity 的 onCreate 中注冊(cè)廣播和啟動(dòng) Socket Service,如下:

socketReceiver?=?new?SocketReceiver();
IntentFilter?socketIntentFilter?=?new?IntentFilter();
socketIntentFilter.addAction(SOCKER_RCV);
registerReceiver(socketReceiver,socketIntentFilter);


Intent?socketIntent?=?new?Intent();
socketIntent.setClass(MainActivity.this,?SocketService.class);
startService(socketIntent);???????//?啟動(dòng)??Socket?服務(wù)



(3) SocketReceiver 是繼承自 BroadcastReceiver 的類(lèi),實(shí)現(xiàn)如下:

public?class?SocketReceiver?extends?BroadcastReceiver?{


??@Override
??public?void?onReceive(Context?context,?Intent?intent)?{
????//?TODO?Auto-generated?method?stub
????String?action?=?intent.getAction();
??????????if(action.equals(SOCKER_RCV))?{
????????????String?url?=?intent.getExtras().getString("action");
????????????if(url.equals("ClientIP"))?{
??????????????String?strIP?=?intent.getExtras().getString("content");
????????????}
????????????else?if(url.equals("RcvStr"))?{
??????????????String?strContent?=?intent.getExtras().getString("content");
????????????}
????????????else?if(url.equals("Disconnect"))?{
??????????????String?strContent?=?intent.getExtras().getString("content");
????????????}
????????}
????}
}



(4) Socket 功能實(shí)現(xiàn)后,測(cè)試時(shí)發(fā)現(xiàn)客戶端(也就是 PC 端)斷開(kāi)時(shí)手機(jī)端未檢測(cè)到 Socket 連接斷開(kāi)。
以前使用 WinCE 時(shí),Socket(TCP) 斷開(kāi)時(shí),無(wú)論是客戶端、還是服務(wù)器都可以檢測(cè)到 TCP 斷開(kāi)的事件,并處理。但 Android 下的 Socket 編程機(jī)制竟然沒(méi)有這個(gè)東東。
一,測(cè)試時(shí)發(fā)現(xiàn)當(dāng) PC 端斷開(kāi)后,手機(jī)端的服務(wù)程序在執(zhí)行到下面的代碼段時(shí)不會(huì)阻塞,且函數(shù)的返回值是: -1。
二,在網(wǎng)上查找發(fā)現(xiàn)這個(gè)問(wèn)題是 Android 下 Socket 都有的問(wèn)題,可以通過(guò)發(fā)心跳包來(lái)處理。
所以將下面這段代碼:

//?讀取輸入的數(shù)據(jù)(阻塞讀)
try?{
????this.mInputStream.read(buf);
}?catch?(IOException?e1)?{
????//?TODO?Auto-generated?catch?block
????e1.printStackTrace();
}
修改為如下的代碼:?
try?{
????int?length?=?this.mInputStream.read(buf);
????if(-1?==?length)?{
??????try?{
????????sUsed.sendUrgentData(0xff);
??????}
??????catch(Exception?ex)?{
????????//?鏈接已斷開(kāi)
????????Log.v("service",?"disconnect!!!");
????????stop?=?true;
????????if(null?!=?mServerSocket)?{
??????????mServerSocket.close();
????????}


????????Intent?sendIntent?=?new?Intent(SOCKER_RCV);
????????sendIntent.putExtra("action",?"Disconnect");
????????sendIntent.putExtra("content",?"read?is?-1?&?Urgent?Exception!");
????????sendBroadcast(sendIntent);


????????continue;
??????}
????}
}?catch?(IOException?e1)?{
????//?TODO?Auto-generated?catch?block
????e1.printStackTrace();
}
本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專(zhuān)欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動(dòng)電源

在工業(yè)自動(dòng)化蓬勃發(fā)展的當(dāng)下,工業(yè)電機(jī)作為核心動(dòng)力設(shè)備,其驅(qū)動(dòng)電源的性能直接關(guān)系到整個(gè)系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動(dòng)勢(shì)抑制與過(guò)流保護(hù)是驅(qū)動(dòng)電源設(shè)計(jì)中至關(guān)重要的兩個(gè)環(huán)節(jié),集成化方案的設(shè)計(jì)成為提升電機(jī)驅(qū)動(dòng)性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機(jī) 驅(qū)動(dòng)電源

LED 驅(qū)動(dòng)電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個(gè)照明設(shè)備的使用壽命。然而,在實(shí)際應(yīng)用中,LED 驅(qū)動(dòng)電源易損壞的問(wèn)題卻十分常見(jiàn),不僅增加了維護(hù)成本,還影響了用戶體驗(yàn)。要解決這一問(wèn)題,需從設(shè)計(jì)、生...

關(guān)鍵字: 驅(qū)動(dòng)電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動(dòng)電源的公式,電感內(nèi)電流波動(dòng)大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計(jì) 驅(qū)動(dòng)電源

電動(dòng)汽車(chē)(EV)作為新能源汽車(chē)的重要代表,正逐漸成為全球汽車(chē)產(chǎn)業(yè)的重要發(fā)展方向。電動(dòng)汽車(chē)的核心技術(shù)之一是電機(jī)驅(qū)動(dòng)控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機(jī)驅(qū)動(dòng)系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動(dòng)汽車(chē)的動(dòng)力性能和...

關(guān)鍵字: 電動(dòng)汽車(chē) 新能源 驅(qū)動(dòng)電源

在現(xiàn)代城市建設(shè)中,街道及停車(chē)場(chǎng)照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進(jìn)步,高亮度白光發(fā)光二極管(LED)因其獨(dú)特的優(yōu)勢(shì)逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動(dòng)電源 LED

LED通用照明設(shè)計(jì)工程師會(huì)遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動(dòng)電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動(dòng)電源的電磁干擾(EMI)問(wèn)題成為了一個(gè)不可忽視的挑戰(zhàn)。電磁干擾不僅會(huì)影響LED燈具的正常工作,還可能對(duì)周?chē)娮釉O(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來(lái)解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動(dòng)電源

開(kāi)關(guān)電源具有效率高的特性,而且開(kāi)關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機(jī)重量也有所下降,所以,現(xiàn)在的LED驅(qū)動(dòng)電源

關(guān)鍵字: LED 驅(qū)動(dòng)電源 開(kāi)關(guān)電源

LED驅(qū)動(dòng)電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動(dòng)LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動(dòng)電源
關(guān)閉