貌似進(jìn)入2016年就沒(méi)再更新簡(jiǎn)書(shū),把寫作忘了末患。罪過(guò)研叫。
OK,進(jìn)入正題璧针,開(kāi)始今天的技術(shù)講解嚷炉。
太概念的知識(shí)網(wǎng)上有很多,我只做概括和快速并正確的使用探橱。
Socket申屹,即使用套接字連接,實(shí)際上是對(duì)TCP/UDP的再次封裝隧膏。
在一般項(xiàng)目中独柑,使用Socket的情況很少,一般都會(huì)使用http實(shí)現(xiàn)客戶端與服務(wù)器端的通信私植。并且是單向的。
但http只能通過(guò)客戶端向服務(wù)器端主動(dòng)發(fā)送網(wǎng)絡(luò)請(qǐng)求车酣,服務(wù)器會(huì)對(duì)該次請(qǐng)求進(jìn)行響應(yīng)曲稼,回傳給客戶端一些數(shù)據(jù)索绪。如果客戶端不主動(dòng)向服務(wù)器端發(fā)送網(wǎng)絡(luò)請(qǐng)求,服務(wù)器端是不能主動(dòng)向客戶端做出請(qǐng)求/響應(yīng)操作的贫悄。
所以瑞驱,為了實(shí)現(xiàn)服務(wù)器端可以主動(dòng)向客戶端發(fā)送請(qǐng)求,我們使用Socket來(lái)實(shí)現(xiàn)這一需求窄坦。
上代碼:
1唤反、先到github下載一個(gè)非常牛逼的庫(kù),地址:https://github.com/robbiehanson/CocoaAsyncSocket
下載到本地后鸭津,有兩個(gè)文件夾:GCD / RunLoop
我個(gè)人比較喜歡使用GCD文件彤侍。看你擅長(zhǎng)哪個(gè)技術(shù)就選擇哪個(gè)文件吧逆趋。原理都是相同的盏阶。
2、將你想使用的文件拖到項(xiàng)目中闻书。
3名斟、一般我們使用第三方庫(kù)都不會(huì)直接使用第三庫(kù)中某個(gè)類或方法,而是對(duì)第三方庫(kù)進(jìn)行再次封裝魄眉。.h文件如圖:
GCDSocketManager是我封裝第三方的一個(gè)封裝類砰盐。
對(duì)外操作都靠這個(gè)類來(lái)完成。所以我使用了單例模式供全局使用坑律。
4岩梳、.m文件
幾個(gè)需要注意的點(diǎn):
(1)封裝類必須要實(shí)現(xiàn)第三方庫(kù)的GCDAsynSocketDelegate協(xié)議
(2)握手次數(shù):客戶端和服務(wù)器端每次連接傳遞消息都會(huì)校驗(yàn),校驗(yàn)就會(huì)根據(jù)握手次數(shù)脾歇。(socket的原理不多說(shuō)了)
(3)斷開(kāi)重連定時(shí)器:特別需要謹(jǐn)記的是只有當(dāng)前項(xiàng)目在前臺(tái)運(yùn)行時(shí)才可以保持長(zhǎng)連接的狀態(tài)蒋腮,當(dāng)項(xiàng)目進(jìn)入后臺(tái)就會(huì)斷開(kāi)長(zhǎng)連接,別問(wèn)為什么藕各。如果你想在程序進(jìn)入后臺(tái)扔可以接到服務(wù)器主動(dòng)給你發(fā)送的消息池摧,使用推送服務(wù)。定時(shí)器在這里的作用就是激况,在當(dāng)前程序處于前臺(tái)作彤,如果socket連接失敗了要讓它自動(dòng)重連。每隔一段時(shí)間讓socket自動(dòng)連接一次乌逐。為了運(yùn)行效率著想竭讳,這個(gè)時(shí)間可以每次進(jìn)行累加。
(4)重連次數(shù):我的重連次數(shù)就是用于時(shí)間的累加浙踢,第一次連接不上就1秒后重連绢慢,第二次連接不上就2秒后重連,以此累加洛波。
5胰舆、.m文件的具體實(shí)現(xiàn)
繼續(xù)下一個(gè)
繼續(xù)下一個(gè)
繼續(xù)下一個(gè)
有連接成功骚露,肯定也會(huì)有連接失敗,以下是對(duì)失敗事件的處理:
繼續(xù)下一個(gè)
連接成功處理完了缚窿,連接失敗也處理完了棘幸。
做事有始有終,有連接倦零,自然也要有斷開(kāi)误续。
好,以下是對(duì)代碼的調(diào)用時(shí)機(jī)和代碼的使用細(xì)節(jié)進(jìn)行一些講解。
(1)什么時(shí)候進(jìn)行長(zhǎng)連扫茅?
長(zhǎng)連當(dāng)然是用戶登錄成功之后需要進(jìn)行長(zhǎng)連蹋嵌。
程序從后臺(tái)切換到前臺(tái)也需要進(jìn)行長(zhǎng)連。
所以可以把進(jìn)行連接的代碼寫在根控制器中诞帐。
(2)什么時(shí)候斷開(kāi)長(zhǎng)連欣尼?
前面說(shuō)過(guò),程序進(jìn)入后臺(tái)是要斷開(kāi)的停蕉。所以在Application進(jìn)入后臺(tái)時(shí)調(diào)用的那個(gè)生命周期方法里寫斷開(kāi)長(zhǎng)連的代碼愕鼓。
(3)關(guān)于兩句很重要的代碼
timeout,超時(shí)時(shí)間慧起。根據(jù)實(shí)際需求去設(shè)置菇晃,我這里設(shè)置為-1。
tag:發(fā)送數(shù)據(jù)和讀取數(shù)據(jù)的tag值一定不能設(shè)置成一樣的蚓挤。
為什么不能設(shè)置成一樣的呢磺送?舉個(gè)例子。
在這里tag相當(dāng)于一個(gè)通道灿意,并且在這個(gè)通道中只能完成一件事情估灿。
如果這個(gè)通道專門用來(lái)發(fā)送數(shù)據(jù)的,就給這個(gè)通道打個(gè)標(biāo)簽缤剧,標(biāo)簽值就是1吧馅袁。
如果這個(gè)通道專門用來(lái)讀取數(shù)據(jù)的,就給這個(gè)通道打個(gè)標(biāo)簽荒辕,標(biāo)簽紙就是200把汗销。
數(shù)值隨便定義,只能你能確保兩者的tag值不一樣抵窒。
但還有一個(gè)細(xì)節(jié)需要注意弛针。
我在上圖代碼中再次調(diào)用了讀取數(shù)據(jù)的代碼,這里的tag值要始終保持一致李皇。
那么在這里寫這句代碼有啥用呢削茁?
在寫這句代碼前,我們已經(jīng)在連接成功的代理方法中寫了這句代碼。
在上圖的代理方法中我可以讀取服務(wù)器響應(yīng)的數(shù)據(jù)付材。
讀取完這一次服務(wù)器響應(yīng)的數(shù)據(jù)后朦拖,下一次服務(wù)器再次發(fā)送數(shù)據(jù)我們?cè)撛趺醋x取厌衔?
就是通過(guò)這句代碼,所以在這里再寫一次捍岳。
這樣我們就可以保證每次服務(wù)器響應(yīng)的數(shù)據(jù)富寿,我們客戶端都可以正常讀取了。
6锣夹、心跳包
心跳包页徐。。就是一個(gè)數(shù)據(jù)包银萍。
每隔一段時(shí)間向服務(wù)器傳輸一次你的數(shù)據(jù)变勇。
當(dāng)然是使用定時(shí)器實(shí)現(xiàn)了。代碼就不上了哈贴唇。
到這里搀绣,又出現(xiàn)了一個(gè)新問(wèn)題。
剛才已經(jīng)說(shuō)了戳气,程序在前臺(tái)才能保持長(zhǎng)連链患。
程序進(jìn)入后臺(tái)長(zhǎng)連就斷了。
那么瓶您,當(dāng)程序在后臺(tái)時(shí)依然需要每隔一段時(shí)間向服務(wù)器發(fā)送一次心跳包麻捻,這個(gè)需求,怎么實(shí)現(xiàn)呀袱?
請(qǐng)聽(tīng)下回講解贸毕。