• 对dispatch_async到主线程的逻辑封装成C/C++接口类型


    背景:代码里面有时候会把将要运行的内容放到主线程里面运行,但假设已经是主线程里面的代码调用dispatch_async的时候偶尔会出现crash,所以就须要推断是否已经在主线程里面了。

    通常的做法类似于以下所代码:

        ......
        if ([NSThread isMainThread]) {
            block();
        } else {
            dispatch_async(dispatch_get_main_queue(), block);
        }
        ......

    所以在SDWebImage库里面进行了一次封装,把这部分内容放在了一个宏里面:
    #define dispatch_main_async_safe(block)
        if ([NSThread isMainThread]) {
            block();
        } else {
            dispatch_async(dispatch_get_main_queue(), block);
        }

    这样就能够在代码里面调用dispatch_main_async_safe安全的分发任务到主线程里面执行

    可是这个的问题是,宏里面的block是无法打断点调试的。比方:
    dispatch_main_async_safe(^{
        if(a) {
            [self foo1];
        }
        else {
            [self foo2];
        }
    });

    我们无法在if推断那里加个断点,来step by step的进行调试。

    所以,我才有了想要自己封装一个函数来调用的想法。

    我们能够发现dispatch_async这类的方法事实上是C接口的,所以在封装的过程里面遇到非常多问题,搞不定。

    最后经过网友的提醒,OC是能够直接调用C方法,而不须要把文件改为.mm后缀的,但假设调用C++方法,就须要把文件改为.mm后缀了。

    我的代码里面有非常多.m文件,也有非常多.mm文件,于是我不得不封装C和C++的接口来使用。




    下面是C接口:
    头文件

    //
    //  XXXDispatchMainQueueSafeC.h
    //
    //  Created by robyzhou on 15/7/16.
    //  Copyright (c) 2015年 robyzhou. All rights reserved.
    //
    
    #ifndef __XXX__XXXDispatchMainQueueSafeM__
    #define __XXX__XXXDispatchMainQueueSafeM__
    
    #include <stdio.h>
    
    typedef void(^Cblock)();
    
    void dispatch_main_async_XXX_m(Cblock block);
    
    void dispatch_main_sync_XXX_m(Cblock block);
    
    #endif


    m文件

    //
    //  XXXDispatchMainQueueSafeC.c
    //  XXX
    //
    //  Created by robyzhou on 15/7/16.
    //  Copyright (c) 2015年 robyzhou. All rights reserved.
    //
    
    #include "XXXDispatchMainQueueSafeM.h"
    
    void dispatch_main_async_XXX_m(Cblock block) {
        if ([NSThread isMainThread]) {
            block();
        } else {
            dispatch_async(dispatch_get_main_queue(), block);
        }
    }
    
    void dispatch_main_sync_XXX_m(Cblock block) {
        if ([NSThread isMainThread]) {
            block();
        } else {
            dispatch_sync(dispatch_get_main_queue(), block);
        }
    }

    下面是C++接口:
    头文件

    //
    //  XXXDispatchMainQueueSafe.h
    //  XXX
    //
    //  Created by robyzhou on 15/7/15.
    //  Copyright (c) 2015年 robyzhou. All rights reserved.
    //
    
    #import <Foundation/Foundation.h>
    
    @interface XXXDispatchMainQueueSafe : NSObject
    
    typedef void(^block)();
    
    void dispatch_main_async_XXX_mm(block block);
    
    void dispatch_main_sync_XXX_mm(block block);
    
    @end

    mm文件

    //
    //  XXXDispatchMainQueueSafe.m
    //  XXX
    //
    //  Created by robyzhou on 15/7/15.
    //  Copyright (c) 2015年 robyzhou. All rights reserved.
    //
    
    #import "XXXDispatchMainQueueSafeMM.h"
    
    @implementation XXXDispatchMainQueueSafe
    
    void dispatch_main_async_XXX_mm(block block) {
        if ([NSThread isMainThread]) {
            block();
        } else {
            dispatch_async(dispatch_get_main_queue(), block);
        }
    }
    
    void dispatch_main_sync_XXX_mm(block block) {
        if ([NSThread isMainThread]) {
            block();
        } else {
            dispatch_sync(dispatch_get_main_queue(), block);
        }
    }
    
    @end

    这样,假设你是在.m文件调用。就直接调用dispatch_main_async_XXX_m / dispatch_main_sync_XXX_m。假设你是在.mm文件调用。就直接调用dispatch_main_async_XXX_mm / dispatch_main_sync_XXX_mm。 比方:

    dispatch_main_async_XXX_m(^{
        if(a) {
            [self foo1];
        }
        else {
            [self foo2];
        }
    });

    而且你能够在 block里面加断点进行调试

    假设你有不论什么问题,欢迎留言。一起讨论,谢谢。
  • 相关阅读:
    一起来学linux:网络命令
    python网络爬虫之使用scrapy自动登录网站
    python网络爬虫之requests库 二
    一起来学linux:网络配置
    一起来学linux:日志文件
    一起来学linux:进程
    python网络爬虫之requests库
    【原创】大数据基础之Ambari(1)简介、编译安装、使用
    【原创】数据库基础之Mysql(1)常用命令
    【原创】大叔经验分享(22)securecrt连接自动断开
  • 原文地址:https://www.cnblogs.com/mfmdaoyou/p/6816118.html
Copyright © 2020-2023  润新知