zoukankan      html  css  js  c++  java
  • GCD之信号量机制二

    在前面GCD之信号量机制一中介绍了通过信号量设置并行最大线程数,依此信号量还可以防止多线程访问公有变量时数据有误,下面的代码能说明。

    1.下面是不采用信号量修改公有变量的值

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    dispatch_group_t group=dispatch_group_create();
    //    dispatch_semaphore_t semaphore=dispatch_semaphore_create(1);
        dispatch_queue_t queue=dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
        __block int count=1000;
        for (int i=0; i<100; i++) {
            //信号量减1,如果同时开启10个以上的线程,则信号量小于等于0,此时就会阻塞该线程。
    //        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
            dispatch_group_async(group, queue, ^{
                int value = (arc4random() % 4) + 6;
                NSLog(@"%d-%d= %d",count,value,count-value);
                count=count-value;
            //每个线程执行减1后通过信号量通知加1,这样始终保持线程在10个之内
    //        dispatch_semaphore_signal(semaphore);
            });
        }
         
        dispatch_group_wait(group, DISPATCH_TIME_FOREVER);

     2.运行结果如下:

    3.声明一个初始值为1的信号量来开启线程修改公有变量时

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    dispatch_group_t group=dispatch_group_create();
       dispatch_semaphore_t semaphore=dispatch_semaphore_create(1);
       dispatch_queue_t queue=dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
       __block int count=1000;
       for (int i=0; i<100; i++) {
           //信号量减1,如果同时开启10个以上的线程,则信号量小于等于0,此时就会阻塞该线程。
           dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
           dispatch_group_async(group, queue, ^{
               int value = (arc4random() % 4) + 6;
               NSLog(@"%d-%d= %d",count,value,count-value);
               count=count-value;
           //每个线程执行减1后通过信号量通知加1,这样始终保持线程在10个之内
           dispatch_semaphore_signal(semaphore);
           });
       }
        
       dispatch_group_wait(group, DISPATCH_TIME_FOREVER);

     4.运行结果如下:

    可以看到,第一代码段时修改公有变量时不是有序的,第二个代码段才是真正正确的修改顺序。这过程可取款的过程一样,金额是1000,可能在不同的地方同时取款,取款时不可能金额是像第一代码段那样。这个和C#的lock关键字有一样的效果。

  • 相关阅读:
    解决pip3的ImportError: cannot import name 'main'
    linux 安装Python3.6
    Linux安装redis和部署
    redis密码管理
    CentOS7使用firewalld打开关闭防火墙与端口
    scrapy 从Windwos平台移植到 Linux平台之实操
    Linux 环境下安装Maven
    解决:安装Jenkins时web界面出现该jenkins实例似乎已离线
    持续集成工具Jenkins结合SVN的安装和使用
    Linux下的SVN服务器搭建
  • 原文地址:https://www.cnblogs.com/yulei126/p/6783209.html
Copyright © 2011-2022 走看看