#import "ViewController.h"

@interface ViewController ()<NSURLConnectionDataDelegate>

@end

@implementation ViewController

-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
[self newThreadDelegate2];
} -(void)delegate1
{
NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://120.25.226.186:32812/login?username=123&pwd=123&type=JSON"]]; //设置代理
//代理方法:默认是在主线程中调用的
NSURLConnection *connect = [NSURLConnection connectionWithRequest:request delegate:self]; //设置代理方法在哪个线程中调用
//[NSOperationQueue alloc]init]] 开子线程
//[NSOperationQueue mainQueue] 不能这样设置
[connect setDelegateQueue:[[NSOperationQueue alloc]init]];
//[connect setDelegateQueue:[NSOperationQueue mainQueue]]; NSLog(@"-------");
} -(void)delegate2
{
NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://120.25.226.186:32812/login?username=123&pwd=123&type=JSON"]]; //设置代理
//代理方法:默认是在主线程中调用的
NSURLConnection *connect = [[NSURLConnection alloc]initWithRequest:request delegate:self startImmediately:NO]; [connect setDelegateQueue:[[NSOperationQueue alloc]init]]; //开始发送请求
[connect start];
NSLog(@"-------");
} -(void)newThreadDelegate1
{
dispatch_async(dispatch_get_global_queue(, ), ^{ NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://120.25.226.186:32812/login?username=123&pwd=123&type=JSON"]]; //设置代理
//代理方法:默认是在主线程中调用的
//该方法内部其实会将connect对象作为一个source添加到当前的runloop中,指定运行模式为默认
NSURLConnection *connect = [NSURLConnection connectionWithRequest:request delegate:self]; //设置代理方法在哪个线程中调用
[connect setDelegateQueue:[[NSOperationQueue alloc]init]]; //[[NSRunLoop currentRunLoop] runMode:UITrackingRunLoopMode beforeDate:[NSDate dateWithTimeIntervalSinceNow:1000]];
[[NSRunLoop currentRunLoop]run]; NSLog(@"---%@----",[NSThread currentThread]);
}); } -(void)newThreadDelegate2
{
dispatch_async(dispatch_get_global_queue(, ), ^{
NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://120.25.226.186:32812/login?username=123&pwd=123&type=JSON"]]; //设置代理
//代理方法:默认是在主线程中调用的
NSURLConnection *connect = [[NSURLConnection alloc]initWithRequest:request delegate:self startImmediately:NO]; [connect setDelegateQueue:[[NSOperationQueue alloc]init]]; //开始发送请求
//如如果connect对象没有添加到runloop中,那么该方法内部会自动的添加到runloop
//注意:如果当前的runloop没有开启,那么该方法内部会自动获得当前线程对应的runloop对象并且开启
[connect start];
NSLog(@"---%@----",[NSThread currentThread]);
});
} #pragma mark ----------------------
#pragma mark NSURLConnectionDataDelegate
-(void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
{
NSLog(@"didReceiveResponse---%@",[NSThread currentThread]);
} @end

#####1 NSURLConnection和Runloop(面试)

(1)两种为NSURLConnection设置代理方式的区别

```objc

//第一种设置方式:

//通过该方法设置代理,会自动的发送请求

// [[NSURLConnection alloc]initWithRequest:request delegate:self];

//第二种设置方式:

//设置代理,startImmediately为NO的时候,该方法不会自动发送请求

NSURLConnection *connect = [[NSURLConnection alloc]initWithRequest:request delegate:self startImmediately:NO];

//手动通过代码的方式来发送请求

//注意该方法内部会自动的把connect添加到当前线程的RunLoop中在默认模式下执行

[connect start];

```

(2)如何控制代理方法在哪个线程调用

```objc

//说明:默认情况下,代理方法会在主线程中进行调用(为了方便开发者拿到数据后处理一些刷新UI的操作不需要考虑到线程间通信)

//设置代理方法的执行队列

[connect setDelegateQueue:[[NSOperationQueue alloc]init]];

```

(3)开子线程发送网络请求的注意点,适用于自动发送网络请求模式

```objc

//在子线程中发送网络请求-调用startf方法发送

-(void)createNewThreadSendConnect1

{

//1.创建一个非主队列

NSOperationQueue *queue = [[NSOperationQueue alloc]init];

//2.封装操作,并把任务添加到队列中执行

[queue addOperationWithBlock:^{

NSLog(@"%@",[NSThread currentThread]);

//2-1.确定请求路径

NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/login?username=dd&pwd=ww&type=JSON"];

//2-2.创建请求对象

NSURLRequest *request = [NSURLRequest requestWithURL:url];

//2-3.使用NSURLConnection设置代理,发送网络请求

NSURLConnection *connection = [[NSURLConnection alloc]initWithRequest:request delegate:self startImmediately:YES];

//2-4.设置代理方法在哪个队列中执行,如果是非主队列,那么代理方法将再子线程中执行

[connection setDelegateQueue:[[NSOperationQueue alloc]init]];

//2-5.发送网络请求

//注意:start方法内部会把当前的connect对象作为一个source添加到当前线程对应的runloop中

//区别在于,如果调用start方法开发送网络请求,那么再添加source的过程中,如果当前runloop不存在

//那么该方法内部会自动创建一个当前线程对应的runloop,并启动。

[connection start];

}];

}

//在子线程中发送网络请求-自动发送网络请求

-(void)createNewThreadSendConnect2

{

NSLog(@"-----");

//1.创建一个非主队列

NSOperationQueue *queue = [[NSOperationQueue alloc]init];

//2.封装操作,并把任务添加到队列中执行

[queue addOperationWithBlock:^{

//2-1.确定请求路径

NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/login?username=dd&pwd=ww&type=JSON"];

//2-2.创建请求对象

NSURLRequest *request = [NSURLRequest requestWithURL:url];

//2-3.使用NSURLConnection设置代理,发送网络请求

//注意:该方法内部虽然会把connection添加到runloop,但是如果当前的runloop不存在,那么不会主动创建。

NSURLConnection *connection = [NSURLConnection connectionWithRequest:request delegate:self];

//2-4.设置代理方法在哪个队列中执行,如果是非主队列,那么代理方法将再子线程中执行

[connection setDelegateQueue:[[NSOperationQueue alloc]init]];

//2-5 创建当前线程对应的runloop,并开启

[[NSRunLoop currentRunLoop]run];

}];

}

```

总结:也就是说发送网络请求在设置回调的队列的时候,回调的任务默认在主线程,若是设置回调队列在子线程,则毁掉成功后不会调用代理方法,解决办法:1:调用start方法: start方法内部会把当前的connect对象作为一个source添加到当前线程对应的runloop中,

如果调用start方法开发送网络请求,那么再添加source的过程中,如果当前runloop不存在那么该方法内部会自动创建一个当前线程对应的runloop,并启动。每条线程都必须有自己的runloop来处理事件   2:若是不调用start方法,则需要自己创建runloop并调用run方法,

  dispatch_async(dispatch_get_global_queue(0, 0), ^{

       NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://120.25.226.186:32812/login?username=123&pwd=123&type=JSON"]];

       //设置代理
//代理方法:默认是在主线程中调用的
//该方法内部其实会将connect对象作为一个source添加到当前的runloop中,指定运行模式为默认
NSURLConnection *connect = [NSURLConnection connectionWithRequest:request delegate:self]; //设置代理方法在哪个线程中调用
[connect setDelegateQueue:[[NSOperationQueue alloc]init]]; //[[NSRunLoop currentRunLoop] runMode:UITrackingRunLoopMode beforeDate:[NSDate dateWithTimeIntervalSinceNow:1000]];
[[NSRunLoop currentRunLoop]run]; NSLog(@"---%@----",[NSThread currentThread]);
}); }

