AFNetWorking2.0&WebView

AFNNetworking 2.0你相信你一定知道AFNNetworking,不知道你还可以看看该作者的博文,所以我就不多说关于它的强大之处了,AFNetworking 提供了比NSURLSession更高层次的抽象,这篇文章主要总结AFNNetworking 2.0的几个常用方法。1).GET和POST请求GET:- (NSURLSessionDataTask *)GET:(NSString *)URLString                  parameters:(id)parameters                      success:(void (^)(NSURLSessionDataTask *task, id responseObject))success                      failure:(void (^)(NSURLSessionDataTask *task, NSError *error))failurePOST:- (NSURLSessionDataTask *)POST:(NSString *)URLString                    parameters:(id)parameters                      success:(void (^)(NSURLSessionDataTask *task, id responseObject))success                      failure:(void (^)(NSURLSessionDataTask *task, NSError *error))failure在实际的开发中,我比较习惯把第三方的代码进行一层封装,如下我们把GET和POST请求封装起来://GET+ (void)getWithPath:(NSString *)path params:(NSDictionary *)params success:(HttpSuccessBlock)success failure:(HttpFailureBlock)failure{    [self requestWithPath:path params:params success:success failure:failure method:@"GET"];}//POST+ (void)postWithPath:(NSString *)path params:(NSDictionary *)params success:(HttpSuccessBlock)success failure:(HttpFailureBlock)failure{    [self requestWithPath:path params:params success:success failure:failure method:@"POST"];}typedef void (^HttpSuccessBlock)(id JSON);typedef void (^HttpFailureBlock)(NSError *error);...省略+ (void)requestWithPath:(NSString *)path params:(NSDictionary *)params success:(HttpSuccessBlock)success failure:(HttpFailureBlock)failure method:(NSString *)method{    AFHTTPSessionManager *manager = [[AFHTTPSessionManager alloc] initWithBaseURL:[NSURL URLWithString:kBaseURL]];    manager.responseSerializer = [AFJSONResponseSerializer serializer];    manager.responseSerializer.acceptableContentTypes = [NSSet setWithObjects:@"text/plain",@"application/json", @"text/json", @"text/javascript", @"text/html", nil];    if ([method  isEqual: @"GET"]) {        [manager GET:path parameters:params success:^(NSURLSessionDataTask *task, id responseObject) {            success(responseObject);        } failure:^(NSURLSessionDataTask *task, NSError *error) {            NSLog(@"fail Get!!%@",error);            failure(error);        }];    }else if ([method isEqual:@"POST"]){        [manager POST:path parameters:params success:^(NSURLSessionDataTask *task, id responseObject) {            NSLog(@"POST成功:%@",responseObject);            success(responseObject);        } failure:^(NSURLSessionDataTask *task, NSError *error) {            NSLog(@"fail POST!!%@",error);            failure(error);        }];    }}你有没有注意到上面我设置了如下代码:manager.responseSerializer = [AFJSONResponseSerializer serializer];manager.responseSerializer.acceptableContentTypes = [NSSet setWithObjects:@"text/plain",@"application/json", @"text/json", @"text/javascript", @"text/html", nil];因为AFNetworking的responseSerializer 属性只能默认只能处理特定的Content-Type,如果你想处理"text/html"等其它类型,你需要明确指定它的acceptableContentTypes。2).多文件上传如下,这是我写的第三方微博客户端中多图上传的实例代码:[manager POST:@"2/statuses/upload.json"          parameters:@{@"access_token": accseeToken,                          @"status" : encodeStatus,                          @"visible" : @(_intVisible),                          @"lat" : @(_latitude),                          @"long" : @(_longtitude)}            constructingBodyWithBlock:^(idformData) {                NSMutableArray *images = [NSMutableArray arrayWithArray:weakSelf.images];                for (id asset in images) {                    NSData *data = nil;                    if ([asset isKindOfClass:[UIImage class]]) {                        data = UIImageJPEGRepresentation(asset, 0.4);                    }                    if ([asset isKindOfClass:ALAsset.class]) {                        UIImage *original = [UIImage imageWithCGImage: [[asset defaultRepresentation] fullScreenImage]];                        data = UIImageJPEGRepresentation(original, 0.4);                    }                    [formData appendPartWithFileData:data name:@"pic" fileName:@"pic.jpg" mimeType:@"multipart/form-data"];                }            } success:^(NSURLSessionDataTask *task, id responseObject) {                NSLog(@"发送成功");                [self back];            } failure:^(NSURLSessionDataTask *task, NSError *error) {                [self showFailHUD];            }];3).多线程操作如果你需要开启多个线程, 你需要使用AFHTTPRequestSerializer,AFHTTPRequestOperation和NSOperationQueue以下是AFNetworking的实例代码NSMutableArray *mutableOperations = [NSMutableArray array];for (NSURL *fileURL in filesToUpload) {    NSURLRequest *request = [[AFHTTPRequestSerializer serializer] multipartFormRequestWithMethod:@"POST" URLString:@"http://example.com/upload" parameters:nil constructingBodyWithBlock:^(idformData) {

