iOS开发——网络编程OC篇&GCDAsyncSocket编程

GCDAsyncSocket编程

同上一篇文章一样,这里也是使用Socket实现一个聊天室,但是这里使用的是一个常用的框架实现的:GCDAsyncSocket

一:导入这个框架

二:声明这个Socket的成员变量,定义一个消息数组  

 1 @interface ViewController ()<UITextFieldDelegate,UITableViewDataSource,UITableViewDelegate,GCDAsyncSocketDelegate>{
 2
 3     GCDAsyncSocket *_socket;
 4 }
 5 @property (weak, nonatomic) IBOutlet NSLayoutConstraint *inputViewConstraint;
 6 @property (weak, nonatomic) IBOutlet UITableView *tableView;
 7
 8 @property (nonatomic, strong) NSMutableArray *chatMsgs;//聊天消息数组
 9
10 @end

懒加载消息数组

1 -(NSMutableArray *)chatMsgs{
2     if (!_chatMsgs) {
3         _chatMsgs = [NSMutableArray array];
4     }
5
6     return _chatMsgs;
7 }

三:链接服务器

 1 - (IBAction)connectToHost:(id)sender {
 2     // 1.建立连接
 3     NSString *host = @"127.0.0.1";
 4     int port = 12345;
 5
 6     // 创建一个Socket对象
 7     _socket = [[GCDAsyncSocket alloc] initWithDelegate:self delegateQueue:dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0)];
 8
 9     // 连接
10     NSError *error = nil;
11     [_socket connectToHost:host onPort:port error:&error];
12     if (error) {
13         NSLog(@"%@",error);
14     }
15 }

链接与断开连接的代理方法

 1 #pragma mark -AsyncSocket的代理
 2 #pragma mark 连接主机成功
 3 -(void)socket:(GCDAsyncSocket *)sock didConnectToHost:(NSString *)host port:(uint16_t)port{
 4     NSLog(@"连接主机成功");
 5 }
 6
 7 #pragma mark 与主机断开连接
 8 -(void)socketDidDisconnect:(GCDAsyncSocket *)sock withError:(NSError *)err{
 9
10     if(err){
11         NSLog(@"断开连接 %@",err);
12     }
13 }

四:登陆

 1 - (IBAction)loginBtnClick:(id)sender {
 2
 3     // 登录
 4     // 发送用户名和密码
 5     // 在这里做的时候,只发用户名,密码就不用发送
 6
 7     // 如果要登录,发送的数据格式为 "iam:zhangsan";
 8     // 如果要发送聊天消息,数据格式为 "msg:did you have dinner";
 9
10     //登录的指令
11     NSString *loginStr = @"iam:zhangsan";
12
13     //把Str转成NSData
14     NSData *data = [loginStr dataUsingEncoding:NSUTF8StringEncoding];
15
16
17     //[_outputStream write:data.bytes maxLength:data.length];
18     // 发送登录指令给服务
19     [_socket writeData:data withTimeout:-1 tag:101];
20 }

