深度解析Cocoa异步请求和libxml2.dylib教程(1)

深度解析Cocoa异步请求libxml2.dylib教程是本文要介绍的内容,不多说,直接进入话题,很早就在cocoachina上看到这个框架了,今天终于有机会来使用这个东东了.

我这里写一下,如何往iphone项目中添加这个框架.

步骤如下:

1.下载该framework : http://github.com/pokeb/asi-http-request/tree

2.将class根目录下的文件全拷贝到自己的项目中,另外还要在 External/Reachability/下将其中的Reachability.h/m

也拷贝到自己的项目中.

3.添加需要的framework.可以参考 http://allseeing-i.com/ASIHTTPRequest/Setup-instructions

需要额外添加的有: CFNetwork.framework, MobileCoreServices.framework,SystemConfiguration.framework,libz.1.2.3.dylib,libxml2.dylib

然后运行项目,会发现有很多xml相关的error,不用急,这时因为libxml2.dylib这个framework(这个框架不是很friendly,我们还需要做一些工作).

在xcode中project->edit project settings->然后search "search paths",然后在path中添加 /usr/include/libxml2

这样就ok了,可以根据官方的教程来学习了.

http://allseeing-i.com/ASIHTTPRequest/How-to-use

我下了一个sample code  XMLPerformance 解析xml,我建了一个工程照着上面做,但是编译时提示错误,

  1. error libxml/tree.h: No such file or directory 

我立刻想到没有add Frameworks ,我把libsqlite3.dylib 和 libxml2.dylib都加进去了,但是还是报错。

  1. error libxml/tree.h: No such file or directory  
  2. An error on the .h is a compile-time error with your Header Search Paths, not a .dylib or a linker error.  
  3. You have to ensure that /usr/include/libxml2 is in your Header Search Paths in your Release configuration。 

在iphone开发中,异步操作是一个永恒的话题,尤其当iphone手机需要和远程服务器进行交互时,使用异步请求是很普遍的做法。

通常,这需要NSURLConnection和NSOperation结合起来使用。这方面的资料网络上自然有不少的介绍,不过要找一个能运行的代码也并不容易。许多文章介绍的并不全面,或者使用了过时的SDK,在新IOS版本下并不适用(当前最新的ios是4.2了)。这些代码很经典,但仍然很容易使人误入歧途。

本文总结了众多文档介绍的方法和代码,揭示了异步操作中的实现细节和初学者(包括笔者)易犯的错误,使后来者少走弯路。

一、使用NSOperation实现异步请求

1、新建类,继承自NSOperation。

  1. @interface URLOperation : NSOperation  
  2. {  
  3.     NSURLRequest*  _request;  
  4.     NSURLConnection* _connection;  
  5.     NSMutableData* _data;  
  6.     //构建gb2312的encoding  
  7.     NSStringEncoding enc;  
  8. }  
  9. - (id)initWithURLString:(NSString *)url;  
  10. @property (readonly) NSData *data;  
  11. @end 

接口部分不多做介绍,我们来看实现部分。

首先是带一个NSString参数的构造函数。在其中初始化成员变量。

其中enc是 NSStringEncoding 类型,因为服务器返回的字符中使用了中文,所以我们通过它指定了一个gb2312的字符编码。