[formData appendPartWithFileURL:fileURL name:@"images[]" error:nil];

}];

AFHTTPRequestOperation *operation = [[AFHTTPRequestOperation alloc] initWithRequest:request];

[mutableOperations addObject:operation];

}

NSArray *operations = [AFURLConnectionOperation batchOfRequestOperations:@[...] progressBlock:^(NSUInteger numberOfFinishedOperations, NSUInteger totalNumberOfOperations) {

NSLog(@"%lu of %lu complete", numberOfFinishedOperations, totalNumberOfOperations);

} completionBlock:^(NSArray *operations) {

NSLog(@"All operations in batch complete");

}];

[[NSOperationQueue mainQueue] addOperations:operations waitUntilFinished:NO];

3).网络状态检查

网络状态检查在早期都是通过苹果官方的Reachability类进行检查,但是这个类本身存在一些问题,并且官方后来没有再更新。我们可以直接使用AFNetworking框架检测。不管使用官方提供的类还是第三方框架,用法都是类似的,通常是发送一个URL然后去检测网络状态变化,网络改变后则调用相应的网络状态改变方法。如下:

-(void)alert:(NSString *)message{

UIAlertView *alertView=[[UIAlertView alloc]initWithTitle:@"System Info" message:message delegate:nil cancelButtonTitle:@"Cancel" otherButtonTitles: nil];

[alertView show];

}

-(void)checkNetworkStatus{

//创建一个用于测试的url

NSURL *url=[NSURL URLWithString:@"http://www.apple.com"];

AFHTTPRequestOperationManager *operationManager=[[AFHTTPRequestOperationManager alloc]initWithBaseURL:url];

//根据不同的网络状态改变去做相应处理

[operationManager.reachabilityManager setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) {

switch (status) {

case AFNetworkReachabilityStatusReachableViaWWAN:

[self alert:@"2G/3G/4G Connection."];

break;

case AFNetworkReachabilityStatusReachableViaWiFi:

[self alert:@"WiFi Connection."];

break;

case AFNetworkReachabilityStatusNotReachable:

[self alert:@"Network not found."];

break;

default:

[self alert:@"Unknown."];

break;

}

}];

//开始监控

[operationManager.reachabilityManager startMonitoring];

}

2.UIWebView

UIWebView不仅能加载网络资源还可以加载本地资源,目前支持的常用的文档格式如:html、pdf、docx、txt等。

UIWebView整个使用相当简单:创建URL->创建请求->加载请求,无论是加载本地文件还是Web内容都是这三个步骤。UIWebView内容加载事件同样是通过代理通知外界,常用的代理方法如开始加载、加载完成、加载出错等,这些方法通常可以帮助开发者更好的控制请求加载过程。

加载资源:

- (void)loadRequest:(NSURLRequest *)request;

常用的属性和方法:

//重新加载(刷新)

- (void)reload;

//停⽌止加载

- (void)stopLoading;

//回退

- (void)goBack;

//前进

- (void)goForward;

//需要进⾏检测的数据类型

@property(nonatomic) UIDataDetectorTypes dataDetectorTypes

//是否能回退

@property(nonatomic,readonly,getter=canGoBack) BOOL canGoBack;

//是否能前进

@property(nonatomic,readonly,getter=canGoForward) BOOL canGoForward;

//是否正在加载中

@property(nonatomic,readonly,getter=isLoading) BOOL loading;

//是否伸缩内容至适应屏幕当前尺寸

@property(nonatomic) BOOL scalesPageToFit;

遵守UIWebViewDelegate协议,监听UIWebView的加载过程:

//开始发送请求(加载数据)时调用:

- (void)webViewDidStartLoad:(UIWebView *)webView;

//请求完毕(加载数据完毕)时调⽤:

- (void)webViewDidFinishLoad:(UIWebView *)webView;

//请求错误时调用:

- (void)webView:(UIWebView *)webView didFailLoadWithError:(NSError *)error;

//监听UIWebView的加载过程:

- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType;

下面是一个例子:

在storyBoard中拖入如下控件:

searchBar的代理方法:

-(void)searchBarSearchButtonClicked:(UISearchBar *)searchBar{

[self request:_searchBar.text];

[searchBar resignFirstResponder];

}

加载searchBar中的请求:

-(void)request:(NSString *)urlStr{

//创建url

NSURL *url;

//如果file://开头的字符串则加载bundle中的文件

if([urlStr hasPrefix:kFILEPROTOCOL]){

//取得文件名

NSRange range= [urlStr rangeOfString:kFILEPROTOCOL];

NSString *fileName=[urlStr substringFromIndex:range.length];

url=[[NSBundle mainBundle] URLForResource:fileName withExtension:nil];

}else if(urlStr.length>0){

//如果是http请求则直接打开网站

if ([urlStr hasPrefix:@"http"]) {

url=[NSURL URLWithString:urlStr];

}else{//如果不符合任何协议则进行搜索

urlStr=[NSString stringWithFormat:@"http://m.bing.com/search?q=%@",urlStr];

}

urlStr=[urlStr stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];//url编码

url=[NSURL URLWithString:urlStr];

}

//创建请求

NSURLRequest *request=[NSURLRequest requestWithURL:url];

//加载请求页面

[_webView loadRequest:request];

}

WebView的代理方法:

-(void)webViewDidStartLoad:(UIWebView *)webView{

//显示网络请求加载

[UIApplication sharedApplication].networkActivityIndicatorVisible=true;

}

-(void)webViewDidFinishLoad:(UIWebView *)webView{

//隐藏网络请求加载图标

[UIApplication sharedApplication].networkActivityIndicatorVisible=false;

//设置按钮状态

[self setBarButtonStatus];

}

-(void)webView:(UIWebView *)webView didFailLoadWithError:(NSError *)error{

NSLog(@"error detail:%@",error.localizedDescription);

UIAlertView *alert=[[UIAlertView alloc]initWithTitle:@"系统提示" message:@"网络连接发生错误!" delegate:self cancelButtonTitle:nil otherButtonTitles:@"确定", nil];

[alert show];

}

设置前进后退按钮:

-(void)setBarButtonStatus{

if (_webView.canGoBack) {

_barButtonBack.enabled=YES;

}else{

_barButtonBack.enabled=NO;

}

if(_webView.canGoForward){

_barButtonForward.enabled=YES;

}else{

_barButtonForward.enabled=NO;

}

}

运行结果如下:

你可以在这里下载到代码。

- (void)getAccessToken:(NSString *)requestToken

{

[HttpTool postWithPath:@"oauth2/access_token" params:@{

@"client_id" : kAppKey,

@"client_secret" : kAppSecret,

@"grant_type" : @"authorization_code",

@"redirect_uri" : kRedirectURI,

@"code" : requestToken

} success:^(id JSON) {

// 保存账号信息

Account *account = [[Account alloc] init];

account.accessToken = JSON[@"access_token"];

account.uid = JSON[@"uid"];

[[AccountTool sharedAccountTool] saveAccount:account];

// 回到主页面

ViewController *main = [[ViewController alloc]init];

if (main) {

[self presentViewController:main animated:YES completion:nil];

}

} failure:^(NSError *error) {

}];

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,911评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,014评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 142,129评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,283评论 1 264
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,159评论 4 357
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,161评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,565评论 3 382
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,251评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,531评论 1 292
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,619评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,383评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,255评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,624评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,916评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,199评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,553评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,756评论 2 335

推荐阅读更多精彩内容

  • iOS开发系列--网络开发 概览 大部分应用程序都或多或少会牵扯到网络开发,例如说新浪微博、微信等,这些应用本身可...
    lichengjin阅读 3,619评论 2 7
  • 导语 在上家公司,网络请求一直是AFNetworking2.0,现在该升级了!话不多说,直接开始咱们自己的WebR...
    欢欢1206阅读 2,318评论 4 31
  • //需要AFN //.h //AFNetworking + (void)post:(NSString *)url ...
    CHADHEA阅读 742评论 0 0
  • 1.在开发的时候可以创建一个工具类,继承自我们的AFN中的请求管理者,再控制器中真正发请求的代码使用自己封装的工具...
    红楼那一场梦阅读 3,457评论 2 3
  • 文丨张宗超 与读者书:人生没有标准答案,比我说的更重要的是你的思考。 2013年以来,每年的这几天我都特别的忙,其...
    宗超想对你说阅读 310评论 0 0