网络多线程的入门级理解

模拟耗时操作

  • 耗时操作对UI的影响 : 会卡死UI / 界面 / 主线程
  • 如何解决耗时操作卡死主线程?
    • 使用多线程技术,把耗时的操作放进子线程中执行,使得主线程有资源处理UI
  • 多线程的核心思想 : 把耗时操作放到子线程异步执行

多线程基本概念

同步和异步

  • 同步和异步是任务 / 代码 执行的两种方式

同步

  • 多个任务按顺序依次执行,就是同步执行

异步

  • 多个任务同时执行,就是异步执行
  • 如何保证多个任务同时执行?
    • 开线程,开多个线程,就可以保证多个任务同时执行
  • 提示 : 凡是遇到异步 / 多线程 / 耗时操作 第一反应就是需要开启新的子线程
  • 学习多线程就是为了如何让任务在子线程异步执行

进程和线程

进程

  • 系统中正在运行的应用程序叫做进程
  • 进程可以类必成公司

线程 / 多线程

  • 程序一启动就会默认开启一个线程,称之为主线程
  • 线程是进程最基本的执行单元,进程里面所有的任务都在线程中执行的
  • 一个进程,可以开启多个线程,称之为多线程

多线程执行原理

  • CPU在多个线程之间,快速来回的切换,调度线程执行任务,如果切换的速度足够快,就造成多个任务同时执行的假象

多线程优缺点

优点

  • 可以适当提高程序执行的效率 (开启多个线程下载视频)

缺点

  • 前提 : 当线程非常多的时候,就暴露缺点
  • 会消耗大量的CPU资源
  • 时间开销 / 空间开销
  • 多线程的使用原则 : 能不用就不用,如果非要用,就简单的使用,少用

主线程

  • 作用 : 刷新UI / 处理UI事件
  • 使用时的注意点 : 不要把耗时操作放到主线程中执行

pthread

  • 学习pthread的目的 : 就是为了复习C语言相关的知识点
  • 在C语言中,一般带_t / _ref标识数据类型
  • NULL : 表示空地址,一般在C语言使用;
  • nil : 表示空对象,一般在OC使用;
  • 其实,NULLh和nil本质上没有半点儿区别
  • void *(*)(void *) : 表示指向函数的指针,即函数名;函数名就是表示函数地址;
  • 数组地址就是数组名或者数组第0个角标元素的地址
  • void * 表示可以指向任何地址的指针,代表任意数据类型;类似于OC的id;
返回值    函数名    函数参数
void *    (*)    (void *)

桥接

  • 使用场景 : 在C语言和OC语言混合开发时,需要做数据类型转换,有时候需要使用桥接;
  • 桥接作用 : 在做数据类型转换时,告诉编译器如何管理C语言的内存
  • 提示 : 在ARC环境下,编译器在编译时,不会自动管理C语言申请的内存空间
  • 提示: 在ARC环境下,加上__bridge 表示告诉编译器C语言申请的内存也是自动管理的,因为大环境是ARC的
  • 提问 : MRC环境下,需要使用__bridge 吗? 不需要,因为本来就是手动管理的

NSThread创建线程三种方式

构造方法

  • 可以拿到线程对象
  • 需要自己启动线程
// 创建线程对象
NSThread *thread = [[NSThread alloc] initWithTarget:self selector:@selector(demo:) object:@"alloc"];
// 启动线程
[thread start];

类方法

  • 不可以拿到线程对象
  • 不需要自己启动线程
[NSThread detachNewThreadSelector:@selector(demo:) toTarget:self withObject:@"detach"];

NSObject分类方法

  • 不可以拿到线程对象
  • 不需要自己启动线程
[self performSelectorInBackground:@selector(demo:) withObject:@"perform"];

target和selector的关系

  • 执行哪个对象的哪个方法
  • 需求 : 执行Person对象的run方法,run方法需要在子线程执行
// 创建线程对象
NSThread *thread = [[NSThread alloc] initWithTarget:_p selector:@selector(run:) object:@"person"];
// 启动线程
[thread start];

线程生命周期 / 线程状态

  • 新建状态
NSThread *thread = [[NSThread alloc] initWithTarget:self selector:@selector(demo) object:nil];
  • 就绪状态
[thread start];
  • 运行状态 : 程序员无法干预
  • 阻塞状态 : 调用sleep方法 / 添加互斥锁(同步锁)
  • 死亡状态
    • 正常死亡 : 任务执行结束
    • 异常死亡 : exit

线程属性

  • name : 标识唯一的线程对象,方便定位线程对象
  • threadPriority : 决定了线程有更多的机会被CPU调度执行;等同于qualityOfService;实际开发中千万不要随意修改
  • stackSize : 线程对象占用内存空间大小.主线程 / 子线程 512KB

多线程访问共享资源 (会造成线程安全问题)

  • 当多个线程同时操作共享资源,就会出现线程安全问题
  • 解决办法 : 加锁 (互斥锁 / 同步锁)
  • 互斥锁 / 同步锁 : 使用了线程同步技术
  • 特点 : 可以保证被锁定的代码,同一时间只有一个线程可以访问
  • self : 表示互斥锁的参数;互斥锁的参数,又叫做锁对象;
  • 锁对象 : 任何继承自NSObject的对象,都可以作为互斥锁的参数;内部有把锁,默认是开启的
  • 锁对象必须是全局的对象;self是最方便获取的全局的锁对象
  • 局部锁对象是锁不住的,因为每次线程进来之前会新建一把锁
  • 提示 : 加锁的事情,不是再客户端操作的;是服务器加锁的,多线程资源共享绝大多数是在服务器发生;
  • 提示 : 加锁是牺牲了性能,保证安全.客户端的性能不能轻易牺牲

异步下载网络图片

  • 在 iOS 开发中,使用多线程只有一个目的:将耗时操作放在后台工作,待工作完成后,通知主线程更新 UI

  • 耗时的下载操作放在子线程

- (void)viewDidLoad {
    [super viewDidLoad];
    
//    [self loadImageData];
    
    // 在子线程执行耗时操作
    [self performSelectorInBackground:@selector(loadImageData) withObject:nil];
}
// 下载图片的主方法
- (void)loadImageData
{
    // URL
    NSURL *URL = [NSURL URLWithString:@"https://ss0.bdstatic.com/94oJfD_bAAcT8t7mm9GUKT-xh_/timg?image&quality=100&size=b4000_4000&sec=1476696781&di=f721c3cba572282b9d4b135866894858&src=http://www.hn.xinhuanet.com/2016-08/31/1119483302_14726048769591n.jpg"];
    // 发送网络请求,获取图片二进制数据,是个耗时操作
    NSData *data = [NSData dataWithContentsOfURL:URL];
    // image : 就是子线程执行的结果,需要传递到主线程
    UIImage *image = [UIImage imageWithData:data];
    
    // 下载完成之后,通知主线程刷新UI
    // waitUntilDone : 是否等待updateUI执行完,再执行后面的代码,一般传入NO
    [self performSelectorOnMainThread:@selector(updateUI:) withObject:image waitUntilDone:NO];
    
    NSLog(@"后面的代码");
}
  • 更新UI的操作在主线程
// 回到主线程更新UI
- (void)updateUI:(UIImage *)image
{
    self.imgView.image = image;
    [self.imgView sizeToFit];
    self.scrollView.contentSize = image.size;
}
  • 在子线程下载图片,在主线程更新UI,是线程间通信的一种;
  • 线程间通信 : 一个线程把他执行的结果,传递到另外的一个线程

你可能感兴趣的:(网络多线程的入门级理解)