二者的区别就在于:1:调用start如果调用start方法开发送网络请求,那么再添加source的过程中,如果当前runloop不存在那么该方法内部会自动创建一个当前线程对应的runloop,并启动 2:而下面的方法虽自动发送请求,也会

将connect对象作为一个source添加到当前的runloop中,指定运行模式为默认,但是如果当前子线程的runloop不存在则不会自动创建需要手动创建
 NSURLConnection *connect = [NSURLConnection connectionWithRequest:request delegate:self];

ios开发网络学习六:设置队列请求与RunLoop的更多相关文章

  1. ios开发网络学习:一:NSURLConnection发送GET,POST请求

    #import "ViewController.h" @interface ViewController ()<NSURLConnectionDataDelegate> ...

  2. iOS开发——网络篇——HTTP/NSURLConnection(请求、响应)、http响应状态码大全

    一.网络基础 1.基本概念> 为什么要学习网络编程在移动互联网时代,移动应用的特征有几乎所有应用都需要用到网络,比如QQ.微博.网易新闻.优酷.百度地图只有通过网络跟外界进行数据交互.数据更新, ...

  3. ios开发网络学习十二:NSURLSession实现文件上传

    #import "ViewController.h" // ----WebKitFormBoundaryvMI3CAV0sGUtL8tr #define Kboundary @&q ...

  4. ios开发网络学习五:输出流以及文件上传

    一:输出流 #import "ViewController.h" @interface ViewController ()<NSURLConnectionDataDelega ...

  5. ios开发网络学习四:NSURLConnection大文件断点下载

    #import "ViewController.h" @interface ViewController ()<NSURLConnectionDataDelegate> ...

  6. ios开发网络学习AFN框架的使用一:get和post请求

    #import "ViewController.h" #import "AFNetworking.h" @interface ViewController () ...

  7. iOS开发网络学习七:NSURLSession的基本使用get和post请求

    #import "ViewController.h" @interface ViewController () @end @implementation ViewControlle ...

  8. ios开发网络学习十一:NSURLSessionDataTask离线断点下载(断点续传)

    #import "ViewController.h" #define FileName @"121212.mp4" @interface ViewControl ...

  9. ios开发网络学习九:NSURLSessionDownloadTask实现大文件下载

    一:NSURLSessionDownloadTask:实现文件下载:无法监听进度 #import "ViewController.h" @interface ViewControl ...

