首页 > 系统相关 >OC-从内存角度理解block可作为方法传入参数的原因

OC-从内存角度理解block可作为方法传入参数的原因

时间:2024-07-08 16:21:18浏览次数:16  
标签:completion block void OC ARC 内存 Block

从内存管理的角度来看,block可以作为方法的传入参数是因为block在Objective-C中被设计为一种特殊的对象,它们可以在堆(heap)上分配和管理。这使得block可以像其他对象一样被传递、复制和持有。以下是一些关键点,解释为什么block可以作为方法的传入参数:

1. Block的类型和内存管理

在Objective-C中,block有三种类型:

  • 栈上的block(Stack Block):默认情况下,block是在栈上分配的。这种block的生命周期与其作用域相同,当作用域结束时,block会被销毁。
  • 堆上的block(Heap Block):当block被复制(使用Block_copy[block copy])时,它会被移动到堆上。堆上的block可以在作用域之外存在,并且可以被多个对象持有。
  • 全局block(Global Block):如果block不捕获任何外部变量,它会被优化为全局block,类似于全局函数。

2. Block的复制

当block作为方法参数传递时,通常会被复制到堆上,以确保它在方法执行期间和之后仍然有效。复制block会将其从栈移动到堆,并增加其引用计数。这样,即使方法返回后,block仍然可以被安全地调用。

void (^myBlock)(void) = ^{
    NSLog(@"Hello, World!");
};

// 将block复制到堆上
void (^heapBlock)(void) = [myBlock copy];

3. ARC和Block

在使用自动引用计数(ARC)时,编译器会自动处理block的内存管理。当block作为方法参数传递时,ARC会自动复制block并管理其生命周期。

- (void)performOperationWithCompletion:(void (^)(BOOL success))completion {
    // ARC会自动复制completion block到堆上
    if (completion) {
        completion(YES);
    }
}

4. 捕获变量

block可以捕获其作用域中的变量,并在block内部使用这些变量。捕获的变量会被复制到block的内部结构中,以确保它们在block执行时仍然有效。

int multiplier = 7;
int (^myBlock)(int) = ^(int num) {
    return num * multiplier; // multiplier被捕获并复制到block内部
};

5. Block作为参数的内存管理

当block作为方法参数传递时,通常会发生以下步骤:

  1. 复制block:如果block在栈上,ARC会自动将其复制到堆上。
  2. 传递block:复制后的block被传递给方法,并增加其引用计数。
  3. 持有block:方法可以选择持有block(例如,将其存储在实例变量中),以便在方法返回后继续使用。
  4. 释放block:当block不再需要时,ARC会自动减少其引用计数,并在引用计数为零时释放block。

示例代码

以下是一个示例,展示了block作为方法参数的内存管理:

@interface MyClass : NSObject
@property (nonatomic, copy) void (^completionBlock)(BOOL success);
- (void)performOperationWithCompletion:(void (^)(BOOL success))completion;
@end

@implementation MyClass

- (void)performOperationWithCompletion:(void (^)(BOOL success))completion {
    // 复制block并持有
    self.completionBlock = completion;
    
    // 模拟异步操作
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        // 操作完成后调用block
        if (self.completionBlock) {
            self.completionBlock(YES);
        }
    });
}

@end

// 使用示例
MyClass *myObject = [[MyClass alloc] init];
[myObject performOperationWithCompletion:^(BOOL success) {
    if (success) {
        NSLog(@"Operation was successful");
    }
}];

在这个示例中,completion block被复制到堆上,并由self.completionBlock持有。即使performOperationWithCompletion:方法返回后,block仍然有效,并可以在异步操作完成后被调用。

总结

从内存管理的角度来看,block可以作为方法的传入参数是因为block在Objective-C中被设计为一种特殊的对象,可以在堆上分配和管理。ARC会自动处理block的复制和引用计数,使得block可以安全地在方法之间传递和持有。这种设计使得block在处理异步操作和回调时非常灵活和强大。

标签:completion,block,void,OC,ARC,内存,Block
From: https://www.cnblogs.com/zjq1999/p/18289802

相关文章

  • go 使用websocket
    packagechatimport( "encoding/json" "github.com/gorilla/websocket" "github.com/zeromicro/go-zero/core/logx" "log" "net/http" "sync")typeClientstruct{ conn*websocket.Conn......
  • 逆向内存加载Dex(动态加载class)
    逆向一个app,其核心算法是通过反射调用的,反编译软件中无法找到该类,并且也无法hook.Java.perform(function(){Java.enumerateClassLoaders({onMatch:function(loader){ try{ if(loader.loadClass("com.xxxxx")){ console.log("=============......
  • debian11 podman搭建rocketmq (初步测试)
    前言由于个人学习测试需要,想要在debian11环境下,使用podman搭建一个简单的单节点rocketmq,搭建过程比较繁琐,所以记录下来留作后续参考。由于对rocketmq镜像不熟悉,有些配置可能存在错误,会在后续有能力优化时继续更新此文档。后续可能会补充podman-compose实现方式,目前是shell脚本......
  • 常用 Docker 命令和配置指南 新手几乎够用 老手持续更新
    容器管理命令查看容器:查看所有已启动/未启动的容器:dockerps-a查看所有已启动的容器:dockerps启动和停止容器:启动容器:dockerstart容器ID停止容器:dockerstop容器ID重启容器:dockerrestart容器ID删除容器:删除容器:dockerrm容器ID镜像管理命令......
  • Docker基本操作
    1.简介Docker是一个开源的应用容器引擎,它允许开发者打包他们的应用以及应用的依赖包到一个可移植的容器中,然后发布到任何流行的Linux机器上,也可以实现虚拟化。容器是完全使用沙箱机制,相互之间不会有任何接口。Docker的主要用途包括:简化配置:Docker可以确保应用的运......
  • Docker部署Django+MySQL+Redis+Nginx+uWSGI+Celery(超详细)
    一、知识储备经过我们之前学习的Docker相关知识,现在我们来进行实战,以下介绍如何通过DockerCompose部署Django项目:先前知识:Docker学习笔记(一)概念理解-CSDN博客Docker学习笔记(二)镜像、容器、仓库相关命令操作-CSDN博客Docker学习笔记(三)Dockerfile-CSDN博客DockerCompose......
  • 你真的了解Java内存模型JMM吗?
    哈喽,大家好......
  • Docker安装命令
    事先声明-对于命令直接粘进去执行就可以Docker分为CE和EE两大版本。CE即社区版(免费,支持周期7个月),EE即企业版,强调安全,付费使用,支持周期24个月。DockerCE分为stabletest和nightly三个更新频道。最重要一点:DockerCE支持64位版本CentOS7,并且要求内核版本......
  • OC-NSArray的基本介绍
    NSArray是不可变的;存储不同类型的对象。这意味着一个NSArray可以同时包含NSString、NSNumber、NSDictionary等不同类型的对象。同时只能存储对象,不能直接存储基本数据类型(如int、float等)。如果需要存储基本数据类型,应该先将它们封装为相应的对象类型(如NSNumber或NSValue)。......
  • docker-compose vs docker-stack
    docker-composevsdocker-stack都是docker两个容器编排工具,docker-compose是属于第三方容器编排工具需要单独安装,docker-stack是docker内置容器编排工具。docker-compose一般配合K8S使用,目前要容器管理方面K8S有着比较明显的优势,所以docker-compose目前比较流行。docker-stac......