zoukankan      html  css  js  c++  java
  • iOS开发之Run Loop

    1、概述

    (1) Run Loop提供了一种异步执行代码的机制,不能并行执行任务。

    (2) 在主队列中,Main Run Loop直接配合任务的执行,负责处理UI事件、计时器,以及其它内核相关事件。

    (3) Run Loop的主要目的是保证程序执行的线程不会被系统终止。

    Run Loop的工作特点:

    (1) 当有事件发生时,Run Loop会根据具体的事件类型通知应用程序做出响应。

    (2) 当没有事件发生时,Run Loop会进入休眠状态,从而达到省电的目的。

    (3) 当事件再次发生时,Run Loop会被重新唤醒,处理事件。

    主线程和其他线程中的Run Loop

    iOS程序的主线程默认已经配置好了Run Loop,其他线程默认情况下没有设置Run Loop。

    一般在开发中很少会主动创建RunLoop,而通常会把事件添加到RunLoop中。

    loop示意图:

    2UIApplication中的Run Loop

    3多线程中的循环引用

    如果self对象持有操作对象的引用,同时操作对象当中又直接访问了self时,才会造成循环引用。

    单纯在操作对象中使用self不会造成循环引用。

    注意:此时不能使用(weakSelf)

    4、多线程中的资源共享

    并发编程中许多问题的根源就是在多线程中访问共享资源。资源可以是一个属性、一个对象、网络设备或者一个文件等。

    在多线程中任何一个共享的资源都可能是一个潜在的冲突点,必须精心设计以防止这种冲突的发生。

    资源共享示例:

    5互斥锁(@synchronized)

    互斥锁使用注意:

    (1)加互斥锁,在共享资源的“读”“写”范围添加互斥锁

    (2)要让锁的范围尽可能小!

    (3)资源抢夺做简单的做法就是使用互斥锁@synchronized

    (4)使用互斥锁,会变慢,互斥锁的代价十分高昂!

    加上互斥锁就可以使一个资源在同一时间只能被一个线程访问,只有这个资源被这个线程用完后其他线程才能访问。

    互斥锁用法:

    @synchronized(self) {

    //线程操作

    }

    例如:

    @interface MJViewController ()

    @property (weak, nonatomic) IBOutlet UITextView *infoText;

    // 票数,若使用原子锁,只需在互斥锁的基础上把下面nonatomic改为atomic即可

    @property (nonatomic, assign) NSInteger tickets;

    @end

    @implementation MJViewController

    - (void)viewDidLoad

    {

        [super viewDidLoad];

    }

    // 卖票循环,一直卖完为止

    - (void)doSaleLoop:(NSString *)opName

    {

        // 所有线程进入都可以循环卖票

        while (YES) {

            @synchronized(self) {

                if (self.tickets > 0) {

                    --self.tickets;

                   

                    NSLog(@"剩余票数 %d - %@ - %@", self.tickets, opName, [NSThread currentThread]);

                } else {

                    break;

                }

            }

            //-----------------------------------------

            // 模拟休眠,跟资源抢夺无关,不用放在锁里

            if ([opName isEqualToString:@"OP 1"]) {

                [NSThread sleepForTimeInterval:1.0f];

            } else {

                [NSThread sleepForTimeInterval:0.3f];

            }

        }

    }

    #pragma mark 模拟多人卖票

    #pragma mark GCD实现

    - (IBAction)doSale:(id)sender

    {

        // 做多线程开始时,永远不要相信一次运行的结果

        self.tickets = 20;

       

        // 1. 队列

        dispatch_queue_t q = dispatch_queue_create("sale", DISPATCH_QUEUE_CONCURRENT);

       

        // 2. 添加任务

        dispatch_async(q, ^{

            [self doSaleLoop:@"OP 1"];

        });

       

        dispatch_async(q, ^{

            [self doSaleLoop:@"OP 2"];

        });

       

        dispatch_async(q, ^{

            [self doSaleLoop:@"OP 3"];

        });

       

        dispatch_async(q, ^{

            [self doSaleLoop:@"OP 4"];

        });

    }

    @end

    【备注】在iOS中还有一种锁的功能,原子锁 —— 多读单写的锁(128位自旋锁),也会消耗性能。

    原子锁只保护写入时的数据正确,而读取不负责。

    对于要写入的资源,保护“写入数据”的正确性重要,还是读取的准确性重要!

    如果只是开发单写多读的功能,只需要使用原子锁即可。

    @synchronized 性能消耗非常的大,苹果官方不建议使用。

    在实际开发中,不要去抢夺资源!

    *** 并发编程最主要的目的是提高性能,让更多的代码同时运行,达到并发运行,提高整体性能的目的!

    *** 手机开发最主要的是流畅,并行,至于资源抢夺的功能开发是属于服务端的范畴!

  • 相关阅读:
    java利用freemarker导出world
    各种Java加密算法
    SM2的非对称加解密java工具类
    Mybatis分页插件--------Pagehelper
    JS实现浏览器打印、打印预览
    java大文件断点续传
    数字证书在web应用中实现登陆
    CA数字加密解密Demo
    使用数字证书进行签名和加密解密
    一个有趣的模拟光照的shader(类似法线贴图)
  • 原文地址:https://www.cnblogs.com/lifengfneg/p/4774014.html
Copyright © 2011-2022 走看看