GCDAynscSocket简单使用-客户端
這是一篇介紹GCDAynscSocket客戶端簡(jiǎn)單使用的文章(服務(wù)端后續(xù)添加)
背景:在這篇文章之前我對(duì)socket的了解僅限于知道有TCP、UDP兩種方式,使用抓包工具時(shí)甚至看不懂抓包數(shù)據(jù)(慚愧...),所以本文介紹內(nèi)容深度有限,主要介紹了一些簡(jiǎn)單用法。
在這篇文章中主要介紹:
1、使用GCDAynscSocket創(chuàng)建連接、發(fā)送數(shù)據(jù)、接收數(shù)據(jù)、斷開連接;
2、發(fā)生數(shù)據(jù)粘包的處理。
------------------------------------------------------------------------------------
1、創(chuàng)建連接
GCDAynscSocket的初始化般使用兩種方式:
//?aDelegate是設(shè)置的委托對(duì)象,而dq是委托所在的線程,sq是socket所在的線程。其中dp不能為空,sq可以為空
- (id)initWithDelegate:(id)aDelegate delegateQueue:(dispatch_queue_t)dq;
- (id)initWithDelegate:(id)aDelegate delegateQueue:(dispatch_queue_t)dq socketQueue:(dispatch_queue_t)sq;
?
/**
* @brief 連接到服務(wù)器
*/
- (void)socketConnectHost
{
self.socket = [[GCDAsyncSocket alloc]initWithDelegate:self delegateQueue:delegateQueue];
NSError *error = nil;
[self.socket connectToHost:self.socketHost onPort:self.socketPort withTimeout:3 error:&error];
[self.socket readDataWithTimeout:30 tag:100];
}
?
連接是否成功都是在委托方法中查看的
// 連接成功的委托方法
- (void)socket:(GCDAsyncSocket *)sock didConnectToHost:(NSString *)host port:(uint16_t)port
{
NSLog(@"連接成功了...");
}
// 連接失敗或中途斷開連接的委托方法
- (void)socketDidDisconnect:(GCDAsyncSocket *)sock withError:(NSError *)err
{
NSLog(@"連接失敗... 斷開連接了...");
}
?
?
2、發(fā)送數(shù)據(jù)
發(fā)送數(shù)據(jù)比較簡(jiǎn)單,調(diào)用[self.socket writeData:data withTimeout:-1 tag:1]即可
?
/**
? *? @brief ? 寫入字符串?dāng)?shù)據(jù)
? *? @param ? sendStr 要寫入的字符串
? */
- (void)writeAndSendData:(NSString *)sendStr
{
? ? NSData *data = [sendStr dataUsingEncoding: NSUTF8StringEncoding];
? ? [self.socket writeData:data withTimeout:-1 tag:1];
}
?
3、接收數(shù)據(jù)
// 在委托方法中接收數(shù)據(jù)
- (void)socket:(GCDAsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag
{
? ? // 對(duì)得到的data值進(jìn)行解析
? ? [self parseSocket:sock withData:data withTag:tag];
}
在此處有一個(gè)疑問:
在接收數(shù)據(jù)之前,是否必須執(zhí)行[self.socket readDataWithTimeout:-1 tag:100]??最初開始使用時(shí),是必須添加的否則接收不到數(shù)據(jù),但是現(xiàn)在工程中沒有執(zhí)行這句代碼,為什么可以一直接收數(shù)據(jù)呢?
?
4、斷開連接
斷開連接直接調(diào)用方法[self.socket disconnect]即可,可以在委托方法查看是否斷開
/**
? *? @brief ? 切斷socket
?*/
-(void)cutOffSocket
{
? ? [self.socket disconnect];
}
?
5、數(shù)據(jù)粘包處理
通常在tcp中都要處理數(shù)據(jù)粘包。我使用的是給數(shù)據(jù)添加包頭的方式,這也是網(wǎng)上比較推薦的一種方式。
思路:定義好包頭協(xié)議后,在數(shù)據(jù)發(fā)送端每次發(fā)送數(shù)據(jù)之前都添加一個(gè)包頭(因?yàn)槭敲看伟l(fā)送都添加,所以我認(rèn)為只要能夠滿足解包需求包頭要盡可能短)。接收端根據(jù)包頭信息對(duì)接收到的數(shù)據(jù)進(jìn)行拆包。
?
一個(gè)簡(jiǎn)單的包頭數(shù)據(jù): ?有5個(gè)字節(jié)包含兩項(xiàng)內(nèi)容開始:字符$和數(shù)據(jù)包長(zhǎng)度msgLen。
? ? char ? ? startStr = '$';
? ? uint32_t msgLen;
? ? msgLen ?= (uint32_t)(str.length + 5);?
?
在接收端接收到數(shù)據(jù)后,根據(jù)包頭信息,找到開始字符$,然后讀出包的長(zhǎng)度,即可正確拆包。
在整個(gè)過程中要注意:如果發(fā)送端對(duì)數(shù)據(jù)進(jìn)行了編碼,那么接收端要進(jìn)行相應(yīng)的解碼,否則會(huì)造成亂碼,拆包失敗。
?
總結(jié):
在整個(gè)使用學(xué)習(xí)過程中,網(wǎng)絡(luò)知識(shí)一竅不通,又無人指導(dǎo),感覺舉步維艱,還好最終實(shí)現(xiàn)了!回頭看看前面自己繞的彎路覺得很可笑,糾結(jié)的問題更是...
且學(xué)且努力!
?
轉(zhuǎn)載于:https://www.cnblogs.com/songshu-yilia/p/4549160.html
總結(jié)
以上是生活随笔為你收集整理的GCDAynscSocket简单使用-客户端的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 北海旅游六大特产、十大景点是什么?
- 下一篇: SIT与UAT的分别