执行异步派发时, 需要拷贝块

2024-06-11 17:20

本文主要是介绍执行异步派发时, 需要拷贝块,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

dispatch_async 在执行异步派发时,需要拷贝 block。

Block 的行为

在 Objective-C 中,block 是带有自动变量(局部变量)和堆栈信息的代码段。当你声明一个 block,它默认存储在栈上。这意味着,当函数返回时,block 可能不再有效。为了确保 block 在需要时仍然有效,block 通常会被拷贝到堆上。

dispatch_async 的行为

当你使用 dispatch_async 将一个 block 任务派发到一个队列时,需要确保这个 block 在队列执行时仍然有效。为了做到这一点,dispatch_async 会拷贝这个 block,并将拷贝后的 block 提交到目标队列。

代码分析

来看一个简单的代码示例:

dispatch_queue_t queue = dispatch_queue_create("com.example.queue", DISPATCH_QUEUE_CONCURRENT);dispatch_async(queue, ^{NSLog(@"Hello from block");
});

在这段代码中,dispatch_async 做了以下工作:

  1. Block 的创建

    • ^{ NSLog(@"Hello from block"); } 是一个 block,默认在栈上。
  2. Block 的拷贝

    • dispatch_async 被调用时,它会拷贝这个 block 到堆上,以确保 block 在异步执行时依然有效。
  3. Block 的提交

    • 拷贝后的 block 被提交到 queue 队列。
  4. Block 的执行

    • 队列异步执行这个 block。

为什么需要拷贝

考虑以下示例:

void myFunction() {int value = 10;dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{NSLog(@"Value: %d", value);});// myFunction 返回后,value 变量会被销毁
}

myFunction 中,value 是一个局部变量,存储在栈上。如果 block 没有被拷贝到堆上,myFunction 返回后,value 变量可能已经被销毁,而 block 中引用的 value 变得无效。

通过将 block 拷贝到堆上,block 会持有 value 的拷贝,确保 dispatch_async 中的 block 可以安全地使用 value,即使 myFunction 已经返回。

概述

  • 拷贝行为: dispatch_async 会拷贝 block,以确保 block 在异步执行时有效。
  • 安全性: 这样做是为了确保 block 中引用的变量在函数返回后依然有效,防止引用无效的局部变量。

这个行为是 GCD (Grand Central Dispatch) 的设计之一,旨在确保异步任务的安全性和稳定性。

这篇关于执行异步派发时, 需要拷贝块的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1051802

相关文章

java中ssh2执行多条命令的四种方法

《java中ssh2执行多条命令的四种方法》本文主要介绍了java中ssh2执行多条命令的四种方法,包括分号分隔、管道分隔、EOF块、脚本调用,可确保环境配置生效,提升操作效率,具有一定的参考价值,感... 目录1 使用分号隔开2 使用管道符号隔开3 使用写EOF的方式4 使用脚本的方式大家平时有没有遇到自

mybatis直接执行完整sql及踩坑解决

《mybatis直接执行完整sql及踩坑解决》MyBatis可通过select标签执行动态SQL,DQL用ListLinkedHashMap接收结果,DML用int处理,注意防御SQL注入,优先使用#... 目录myBATiFBNZQs直接执行完整sql及踩坑select语句采用count、insert、u

Python爬虫HTTPS使用requests,httpx,aiohttp实战中的证书异步等问题

《Python爬虫HTTPS使用requests,httpx,aiohttp实战中的证书异步等问题》在爬虫工程里,“HTTPS”是绕不开的话题,HTTPS为传输加密提供保护,同时也给爬虫带来证书校验、... 目录一、核心问题与优先级检查(先问三件事)二、基础示例:requests 与证书处理三、高并发选型:

一个Java的main方法在JVM中的执行流程示例详解

《一个Java的main方法在JVM中的执行流程示例详解》main方法是Java程序的入口点,程序从这里开始执行,:本文主要介绍一个Java的main方法在JVM中执行流程的相关资料,文中通过代码... 目录第一阶段:加载 (Loading)第二阶段:链接 (Linking)第三阶段:初始化 (Initia

Java中实现对象的拷贝案例讲解

《Java中实现对象的拷贝案例讲解》Java对象拷贝分为浅拷贝(复制值及引用地址)和深拷贝(递归复制所有引用对象),常用方法包括Object.clone()、序列化及JSON转换,需处理循环引用问题,... 目录对象的拷贝简介浅拷贝和深拷贝浅拷贝深拷贝深拷贝和循环引用总结对象的拷贝简介对象的拷贝,把一个

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

Java实现远程执行Shell指令

《Java实现远程执行Shell指令》文章介绍使用JSch在SpringBoot项目中实现远程Shell操作,涵盖环境配置、依赖引入及工具类编写,详解分号和双与号执行多指令的区别... 目录软硬件环境说明编写执行Shell指令的工具类总结jsch(Java Secure Channel)是SSH2的一个纯J

python 线程池顺序执行的方法实现

《python线程池顺序执行的方法实现》在Python中,线程池默认是并发执行任务的,但若需要实现任务的顺序执行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋... 目录方案一:强制单线程(伪顺序执行)方案二:按提交顺序获取结果方案三:任务间依赖控制方案四:队列顺序消

Python异步编程之await与asyncio基本用法详解

《Python异步编程之await与asyncio基本用法详解》在Python中,await和asyncio是异步编程的核心工具,用于高效处理I/O密集型任务(如网络请求、文件读写、数据库操作等),接... 目录一、核心概念二、使用场景三、基本用法1. 定义协程2. 运行协程3. 并发执行多个任务四、关键

Go语言连接MySQL数据库执行基本的增删改查

《Go语言连接MySQL数据库执行基本的增删改查》在后端开发中,MySQL是最常用的关系型数据库之一,本文主要为大家详细介绍了如何使用Go连接MySQL数据库并执行基本的增删改查吧... 目录Go语言连接mysql数据库准备工作安装 MySQL 驱动代码实现运行结果注意事项Go语言执行基本的增删改查准备工作