文档章节

iOS学习之Socket使用简明教程- AsyncSocket

芳仔小脚丫
 芳仔小脚丫
发布于 2014/07/04 11:35
字数 1692
阅读 98532
收藏 115

如果需要在项目中像QQ微信一样做到即时通讯,必须使用socket通讯,本人也是刚学习,分享一下,有什么不对的地方希望大家指正

ios原生的socket用起来不是很直观,所以我用的是AsyncSocket这个第三方库,对socket的封装比较好,只是好像没有带外传输(out—of-band) 如果你的服务器需要发送带外数据,可能得想下别的办法

环境

下载AsyncSockethttps://github.com/robbiehanson/CocoaAsyncSocket类库,将RunLoop文件夹下的AsyncSocket.h, AsyncSocket.m, AsyncUdpSocket.h, AsyncUdpSocket.m 文件拷贝到自己的project中

添加CFNetwork.framework, 在使用socket的文件头

#import <sys/socket.h>
#import <netinet/in.h>
#import <arpa/inet.h>
#import <unistd.h>

使用

1. socket 连接

即时通讯最大的特点就是实时性,基本感觉不到延时或是掉线,所以必须对socket的连接进行监视与检测,在断线时进行重新连接,如果用户退出登录,要将socket手动关闭,否则对服务器会造成一定的负荷。

一般来说,一个用户(对于ios来说也就是我们的项目中)只能有一个正在连接的socket,所以这个socket变量必须是全局的,这里可以考虑使用单例或是AppDelegate进行数据共享,本文使用单例。如果对一个已经连接的socket对象再次进行连接操作,会抛出异常(不可对已经连接的socket进行连接)程序崩溃,所以在连接socket之前要对socket对象的连接状态进行判断

使用socket进行即时通讯还有一个必须的操作,即对服务器发送心跳包,每隔一段时间对服务器发送长连接指令(指令不唯一,由服务器端指定,包括使用socket发送消息,发送的数据和格式都是由服务器指定),如果没有收到服务器的返回消息,AsyncSocket会得到失去连接的消息,我们可以在失去连接的回调方法里进行重新连接。

先创建一个单例,命名为Singleton

Singleton.h

// Singleton.h
#import "AsyncSocket.h"

#define DEFINE_SHARED_INSTANCE_USING_BLOCK(block) \
static dispatch_once_t onceToken = 0; \
__strong static id sharedInstance = nil; \
dispatch_once(&onceToken, ^{ \
sharedInstance = block(); \
}); \
return sharedInstance; \

@interface Singleton : NSObject

+ (Singleton *)sharedInstance;

@end

Singleton.m

+(Singleton *) sharedInstance
{

static Singleton *sharedInstace = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
    
    sharedInstace = [[self alloc] init];
});

return sharedInstace;
}

这样一个单例就创建好了

在.h文件中生命socket变量

@property (nonatomic, strong) AsyncSocket    *socket;       // socket
@property (nonatomic, copy  ) NSString       *socketHost;   // socket的Host
@property (nonatomic, assign) UInt16         socketPort;    // socket的prot

下面是连接心跳失去连接后重连

连接(长连接)

在.h文件中声明方法,并声明代理<AsyncSocketDelegate>

-(void)socketConnectHost;// socket连接

在.m中实现,连接时host与port都是由服务器指定,如果不是自己写的服务器,请与服务器端开发人员交流

// socket连接
-(void)socketConnectHost{

	self.socket    = [[AsyncSocket alloc] initWithDelegate:self];

	NSError *error = nil;

	[self.socket connectToHost:self.socketHost onPort:self.socketPort withTimeout:3 error:&error];

}

心跳

心跳通过计时器来实现 在singleton.h中声明一个定时器

@property (nonatomic, retain) NSTimer        *connectTimer; // 计时器

在.m中实现连接成功回调方法,并在此方法中初始化定时器,发送心跳在后文向服务器发送数据时说明

#pragma mark  - 连接成功回调
-(void)onSocket:(AsyncSocket *)sock didConnectToHost:(NSString 	*)host port:(UInt16)port
{
 	NSLog(@"socket连接成功");
 	
	// 每隔30s像服务器发送心跳包
	self.connectTimer = [NSTimer scheduledTimerWithTimeInterval:30 target:self selector:@selector(longConnectToSocket) userInfo:nil repeats:YES];// 在longConnectToSocket方法中进行长连接需要向服务器发送的讯息

	[self.connectTimer fire];

}

2. socket 断开连接与重连

断开连接

失去连接有几种情况,服务器断开,用户主动cut,还可能有如QQ其他设备登录被掉线的情况,不管那种情况,我们都能收到socket回调方法返回给我们的讯息,如果是用户退出登录或是程序退出而需要手动cut,我们在cut前对socket的userData赋予一个值来标记为用户退出,这样我们可以在收到断开信息时判断究竟是什么原因导致的掉线