许多资料中说,需要在NSOperation中重载一个叫做isConcurrent的函数并在其中返回YES,否则不支持异步执行。但是实际上,我们在这里注释了这个重载方法,程序也没有报任何错误,其执行方式依然是异步的。

  1. @implementation URLOperation  
  2. @synthesize data=_data;  
  3. - (id)initWithURLString:(NSString *)url {  
  4.     if (self = [self init]) {  
  5.         NSLog(@"%@",url);  
  6.         _request = [[NSURLRequest alloc] initWithURL:[NSURL URLWithString:url  
  7.         //构建gb2312的encoding  
  8.         enc =CFStringConvertEncodingToNSStringEncoding(kCFStringEncodingGB_18030_2000);  
  9.         _data = [[NSMutableData data] retain];  
  10.     }  
  11.     return self;  
  12. }  
  13. - (void)dealloc {  
  14.     [_request release],_request=nil;  
  15.     [_data release],_data=nil;  
  16.     [_connection release],_connection=nil;  
  17.     [super dealloc];  
  18. }  
  19. // 如果不重载下面的函数,异步方式调用会出错  
  20. //- (BOOL)isConcurrent {  //如果采用并发,就不要使用queue,add5次后会发生运行错误
  21. //  return YES;//返回yes表示支持异步调用,否则为支持同步调用  
  22. //} 

整个类中最重要的方法是start方法。Start是NSOperation类的主方法,主方法的叫法充分说明了其重要性,因为这个方法执行完后,该NSOperation的执行线程就结束了(返回调用者的主线程),同时对象实例就会被释放,也就意味着你定义的其他代码(包括delegate方法)也不会被执行。很多资料中的start方法都只有最简单的一句(包括“易飞扬的博客 “的博文):

  1. [NSURLConnection connectionWithRequest:_request delegate:self]; 

如果这样的话,delegate方法没有执行机会。因为start方法结束后delegate(即self对象)已经被释放了,delegate的方法也就无从执行。

所以在上面的代码中,还有一个while循环,这个while循环的退出条件是http连接终止(即请求结束)。当循环结束,我们的工作也就完成了。

  1. // 开始处理-本类的主方法  
  2. - (void)start {  
  3.     if (![self isCancelled]) {  
  4.         NSLog(@"start operation");  
  5.         // 以异步方式处理事件,并设置代理  
  6.         _connection=[[NSURLConnection connectionWithRequest:_request delegate:self]retain];  
  7.         //下面建立一个循环直到连接终止,使线程不离开主方法,否则connection的delegate方法不会被调用,因为主方法结束对象的生命周期即终止  
  8.         //这个问题参考 http://www.cocoabuilder.com/archive/cocoa/279826-nsurlrequest-and-nsoperationqueue.html  
  9.         while(_connection != nil) {  
  10.             [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];     
  11.         }  
  12.     }  

接下来,是NSURLConnection的delegate方法,这部分的代码和大部分资料的介绍是一样的,你可以实现全部的delegate方法,但这里我们只实现其中3个就足够了,其余的方法不用理会。如你所见,你可以在其中添加自己想到的任何代码,包括接收数据,进行字符编码或者做xml解析。

  1. #pragma mark NSURLConnection delegate Method  
  2. // 接收到数据(增量)时  
  3. - (void)connection:(NSURLConnection*)connection  
  4.     didReceiveData:(NSData*)data {  
  5.     NSLog(@"connection:");  
  6.     NSLog(@"%@",[[NSString alloc] initWithData:data encoding:enc]);  
  7.     // 添加数据  
  8.  
  9.     [_data appendData:data];  
  10.  
  11. }  
  12. // HTTP请求结束时  
  13. - (void)connectionDidFinishLoading:(NSURLConnection*)connection {  
  14.     [_connection release],_connection=nil;  
  15.     //NSLog(@"%@",[[NSString alloc] initWithData:_data encoding:enc]);  
  16. }  
  17. -(void)connection: (NSURLConnection *) connection didFailWithError: (NSError *) error{  
  18.     NSLog(@"connection error");  
  19. }  
  20. @end 

到此,虽然代码还没有完成,但我们已经可以运行它了。你可以看到console输出的内容,观察程序的运行状态。

2、调用NSOperation

我们的NSOperation类可以在ViewController中调用,也可以直接放在AppDelegate中进行。

在这里,我是通过点击按钮来触发调用代码的:

  1. -(void)loginClicked{  
  2.     //构造登录请求url  
  3.     NSString* url=@”http://google.com”;  
  4.     _queue = [[NSOperationQueue alloc] init];  
  5.     URLOperation* operation=[[URLOperation alloc ]initWithURLString:url];  
  6.     // 开始处理  
  7.     [_queue addOperation:operation];  
  8.     [operation release];//队列已对其retain,可以进行release;  

_queue是一个 NSOperationQueue 对象,当往其中添加 NSOperation 对象后, NSOperation 线程会被自动执行(不是立即执行,根据调度情况)。

3、KVO编程模型

我们的NSOperation完成了向服务器的请求并将服务器数据下载到成员变量_data中了。现在的问题是,由于这一切是通过异步操作进行的,我们无法取得_data中的数据,因为我们不知道什么时候异步操作完成,以便去访问_data属性(假设我们将_data定义为属性了),取得服务器数据。

我们需要一种机制,当NSOperation完成所有工作之后,通知调用线程。

这里我们想到了KVO编程模型(键-值观察模型)。这是cocoa绑定技术中使用的一种设计模式,它可以使一个对象在属性值发生变化时主动通知另一个对象并触发相应的方法。

首先,我们在NSOperation的子类中添加一个BOOL变量,当这个变量变为YES时,标志异步操作已经完成:

  1. BOOL _isFinished; 

在实现中加入这个变量的访问方法:

  1. - (BOOL)isFinished  
  2. {  
  3.     return _isFinished;  

cocoa的KVO模型中,有两种通知观察者的方式,自动通知和手动通知。顾名思义,自动通知由cocoa在属性值变化时自动通知观察者,而手动通知需要在值变化时调用 willChangeValueForKey:和didChangeValueForKey: 方法通知调用者。为求简便,我们一般使用自动通知。

要使用自动通知,需要在 automaticallyNotifiesObserversForKey方法中明确告诉cocoa,哪些键值要使用自动通知:

  1. //重新实现NSObject类中的automaticallyNotifiesObserversForKey:方法,返回yes表示自动通知。  
  2. + (BOOL)automaticallyNotifiesObserversForKey:(NSString*)key  
  3. {  
  4.     //当这两个值改变时,使用自动通知已注册过的观察者,观察者需要实现observeValueForKeyPath:ofObject:change:context:方法  
  5.     if ([key isEqualToString:@"isFinished"])  
  6.     {  
  7.         return YES;  
  8.     }  
  9.     return [super automaticallyNotifiesObserversForKey:key];  

然后,在需要改变_isFinished变量的地方,使用

  1. [self setValue:[NSNumber numberWithBool:YES] forKey:@"isFinished"]; 

方法,而不是仅仅使用简单赋值。

我们需要在3个地方改变isFinished值为YES,请求结束时、连接出错误,线程被cancel。请在对应的方法代码中加入上面的语句。

最后,需要在观察者的代码中进行注册。打开ViewController中调用NSOperation子类的地方,加入:

  1.     //kvo注册  
  2.     [operation addObserver:self forKeyPath:@"isFinished"  
  3.                    options:(NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld) context:operation];  
  4. 并实现 observeValueForKeyPath 方法:  
  5. //接收变更通知  
  6. - (void)observeValueForKeyPath:(NSString *)keyPath  
  7.                       ofObject:(id)object  
  8.                        change:(NSDictionary *)change  
  9.                        context:(void *)context  
  10. {  
  11.     if ([keyPath isEqual:@"isFinished"]) {  
  12.         BOOL isFinished=[[change objectForKey:NSKeyValueChangeNewKey] intValue];  
  13.         if (isFinished) {//如果服务器数据接收完毕  
  14.             [indicatorView stopAnimating];  
  15.             URLOperation* ctx=(URLOperation*)context;  
  16.             NSStringEncoding enc=CFStringConvertEncodingToNSStringEncoding(kCFStringEncodingGB_18030_2000);  
  17.             NSLog(@"%@",[[NSString alloc] initWithData:[ctx data] encoding:enc]);  
  18.             //取消kvo注册  
  19.             [ctx removeObserver:self  
  20.                     forKeyPath:@"isFinished"];  
  21.         }        
  22.     }else{  
  23.         // be sure to call the super implementation  
  24.         // if the superclass implements it  
  25.         [super observeValueForKeyPath:keyPath  
  26.                              ofObject:object  
  27.                                change:change  
  28.                               context:context];  
  29.     }  

运行程序,查看控制台的输出。

 

转载于:https://www.cnblogs.com/pengyingh/articles/2355273.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/464028.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

UDT源码剖析(四):UDT的GC线程相关过程代码注释

着UDT::Startup()的启动,GC线程也随之运行了。 GC主要关注的就是UDTSocket的释放,下面我们来看看这个GC线程是怎么实现的。 查看源代码 打印帮助1 #ifndef WIN32 2 void* CUDTUnited::garbageCollect(void* p) 3 #else 4 DWORD WINAPI CUDTUnited:…

那年高考

每年高考,都会让我想起很多事情。我第一次高考前几天,我舅舅和舅娘过来看我,他们拉我到教室后门,我舅说我妈叮嘱他一定来看我,然后跟我说不要给自己太大压力,啰嗦了几分钟,要走的时候硬塞给我10…

vimdiff的简单使用

Vimdiff的使用 1、首先vimdiff是一款高效的文件的比较和合并的工具,它具有以下一些特点:命令行方式的比较工具;简单明了的界面;对比较出来的多处差异之间快速定位;进行文件合并。再使用vimdiff命令之前,我们…

拆解一个比亚迪的车钥匙,这个黑黑的元器件是什么?

今天抽屉突然找到一个比亚迪的车钥匙,寻思着没事拆开看看用什么方案做的。正面照,哪款车型的,不知道了背面照,很简洁,只有一个BYD的Logo接下来就是撬开了,过程非常顺利,之前撬开一个宝马车的钥匙…

分布式事物(同样适用于dubbo事务等分布式事务)

转载于:https://www.cnblogs.com/tian1993/p/10081901.html

一些web开发中常用的、做成cs文件的js代码 - 搜刮来的

using System;using System.Web;using System.Web.UI;namespace COCOWO.COMP{ /// <summary> /// 一些常用的Js调用 /// 创建时间&#xff1a;2006-8-3 /// 创建者&#xff1a;马先光 /// </summary> public class Jscript { …

库里扛起了勇士对凯尔特人的第四场

最近临近毕业季&#xff0c;有朋友这样说「我一定是要支持勇士的&#xff0c;毕竟库里和詹姆斯陪了整个大学四年啊」。昨晚上加班到11点半&#xff0c;到家洗完澡已经是凌晨1点了&#xff0c;那时候是想得很清楚的&#xff0c;明天是不想被其他事情打扰我看球赛的。这场比赛&am…

Nginx应用案例分享:压力测试

2019独角兽企业重金招聘Python工程师标准>>> 在运维工作中&#xff0c;压力测试是一项非常重要的工作。比如在一个网站上线之前&#xff0c;能承受多大访问量、在大访问量情况下性能怎样&#xff0c;这些数据指标好坏将会直接影响用户体验。 但是&#xff0c;在压力…

shell 判断文件是否存在,没有则创建

没有该文件则创建&#xff0c;有则 ls -l 输出文件信息。 #!/bin/bash echo "enter the name:" read filename if test -e $filename ; then ls -l $filename else touch $filename fi 输出 enter the name: sss.sh -rwxr-xr-x 1 root root 147 Dec 3 09:02 sss.sh …

声音是什么?

转自博客园《【秒懂音视频开发】02_重识声音》&#xff0c;https://www.cnblogs.com/mjios/p/14466418.html#5035688#!comments声音的产生声音&#xff08;Sound&#xff09;&#xff0c;是由物体的振动产生的。一切正在发声的物体都在振动。我们说话的时候&#xff0c;是声带在…

c#中的常用ToString()方法总结。

很多类都重写了ToString方法&#xff0c; 导致很多类的tostring到底执行了什么&#xff0c;有哪些参数&#xff0c;都不清楚。。。。很郁闷&#xff01;对于int&#xff0c;double等的tostring&#xff1a;C 货币 2.5.ToString("C") &#xffe5;2.50…

UDT源码剖析(五):UDT::cleanup()过程代码注释

调用路线 UDT::cleanup()->CUDT::cleanup()->CUDTUnited::cleanup() 1 int CUDTUnited::cleanup() 2 { 3 // 锁保护 4 CGuard gcinit(m_InitLock); 5 6 // 如果还有实例运行&#xff0c;则到此为止了 7 if (--m_iInstanceCount > 0) 8 return 0; …

《星辰傀儡线》人物续:“灭世者”、“疯狂者”、“叛逆者”三兄妹

“灭世者”、“疯狂者”、“叛逆者”三兄妹&#xff1a;“灭世者”黑卡蒂&#xff1a;卡西欧家族的科技天才&#xff0c;沉默&#xff0c;不爱说话&#xff0c;一对亮闪闪的氪金眼&#xff0c;黑暗军团一战后&#xff0c;以“天空之镜”的设计者而闻名。柏拉图“理想国”的拥有…

[源码和文档分享]基于C语言的语法高亮设计与实现

一 需求分析 在所需高亮的关键字或字符串前后加上class标签&#xff0c;在css定义颜色。 二 程序设计 2.1 设计思路 把.html文件和.css文件中的内容存在两个字符数组中&#xff0c;在.cpp用文件操作写入。 2.2 文件组织架构 Syntax highlighting.exe所在目录为根目录 源代码命…

在.NET中用excel导出(概要)

首先要在项目->添加引用->COM中添加Microsoft Office Web Components &#xff0c;然后定义一个OWC.SpreadsheetClass 如&#xff1a;Dim xlsheet As New OWC.SpreadsheetClass 然后就可以向sheet中添加数据了&#xff0c;使用xlsheet.Cells(2, 2) "表格填充内容&q…

我愿为这二极管奉献我的一生

简 介&#xff1a; 利用二极管完成对信号的整流&#xff0c;求取绝对值等具有很多的应用。然而在小信号下&#xff0c;二极管的前向导通电压以及相应的杂散电容会对信号整流带来严重的影响。本文从“马场清太郎”所著的“运算放大器应用电路设计”中摘取了一些典型应用电路&…

61条Java面向对象设计的经验原则

1&#xff09;所有数据都应该隐藏在所在的类的内部。 &#xff08;2&#xff09;类的使用者必须依赖类的共有接口&#xff0c;但类不能依赖它的使用者。 &#xff08;3&#xff09;尽量减少类的协议中的消息。 &#xff08;4&#xff09;实现所有类都理解的最基本公…

问题?

1、java类空参构造器的作用&#xff1f; 构建对象时候所用的空参构造函数 2、static代码块与非static代码块的区别&#xff1a; 静态代码块&#xff0c;在虚拟机加载类的时候就会加载执行&#xff0c;而且只执行一次&#xff1b;非静态代码块&#xff0c;在创建对象的时候&…

【毕业设计】履带四足复合机器人

1.项目简介本项目中&#xff0c;履足式复合机器人拥有两种行进模式&#xff1a;四足行进模式、履带行进模式&#xff0c;应对不同场景及地形时能够切换形态。四足形态下整体总共拥有8个自由度DOF&#xff0c;单腿各具备2个自由度DOF&#xff0c;足端结构融入了履带机构&#xf…

UDT源码剖析(六):UDT::socket()过程代码注释

调用路线 UDT::socket()->CUDT::socket()->CUDTUnited::newSocket() 1 UDTSOCKET CUDT::socket(int af, int type, int) 2 { 3 // 如果垃圾收集没启动&#xff0c;那么这里自动调用startup 4 // 所以手动调用startup其实不是必须的&#xff0c;但是为了接受作者的风…