RxSwift 判断是否为“主队列”

在开发 iOS 的时候,我们都知道 UI 相关的操作必须放在主线程,但是只要放在主线程就安全了么?

答案是否定的。在苹果的 MapKit 框架中,一个名为 addOverlay 的方法不仅要放在主线程中,而且必须放在主队列中。苹果公司的 Developer Technology Support 承认这是一个 bug。

所以在进行 UI 相关的操作时,最安全的方式是在主线程主队列中进行。那么应该怎么判断当前是不是主线程主队列呢?

首先我们先分清楚线程和队列的关系。

线程和队列

队列不是线程,队列时用来组织任务的,我们将任务添加到队列中,系统会根据资源决定是否创建新的线程去处理队列中的任务。线程的创建、维护和销毁由操作系统来管理。

队列分为两种类型:串行队列并行队列。在 iOS 系统中提供了 5 个不同全局队列,分别为:

  • 主队列(main queue)
  • 4个不同优先级的后台队列,它们的优先级分别为:High Priority QueueDefault Priority QueueLow Priority Queue,及优先级更低的 Background Priority Queue(用于 I/O)。

主队列为串行队列它内部的任务都会放在主线程中执行,UI 相关的操作都应该放在该队列中,获取方式:

// Object-C
dispatch_get_main_queue()  

其他四种不同优先级的全局队列为并行队列,获取方式为:

// Object-C
dispatch_get_global_queue(long identifier, unsigned long flags);  

另外还可以创建自定义队列(custom queue)

// Object-C
dispatch_queue_t dispatch_queue_create(const char *_Nullable label,
            dispatch_queue_attr_t _Nullable attr);

判断主队列

每个 APP 只有一个主线程,但是主线程上可以运行很多不同的队列,所以如果当前是主线程并不能保证当前为主队列。主队列内部的任务都会放在主线程中执行,所以如果当前队列为主队列,就可以确保任务在主线程主队列中运行。那么我们的问题就归结为怎么判断当前为主队列

pthread

pthread 是一套通用使用 C 语言编写的多线程的 API,可以在 Unix/Linux/Windows 等系统跨平台使用。常用 API :

  1. 创建一个线程
pthread_create( )
  1. 退出当前线
pthread_exit ( )
  1. 获取主线程
pthread_main_np ( ) :

应用 :

if (pthread_main_np()) { 
    // do something in main thread 
} else { 
    // do something in other thread 
}

总结:
使用 pthread 只能判断当前是不是主线程,而不能判断当前是不是主队列。

NSThread

NSThread 是一套苹果公司开发的使用 Objective-C 编写的多线程 API,其功能基本类同于 pthread。由于 GCD 本身没有提供判断当前线程是否是主线程的 API,因此我们常常使用 NSThread 中的 API 代替。
常用API :

  1. 是否是主线程
@property (readonly) BOOL isMainThread
  1. 取消线程
- (void)cancel
  1. 开始线程
- (void)start

应用:

if ([NSThread isMainThread]) { 
    // do something in main thread 
} else { 
    // do something in other thread 
}

总结:
该 API 同样只会检查当前的线程是否是主线程,不能检查是不是主队列。

线程关联数据

在 GCD 的 API 中还有通过 dispatch_queue_set_specific()dispatch_get_specific() 将一个值关联到指定的线程上,下面是常见 API:

  1. 获得当前队列,该方法在 iOS6.0之后已被弃用
dispatch_get_current_queue()
  1. 在指定的 queue 上通过 key 管理一个 value。
// Object-C
dispatch_queue_set_specific(dispatch_queue_t queue, const void *key,
void *_Nullable context, dispatch_function_t _Nullable destructor);

// Swift
public func setSpecific<T>(key: DispatchSpecificKey<T>, value: T?)

参数:

  • queue:需要关联的queue,不允许传入NULL。
  • key:唯一的关键字。
  • context:要关联的内容,可以为NULL。
  • destructor:释放context的函数,当新的context被设置时,destructor会被调用
  1. 根据指定的 key 取出当前 queue 的关联的 context,如果当前 queue 没有 key 关联的 context,则会从 queue 的 target queue 中获取,如果依然没有,则会返回 NULL。
// Object-C
dispatch_get_specific(const void *key)
// Swift
public func getSpecific<T>(key: DispatchSpecificKey<T>) -> T?

参数:

  • key:唯一的关键字。

应用:

// Object-C
static void *mainQueueKey = "mainQueueKey"; 
dispatch_queue_set_specific(dispatch_get_main_queue(), mainQueueKey, &mainQueueKey, NULL); 
if (dispatch_get_specific(mainQueueKey)) { 
    // do something in main queue 
} else { 
    // do something in other queue 
}

总结:

  1. 可以实现主线程的判断。
  2. 在判断是否为主队列之前,必须提前在主队列上设置一个关联的值。
  3. 此方法也可以用于判断其他特定的队列。

RxSwift 的具体实现

下面为 RxSwift 的具体实现代码:

extension DispatchQueue {
    // 注意此方法为 static
    private static var token: DispatchSpecificKey<()> = {
        // 初始化一个 key
        let key = DispatchSpecificKey<()>()
        // 在主队列上关联一个空元组
        DispatchQueue.main.setSpecific(key: key, value: ())
        return key
    }()
    
    // 通过队列上是否有绑定 token 对应的值来判断是否为主队列
    static var isMain: Bool {
        return DispatchQueue.getSpecific(key: token) != nil
    }
}

参考

iOS判断是否在主线程的正确姿势

主线程中也不绝对安全的 UI 操作

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,684评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,143评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,214评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,788评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,796评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,665评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,027评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,679评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,346评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,664评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,766评论 1 331
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,412评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,015评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,974评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,073评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,501评论 2 343

推荐阅读更多精彩内容