在.h文件中声明一个枚举类型

enum{
	SocketOfflineByServer,// 服务器掉线,默认为0
	SocketOfflineByUser,  // 用户主动cut
};

声明断开连接方法

-(void)cutOffSocket; // 断开socket连接

.m

// 切断socket
-(void)cutOffSocket{

	self.socket.userData = SocketOfflineByUser;// 声明是由用户主动切断

	[self.connectTimer invalidate];

	[self.socket disconnect];
}

重连

实现代理方法

-(void)onSocketDidDisconnect:(AsyncSocket *)sock
{
    NSLog(@"sorry the connect is failure %ld",sock.userData);
    if (sock.userData == SocketOfflineByServer) {
        // 服务器掉线,重连
        [self socketConnectHost];
    }
    else if (sock.userData == SocketOfflineByUser) {
        // 如果由用户断开,不进行重连
        return;
    }
    
}

3. socket 发送与接收数据

发送数据 我们补充上文心跳连接未完成的方法

// 心跳连接
-(void)longConnectToSocket{
    
    // 根据服务器要求发送固定格式的数据,假设为指令@"longConnect",但是一般不会是这么简单的指令
    
    NSString *longConnect = @"longConnect";
    
    NSData   *dataStream  = [longConnect dataUsingEncoding:NSUTF8StringEncoding];
    
    [self.socket writeData:dataStream withTimeout:1 tag:1];
    
}

socket发送数据是以栈的形式存放,所有数据放在一个栈中,存取时会出现粘包的现象,所以很多时候服务器在收发数据时是以先发送内容字节长度,再发送内容的形式,得到数据时也是先得到一个长度,再根据这个长度在栈中读取这个长度的字节流,如果是这种情况,发送数据时只需在发送内容前发送一个长度,发送方法与发送内容一样,假设长度为8

NSData   *dataStream  = [@8 dataUsingEncoding:NSUTF8StringEncoding];

[self.socket writeData:dataStream withTimeout:1 tag:1];

接收数据 为了能时刻接收到socket的消息,我们在长连接方法中进行读取数据

 [self.socket readDataWithTimeout:30 tag:0];

如果得到数据,会调用回调方法