五:发送数据

 1 -(BOOL)textFieldShouldReturn:(UITextField *)textField{
 2
 3     NSString *text = textField.text;
 4
 5     NSLog(@"%@",text);
 6     // 聊天信息
 7     NSString *msgStr = [NSString stringWithFormat:@"msg:%@",text];
 8
 9     //把Str转成NSData
10     NSData *data = [msgStr dataUsingEncoding:NSUTF8StringEncoding];
11
12     // 刷新表格
13     [self reloadDataWithText:msgStr];
14
15     // 发送数据
16     [_socket writeData:data withTimeout:-1 tag:102];
17
18     // 发送完数据,清空textField
19     textField.text = nil;
20
21     return YES;
22 }
23
24 #pragma mark 数据成功发送到服务器
25 -(void)socket:(GCDAsyncSocket *)sock didWriteDataWithTag:(long)tag{
26     NSLog(@"数据成功发送到服务器");
27     //数据发送成功后,自己调用一下读取数据的方法,接着_socket才会调用下面的代理方法
28     [_socket readDataWithTimeout:-1 tag:tag];
29 }
30
31 #pragma mark 服务器有数据,会调用这个方法
32 -(void)socket:(GCDAsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag{
33     // 从服务器接收到的数据
34     NSString *recStr =  [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
35
36     NSLog(@"%@ %ld %@",[NSThread currentThread],tag, recStr);
37
38     if (tag == 102) {//聊天返回的数据
39         // 刷新表格
40         [self reloadDataWithText:recStr];
41     }
42 //    }else if(tag == 101 ){//登录返回数据,不应该把数据添加到表格里
43 //
44 //
45 //    }
46
47 }

六:加载聊天数据到界面,并且实现滚动到对应的位置,键盘的键盘

 1 -(void)reloadDataWithText:(NSString *)text{
 2     [self.chatMsgs addObject:text];
 3
 4     // UI刷新要主线程
 5     dispatch_async(dispatch_get_main_queue(), ^{
 6         [self.tableView reloadData];
 7
 8         // 数据多,应该往上滚动
 9         NSIndexPath *lastPath = [NSIndexPath indexPathForRow:self.chatMsgs.count - 1 inSection:0];
10         [self.tableView scrollToRowAtIndexPath:lastPath atScrollPosition:UITableViewScrollPositionBottom animated:YES];
11     });
12
13 }
14
15 #pragma mark 表格的数据源
16
17 -(NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section{
18     return self.chatMsgs.count;
19 }
20
21
22 - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
23 {
24     static NSString *ID = @"Cell";
25     UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];
26
27     cell.textLabel.text = self.chatMsgs[indexPath.row];
28
29     return cell;
30 }
31
32 -(void)scrollViewWillBeginDragging:(UIScrollView *)scrollView{
33     [self.view endEditing:YES];
34 }

键盘处理

 1
 2     // 监听键盘
 3     [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(kbFrmWillChange:) name:UIKeyboardWillChangeFrameNotification object:nil];
 4 }
 5
 6
 7 -(void)kbFrmWillChange:(NSNotification *)noti{
 8     NSLog(@"%@",noti.userInfo);
 9
10     // 获取窗口的高度
11
12     CGFloat windowH = [UIScreen mainScreen].bounds.size.height;
13
14
15
16     // 键盘结束的Frm
17     CGRect kbEndFrm = [noti.userInfo[UIKeyboardFrameEndUserInfoKey] CGRectValue];
18      // 获取键盘结束的y值
19     CGFloat kbEndY = kbEndFrm.origin.y;
20
21
22     self.inputViewConstraint.constant = windowH - kbEndY;
23 }
时间: 2024-08-02 20:22:03

iOS开发——网络编程OC篇&GCDAsyncSocket编程的相关文章

iOS开发——网络实用技术OC篇&amp;网络爬虫-使用青花瓷抓取网络数据

网络爬虫-使用青花瓷抓取网络数据 由于最近在研究网络爬虫相关技术,刚好看到一篇的的搬了过来! 望谅解..... 写本文的契机主要是前段时间有次用青花瓷抓包有一步忘了,在网上查了半天也没找到写的完整的教程,于是待问题解决后抽时间截了图,自己写一遍封存在博客园中以便以后随时查阅. charles又名青花瓷,在iOS开发中的抓包中具有重要作用.最大的三点用处,一就是拦截别人软件的发送的请求和后端接口,练习开发.二是自己后端返回的response拦截修改后再接收以达到测试临界数据的作用.三写脚本重复拦截

iOS开发——网络实用技术OC篇&amp;网络爬虫-使用java语言抓取网络数据

网络爬虫-使用java语言抓取网络数据 前提:熟悉java语法(能看懂就行) 准备阶段:从网页中获取html代码 实战阶段:将对应的html代码使用java语言解析出来,最后保存到plist文件 上一片文章已经介绍我们可以使用两个方式来抓取网络数据实现网络爬虫,并且大致介绍了一下怎么使用正则表达式去实现数据的抓取 由于笔者曾经学过一段时间java和android相关的技术,今天就讲讲怎么使用java去抓取网络数据,关于Python有机会等笔者好好研究一下再来分享,但其实会一种就可以,除非你的需求

ios开发——实用技术篇OC篇&amp;iOS的主要框架

iOS的主要框架         阅读目录 Foundation框架为所有的应用程序提供基本系统服务 UIKit框架提供创建基于触摸用户界面的类 Core Data框架管着理应用程序数据模型 Core Graphics框架帮助你创建图形 Core Animation允许你创建高级的动画和虚拟效果 OpenGL ES 框架提供2D和3D绘图工具 将别的框架添加到工程里 本文是<Sunvey the Major Framworks>一文的翻译 框架是一个目录,这个目录包含了共享库,访问共享库里代码

iOS开发——数据持久化OC篇&amp;(三)对象归档

归档 iOS开发UI篇—ios应用数据存储方式(归档)  一.简单说明 在使用plist进行数据存储和读取,只适用于系统自带的一些常用类型才能用,且必须先获取路径相对麻烦: 偏好设置(将所有的东西都保存在同一个文件夹下面,且主要用于存储应用的设置信息) 归档:因为前两者都有一个致命的缺陷,只能存储常用的类型.归档可以实现把自定义的对象存放在文件中. 二.代码示例 1.文件结构 2.代码示例 YYViewController.m文件 1 // 2 // YYViewController.m 3 /

iOS开发——高级技术OC篇&amp;运行时(Runtime)机制

运行时(Runtime)机制 本文将会以笔者个人的小小研究为例总结一下关于iOS开发中运行时的使用和常用方法的介绍,关于跟多运行时相关技术请查看笔者之前写的运行时高级用法及相关语法或者查看响应官方文档. 下面就来看看什么是运行时,我们要怎么在iOS开发中去使用它. 官方介绍: 这里我们主要关注的是最后一种! 下面来看看Runtime的相关总结 #pragma mark 获取属性成员 /********************************************************

iOS开发——高级UI—OC篇&amp;退出键盘

退出键盘 iOS开发中键盘的退出方法用很多中我们应该在合适的地方使用合适的方法才能更好的提高开发的效率和应用的性能 下面给大家介绍几种最常用的键盘退出方法,基本上iOS开发中的键盘退出方法都是这几种中的一种活着几种. 一:textView 1 //通过委托来实现放弃第一响应者 2 #pragma mark - UITextView Delegate Method 3 -(BOOL)textView:(UITextView *)textView shouldChangeTextInRange:(N

iOS开发——UI精选OC篇&amp;UIApplication,UIWindow,UIViewController,UIView(layer)简单介绍

UIApplication,UIWindow,UIViewController,UIView(layer)简单介绍 一:UIApplication:单例(关于单例后面的文章中会详细介绍,你现在只要知道,单例在应用程序的整个生命周期中只有一个对象). App的启动过程 打开程序之后-> 1:Main函数 2:UIapplicationMain函数 3:初始化UIApplication(创建) 4:设置UIApplication代理和相应的代理属性 5:开启事件循环,监听系统事件 6监测info.p

iOS 开发——版本适配OC篇&amp;项目开发之系统适配问题(关于后台)

项目开发之系统适配问题 关于后台 在之前iOS 开发中,我们精彩遇到关于app适配问题,有屏幕适配,系统适配,但是由于苹果新设备(iphone 6/6p)技新术(sizeClass)的推出在屏幕适配上变的非常简单,而且之前关于app在系统适配上还是比较多的,可能是由于ios7的大改革,后面字啊iOS7盒iOS8上面的api并没有太大的差别,所以关于系统适配,我们并没有考虑太多. 但是,作为一个合格的开发者,不管是实际项目开发中,还是平时我们都要注意系统的视频,毕竟ios9又出来了,虽然常用api

iOS开发——运行时OC篇&amp;使用运行时获取系统的属性:使用自己的手势修改系统自带的手势

使用运行时获取系统的属性:使用自己的手势修改系统自带的手势 有的时候我需要实现一个功能,但是没有想到很好的方法或者想到了方法只是那个方法实现起来太麻烦,一或者确实为了装逼,我们就会想到iOS开发中最牛逼的技术运行时(Runtime) 关于运行时这里就不多说了,请查看笔者之前搬过来了精选文章,这里主要是怎么在实际开发中使用运行时实现我们想要的功能. 比如,在iOS开发中有这样一个问题,关于导航栏中我们点击一个按钮会跳到对应的子控制器,但是这里问题就来了,我们在对应的字控制器中可以点击对应的back