iOS开发--浅谈CocoaAsyncSocket编程

Socket就是一种特殊的文件。它是一个连接了两个用户的文件,任何一个用户向Socket里写数据,另一个用户都能看得到,不管这两个用户分布在世界上相距多么遥远的角落,感觉就像坐在一起传纸条一样。

这么讲Socket应该更容易理解吧?这种抽象是非常重要的,因为它屏蔽了更底层的东西,我就想写个程序发送下数据,为什么要关系物理层怎么传输呢,对吧。

所以有了Socket的概念之后,我们在两个客户端之间发送消息可能就是这样的:

指定对方的地址

打开一个和对方连接的Socket

把Socket当成普通的文件,往里写数据

要是发现Socket里有数据,就读出来,那必然是对方发过来的

这样的话,网络编程是不是就非常简单了呢?

那么我们就开始吧

首先呢, 大家先搭配环境,CocoaAsyncSocket是一个三方类库 ,大家需要去guthub上下载或者终端直接安装这里就不过多介绍了。

搭配好框架后我们需要引入 CFNetwork.framework 框架 , 它是xcode本省的框架直接引用就行。

好啦环境搭配好了 让我们封装一个类来操作socket

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文件中声明方法,并声明代理

<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];

时间: 2024-12-20 13:39:05

iOS开发--浅谈CocoaAsyncSocket编程的相关文章

浅谈结对编程

浅谈结对编程 结对编程 结对编程,是一种敏捷软件开发的方法,极限编程的组成部分.结对编程技术是指两位程序员肩并肩地坐在同一台电脑前合作完成同一个设计.同一个算法.同一段代码或同一组测试.一人充当“执行”角色,只负责编程.另外则负责“观察者”(或“导航”),检测bug和把控整体设计.两个程序员具有相同的缺点和盲点的可能性很小,所以当我们采用结对编程的时候会获得一个强大的解决方案.而这个解决方案恰恰是其它软件工程方法学中所没有的. 由于自己长时间都习惯了一个人编程,所以在这次结对编程的初期在做项目的

RCP开发浅谈之SWT,JFACE

RCP开发浅谈之SWT,JFACE SWT 什么是SWT? SWT全名是Standard Widget Toolkit是一个开源的GUI编程框架,我们每一个java开发者,在学习java开发的时候都会接触到awt以及swing这两个图形库,与awt,swing两个图形库不同,swt的优势体现于底层调用本地的图形库,大大提高了运行速度(损失了一定跨平台性).SWT的一个很重要的一点,一个控件并不是单独存在的,而是存在于父控件中.这样当父控件disposed后,子控件也一定很disposed了.每一

iOS开发网络篇—网络编程基础

iOS开发网络篇—网络编程基础 一.为什么要学习网络编程 1.简单说明 在移动互联网时代,移动应用的特征有: (1)几乎所有应用都需要用到网络,比如QQ.微博.网易新闻.优酷.百度地图 (2)只有通过网络跟外界进行数据交互.数据更新,应用才能保持新鲜.活力 (3)如果没有了网络,也就缺少了数据变化,无论外观多么华丽,终将变成一潭死水 移动网络应用 = 良好的UI + 良好的用户体验 + 实时更新的数据 新闻:网易新闻.新浪新闻.搜狐新闻.腾讯新闻 视频:优酷.百度视频.搜狐视频.爱奇艺视频 音乐

Android开发-浅谈架构(二)

写在前面的话 我记得有一期罗胖的<罗辑思维>中他提到 我们在这个碎片化 充满焦虑的时代该怎么学习--用30%的时间 了解70%该领域的知识然后迅速转移芳草鲜美的地方 像游牧民族那样.原话应该不是这样,但是我想说的是 自从我想写一些笔记记录知识的时候 为了不误导其他人 我不得不参考github 一些比较知名大牛的代码.在这个过程中 我下载过很多demo.一点一点的啃 汲取精华的部分 当然也充满疑惑.后来觉得标题取得有点大了 我应该取"学习"类的标题才对.但是另一方面我也只能就

IOS中 浅谈iOS中MVVM的架构设计与团队协作

今天写这篇文章是想达到抛砖引玉的作用,想与大家交流一下思想,相互学习,博文中有不足之处还望大家批评指正.本篇文章的内容沿袭以往博客的风格,也是以干货为主,偶尔扯扯咸蛋(哈哈~不好好工作又开始发表博客啦~). 每日更新关注:http://weibo.com/hanjunqiang  新浪微博 由于本人项目经验有限,关于架构设计方面的东西理解有限,我个人对MVVM的理解主要是借鉴于之前的用过的MVC的Web框架~在学校的时候用过ThinkPHP框架,和SSH框架,都是MVC的架构模式,今天MVVM与

iOS开发-由谈KVC、KVO(重点观察者模式)机制编程

一 不小心,小明在<跟着贝尔去冒险>这个真人秀节目中看到了“一日警察,一世警察”的Laughing哥,整个节目除了贝尔吃牛睾丸都不用刀叉的不雅餐饮文 化外,还是镜头少普通话跟小明一样烂的Laughing Sir那种冷静和沉着稳定留下了深刻印象,不由想起电视剧<学警狙击>中为了不暴露钟立文的身份,要求向自己补一枪的警匪卧底巅峰推动者 --Laughing 哥.那么,卧底这样的工作,在我们程序里有没有呢?答案是肯定的,观察者模式. 文章内容思维导图 一.基本概念 1)KVC概念 KVC

iOS之浅谈纯代码控制UIViewController视图控制器跳转界面的几种方法

一.最普通的视图控制器UIViewContoller 一个普通的视图控制器一般只有模态跳转的功能(ipad我不了解除外,这里只说iPhone),这个方法是所有视图控制器对象都可以用的,而实现这种功能,有两种方法. 1.通过方法 - (void)presentViewController:(UIViewController *)viewControllerToPresent animated: (BOOL)flag completion:(void (^)(void))completion跳转 相

iOS基础——浅谈个人对协议、代理的理解

阅读前的知识储备 请确保在阅读本文时,曾经不止一次亲自动手敲过有关代理模式的代码.如果没有,请在积累一定的经验后阅读,相信会更有收获.也希望大神不吝指教. 什么时候要用协议和代理? 下面举一个例子,谈谈个人对代理.协议的理解,希望能够起到抛砖引玉的效果. 假设现在有这么一个任务需求:页面A需要跳转到页面B(有可能会传入一些参数),页面B填写或者处理一些信息在跳转回页面A的同时还需要把数据返回A. 不要为了用代理模式而用代理模式 代理模式只是一种设计模式,它的价值在于通过一个统一的模式,解决一个原

浅谈中文编程

刚刚接触了中文编程,我感觉挺新鲜.可是美国一会,我就觉得,中文编程实在是不咋滴,比起什么C.C++差的不是一两个档次那么简单,说实在的就是一个天上,一个地上.至于哪个是天,哪个是地那就不用我多说了了. 中文编程起步说实在比较晚,比C/C++/Java晚的不是一点两点,而是几十年.中文编程不仅起步晚,而且中文编程语言还非常单一,单一到什么地步?我想很多人都都会觉得很不可思议,因为知道现在2015年,中文编制语言只有一种,那就是易语言.也许很多人会觉得中文编程会降低中国程序员的开发难度,提高开发效率