随机推荐

  1. iOS报错 -pie can only be used when targeting iOS 4.2 or later

    近期,使用师兄的project时.突然报错之前没发现这个错误.信息例如以下: ld: -pie can only be used when targeting iOS 4.2 or later cla ...

  2. listview-fading 滚动条样式设置

    fadingEdge-属性用来设置拉滚动条时 ,边框渐变的方向.它有三个属性值可以设置 none:(边框颜色不变) horizontal:(水平方向颜色变淡) vertical:(垂直方向颜色变淡). ...

  3. CF #261 div2 D. Pashmak and Parmida&#39;s problem (树状数组版)

    Parmida is a clever girl and she wants to participate in Olympiads this year. Of course she wants he ...

  4. 116.C语言异常抛错

    #include <stdlib.h> #include <stdio.h> #include <setjmp.h> //异常抛错检测 jmp_buf buf1; ...

  5. Flume Interceptors官网剖析(博主推荐)

    不多说,直接上干货! Flume Sources官网剖析(博主推荐) Flume Channels官网剖析(博主推荐) Flume Channel Selectors官网剖析(博主推荐) Flume ...

  6. ASP.NET路径解惑

    对于ASP.NET的路径问题,一直都是云里雾里,没有去详细的理解,今天正好可以梳理一下它们之间的关系和使用方法.而若想明白路径的表示方式的使用方法和区别以及注意事项可以通过下面的几个概念来进一步加深: ...

  7. Android 图片缓存处理

    异步下载 / 本地缓存 异步下载 大家都知道,在Android应用中UI线程5秒没响应的话就会抛出无响应异常,对于远程获取大的资源来说,这种异常还是很容易就会抛出来的,那么怎么避免这种问题的产生.在a ...

  8. Java Web学习总结(14)——JSP基础语法

    任何语言都有自己的语法,JAVA中有,JSP虽然是在JAVA上的一种应用,但是依然有其自己扩充的语法,而且在JSP中,所有的JAVA语句都可以使用. 一.JSP模版元素 JSP页面中的HTML内容称之 ...

  9. 洛谷——U10206 Cx的治疗

    https://www.luogu.org/problem/show?pid=U10206 题目背景 「Cx的故事」众所周知,Cx是一个宇宙大犇.由于Cx在空中花园失足摔下,导致他那蕴含着无穷智慧的大 ...

  10. Dynamics CRM2016 Web API之Expand related entities &amp; $ref &amp; $count

    本篇介绍两个关于1:N关系中通过主实体取关联子实体的api,这两个api会常常被用到并且比原来的odata方式更加方便.之前假设我们要取主实体下全部的关联实体的记录都是通过Retrieve Multi ...