-(void)onSocket:(AsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag
{
	// 对得到的data值进行解析与转换即可

	[self.socket readDataWithTimeout:30 tag:0];

}

4. 简单使用说明

我们在用户登录后的第一个界面进行socket的初始化连接操作,在得到数据后,将所需要显示的数据放在singleton中,对变量进行监听后做出相应的操作即可,延伸起来比较复杂,没有真实数据也不太方便说明,大家自己进行探索吧,有问题请在下方留言


    [Singleton sharedInstance].socketHost = @"192.186.100.21";// host设定
    [Singleton sharedInstance].socketPort = 10045;// port设定
    
    // 在连接前先进行手动断开
    [Singleton sharedInstance].socket.userData = SocketOfflineByUser;
    [[Singleton sharedInstance] cutOffSocket];
    
    // 确保断开后再连,如果对一个正处于连接状态的socket进行连接,会出现崩溃
    [Singleton sharedInstance].socket.userData = SocketOfflineByServer;
    [[Singleton sharedInstance] socketConnectHost];
    

全部代码在这里,只是对本文的整合,运行出来为空白

© 著作权归作者所有

共有 人打赏支持
芳仔小脚丫
粉丝 929
博文 83
码字总数 76301
作品 0
闵行
程序员
私信 提问
加载中

评论(70)

赵俊苏
赵俊苏
-(void)sendMessage:(NSString *)socketMsg{
NSData *msgData = [socketMsg dataUsingEncoding:NSUTF8StringEncoding];
[self.socket writeData: msgData withTimeout:1 tag:1];
NSLog(@"send success");
}
自定义一个发送消息 但是为什么每次必须要断开socket重新连接才能获取消息
董良跃
董良跃
最近在弄这个socket通信,看了你的代码,很有帮助,数据能发送出去,也可以接受到服务器返回的数据,但是,我的问题是怎么在同一个方法内发送数据和接受数据,我需要在这一个方法内处理返回的数据,再进行判断true 或者 是false。就是同步执行代码,而不是异步去执行。
A
Ares称雄
粘包怎么处理
苏旋律
苏旋律
-(void)onSocket:(AsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag
请问楼主,这个方法总是不调用为何?
KevinLwg
KevinLwg

引用来自“毛棕毅”的评论

socket除了host和port外 我还有接口 请问一下应该怎么请求数据啊 比如 我接口是 http://172.160.0.202:3001/allCount
host是http://172.160.0.202 port是3001 那我 allCount应该怎么弄啊 新手
你这是http请求的URL。
芳仔小脚丫
芳仔小脚丫

引用来自“xifeng123456”的评论

我每次发送消息的时候都要先断开在连接,还是只有app第一次运行的时候要先断开在发送,而以后只要直接发送就可以了?

回复@xifeng123456 : 应该是只需要连接一次的,发送消息前检查状态即可,如果每次是手动断开是有问题的,如果是因为网络或进程问题断开,那重连就可以
x
xifeng123456
我每次发送消息的时候都要先断开在连接,还是只有app第一次运行的时候要先断开在发送,而以后只要直接发送就可以了?
x
xifeng123456
我每次发送消息的时候都要先断开在连接,还是只有app第一次运行的时候要先断开在发送,而以后只要直接发送就可以了?
Allisonwangjiaojiao
Allisonwangjiaojiao
你好 可以加下QQ吗 ,希望和你交流下 2807338860
田大宝
田大宝
最近用到这个库,在时候有消息堵塞的问题,如果连续接收消息,就不能时时的了,一条一条的出,也就是有缓存池。
iOS开发之即时通讯之Socket(AsyncSocket)

1、AsyncSocket介绍 如果需要在项目中像QQ微信一样做到即时通讯,必须使用socket通讯。 iOS中Socket编程的方式: BSD Socket: BSD Socket 是UNIX系统中通用的网络接口,它不仅支持各种不同的网...

feng_blog
2015/08/31
0
2
(转)直接拿来用!最火的iOS开源项目(一)

AFNetworking 在众多iOS开源项目中,AFNetworking可以称得上是最受开发者欢迎的库项目。AFNetworking是一个轻量级的iOS、Mac OS X网络通信类库,现在是GitHub上第三大Objective-C库。它建立在...

孙启超
2013/06/21
0
0
IOS实例 实现基于Socket TCP/IP通讯

IOS实例 实现基于Socket TCP/IP通讯是本文要介绍的内容,之前写过基于http的网络传输层的通讯,现在项目需要实现tcp/ip的通讯协议,通过网络查找了一下,已经有人写好了公开的类库AsyncSock...

嘻哈开发者
2014/05/30
0
0
10个优秀Objective-C和iOS开发在线视频教程

如果你自己开发iOS应用,你肯定会发现网上有很多资源。学习编程的一个最好的方法就是自己写代码,而开始写代码的最快的方式就是看其他人怎么写。我们从海量视频和学习网站中整理出了我们认为...

mingxun
2014/05/09
0
0
IOS学习,最简单的表格应用程序,学习,列出博客

IOS编程浅蓝教程,这是博客地址http://www.cnblogs.com/haichao/category/425378.html IOS编程浅蓝教程:锲子 IOS编程浅蓝教程(一)先决条件:开始iOS编程的必要准备 IOS编程浅蓝教程(二) Hel...

andy521zhu
2015/01/17
0
0

没有更多内容

加载失败,请刷新页面

加载更多

day149-2018-11-16-英语流利阅读-待学习

抖音登陆海外市场,歪果仁作何反应? Lala 2018-11-16 1.今日导读 抖音在国内已经火了有一段时间了,但你知道它已经进军海外,而且火了一把吗?中国版抖音的宣传语是“记录美好生活”,但外国...

飞鱼说编程
28分钟前
3
1
Confluence 6 教程:在 Confluence 中导航

当你对 Confluence 有所了解后,你会发现 Confluence 使用起来非常简单。这个教程主要是针对你使用的 Confluence 界面进行一些说明,同时向你展示在那里可以进行一些通用的任务和操作。 空间...

honeymose
今天
2
0
sed, awk 练习

1. sed打印某行到某行之间的内容 2. sed 转换大小写 将单词首字母转化大写 将所有小写转化大写 3. sed 在某一行最后面添加一个数字 4. 删除某行到最后一行 解析: {:a;N;$!ba;d} :a : 是...

Fc丶
今天
2
0
babel6升级到7,jest-babel报错:Requires Babel "^7.0.0-0", but was loaded with "6.26.3".

自从将前端环境更新到babel7,jest-babel之前是基于babel6的,执行时候就会报:Requires Babel "^7.0.0-0", but was loaded with "6.26.3". 很烦,因为连续帮好几台电脑修复这个问题,所以记...

曾建凯
今天
1
0
探索802.11ax

802.11ax承诺在真实条件下改善峰值性能和最差情况。 如何改善今天的Wi-Fi? 在决定如何改进当前版本以外的Wi-Fi时,802.11ac,IEEE和Wi-Fi联盟调查了Wi-Fi部署和行为,以确定更广泛使用的障碍...

linuxprobe16
今天
2
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部