iOS开发之Block

简介:

1、定义

(1) Block是OC中的一种数据类型,在iOS开发中被广泛使用

(2) ^是Block的特有标记

(3) Block的实现代码包含在{}之间

(4) 大多情况下,以内联inline函数的方式被定义和使用

(5) Block与C语言的函数指针有些相似,但使用起来更加灵活

例如:

void(^demoBlock)() = ^ {

    NSLog(@"demo Block");

};

int(^sumBlock)(int, int) = ^(int x, int y) {

    return x + y;

};

格式说明:

(返回类型)(^块名称)(参数类型) = ^(参数列表) {代码实现};

如果没有参数,等号后面参数列表的()可以省略

2、常见相关面试题

Block可以使用在定义之前声明的局部变量:

int i = 10;

void(^myBlock)() = ^{

    NSLog(@"%d", i);

};

i = 100;//实际上并没效果

myBlock();

输出结果为:10

注意:

(1) 在定义Block时,会在Block中建立当前局部变量内容的副本(拷贝)

(2) 后续再对该变量的数值进行修改,不会影响Block中的数值

(3) 如果需要在block中保持局部变量的数值变化,需要使用__block关键字

(4) 使用__block关键字后,同样可以在Block中修改该变量的数值

3、当做参数传递

Block可以被当做参数直接传递:

NSArray *array = @[@"张三", @"李四", @"王五", @"赵六"];

[array enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {

    NSLog(@"第 %d 项内容是 %@", (int)idx, obj);

    if ([@"王五" isEqualToString:obj]) {

        *stop = YES;

    }

}];

说明:遍历并NSLog() array中的内容,当obj 为"王五"时停止遍历

4、使用局部变量

在被当做参数传递时,Block同样可以使用在定义之前声明的局部变量:

int stopIndex = 1;

NSArray *array = @[@"张三", @"李四", @"王五", @"赵六"];

[array enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {

    NSLog(@"第 %d 项内容是 %@", (int)idx, obj);

    if ([@"王五" isEqualToString:obj] || idx == stopIndex) {

        *stop = YES;

    }

}];

注意,默认情况下,Block外部的变量,在Block中是只读的

BOOL flag = NO;

[array enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {

    if ([@"王五" isEqualToString:obj] || idx == stopIndex) {

        *stop = YES;

        flag = YES;      // 编译错误!!!

        }

}];

5__block关键字

如果要修改Block之外的局部变量需要使用__block关键字

__block BOOL flag = NO;

[array enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {

    if ([@"王五" isEqualToString:obj] || idx == stopIndex) {

        *stop = YES;

        flag = YES;      // 现在可以修改了!!!

        }

}];

提示:无需使用__block关键字,在块代码中可以修改成员变量的数值(比较少用)

6传递对象

对象传递进Block的方式

NSString *stopName = @"王五";

NSArray *array = @[@"张三", @"李四", @"王五", @"赵六"];

[array enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {

    NSLog(@"第%d项内容是%@", (int)idx, obj);

    if ([stopName isEqualToString:obj] || idx == stopIndex) {

        *stop = YES;

    }

}];

为保证Block中的代码正常运行,在将stopName的指针传递给Block时,Block会自动对stopName的指针做强引用

7Block在栈区工作示意图

 

8typedef

可以使用typedef定义一个Block的类型,便于在后续直接使用

typedef double(^MyBlock)(double, double);

MyBlock area = ^(double x, double y) {

    return x * y;

};

MyBlock sum = ^(double a, double b) {

    return a + b;

};

NSLog(@"%.2f", area(10.0, 20.0));

NSLog(@"%.2f", sum(10.0, 20.0));

说明:

(1) typedef是关键字用于定义类型,MyBlock是定义的Block类型

(2) area、sum分别是MyBlock类型的两个Block变量。

 

尽管,typedef可以简化Block的定义,但在实际开发中并不会频繁使用typedef关键字。

这是因为Block具有非常强的灵活性,尤其在以参数传递时,使用Block的目的就是为了立即使用。

官方的数组遍历方法声明如下:

而如果使用typedef,则需要:

(1) typedef void(^EnumerateBlock)(id obj, NSUInteger idx, BOOL *stop);

(2) - (void)enumerateObjectsUsingBlock:(EnumerateBlock)block;

而最终的结果却是,除了定义类型之外,EnumerateBlock并没有其他用处。

9添加到数组

既然Block是一种数据类型,那么可以将Block当做比较特殊的对象

#pragma mark 定义并添加到数组

@property (nonatomic, strong) NSMutableArray *myBlocks;

int(^sum)(int, int) = ^(int x, int y) {

    return [self sum:x y:y];

};

[self.myBlocks addObject:sum];

int(^area)(int, int) = ^(int x, int y) {

    return [self area:x y:y];

};

[self.myBlocks addObject:area];

#pragma mark 调用保存在数组中的Block

int(^func)(int, int) = self.myBlocks[index];

return func(x, y);

10循环引用

@property (nonatomic, strong) NSMutableArray *myBlocks;

#pragma mark 将代码改为调用self的方法

int(^sum)(int, int) = ^(int x, int y) {

    return [self sum:x y:y];

};

[self.myBlocks addObject:sum];

#pragma mark 对象被释放时自动调用

- (void)dealloc

{

    NSLog(@"DemoObj被释放");

}

11、解除循环引用

局部变量默认都是强引用的,离开其所在的作用域之后就会被释放。

使用__weak关键字,可以将局部变量声明为弱引用

__weak DemoObj *weakSelf = self;

在Block中引用weakSelf,则Block不会再对self做强引用

int(^sum)(int, int) = ^(int x, int y) {

return [weakSelf sum:x y:y];

 

};

目录
相关文章
|
2月前
|
API 数据安全/隐私保护 iOS开发
利用uni-app 开发的iOS app 发布到App Store全流程
利用uni-app 开发的iOS app 发布到App Store全流程
106 3
|
4月前
|
存储 iOS开发
iOS 开发,如何进行应用的本地化(Localization)?
iOS 开发,如何进行应用的本地化(Localization)?
123 2
|
4月前
|
存储 数据建模 数据库
IOS开发数据存储:什么是 UserDefaults?有哪些替代方案?
IOS开发数据存储:什么是 UserDefaults?有哪些替代方案?
42 0
|
4月前
|
安全 编译器 Swift
IOS开发基础知识: 对比 Swift 和 Objective-C 的优缺点。
IOS开发基础知识: 对比 Swift 和 Objective-C 的优缺点。
106 2
|
2月前
|
API 开发工具 Android开发
iOS 和 Android 平台的开发有哪些主要区别?
iOS与Android开发区别:iOS用Objective-C/Swift,App Store唯一下载渠道;Android用Java/Kotlin,多商店发布(如Google Play、华为市场)。设计上,iOS简洁一致,Android灵活可定制。开发工具,iOS用Xcode,Android用Android Studio。硬件和系统多样性,iOS统一,Android复杂。权限管理、审核流程及API各有特点,开发者需依据目标平台特性进行选择。
36 3
|
11天前
|
前端开发 Android开发 iOS开发
【Flutter前端技术开发专栏】Flutter在Android与iOS上的性能对比
【4月更文挑战第30天】Flutter 框架实现跨平台移动应用,通过一致的 UI 渲染(Skia 引擎)、热重载功能和响应式框架提高开发效率和用户体验。然而,Android 和 iOS 的系统差异、渲染机制及编译过程影响性能。性能对比显示,iOS 可能因硬件优化提供更流畅体验,而 Android 更具灵活性和广泛硬件支持。开发者可采用代码、资源优化和特定平台优化策略,利用性能分析工具提升应用性能。
【Flutter前端技术开发专栏】Flutter在Android与iOS上的性能对比
|
11天前
|
存储 Swift iOS开发
使用Swift开发一个简单的iOS应用的详细步骤。
使用Swift开发iOS应用的步骤包括:创建Xcode项目,设计界面(Storyboard或代码),定义数据模型,实现业务逻辑,连接界面和逻辑,处理数据存储(如Core Data),添加网络请求(必要时),调试与测试,根据测试结果优化改进,最后提交至App Store或其它平台发布。
31 0
|
11天前
|
安全 Swift iOS开发
【Swift 开发专栏】Swift 与 UIKit:构建 iOS 应用界面
【4月更文挑战第30天】本文探讨了Swift和UIKit在构建iOS应用界面的关键技术和实践方法。Swift的简洁语法、类型安全和高效编程模型,加上与UIKit的紧密集成,使开发者能便捷地创建用户界面。UIKit提供视图、控制器、布局、动画和事件处理等功能,支持灵活的界面设计。实践中,遵循设计原则,合理组织视图层次,运用布局和动画,以及实现响应式设计,能提升界面质量和用户体验。文章通过登录、列表和详情界面的实际案例展示了Swift与UIKit的结合应用。
|
11天前
|
存储 安全 Swift
【Swift 开发专栏】使用 Swift 开发一个简单的 iOS 应用
【4月更文挑战第30天】本文介绍了使用 Swift 开发简单 iOS 待办事项应用的步骤。首先,阐述了 iOS 开发的吸引力及 Swift 语言的优势。接着,详细说明了应用的需求和设计,包括添加、查看和删除待办事项的功能。开发步骤包括创建项目、界面搭建、数据存储、功能实现,并提供了相关代码示例。最后,强调了实际开发中需注意的细节和优化,旨在帮助初学者掌握 Swift 和 iOS 开发基础。
|
20天前
|
iOS开发 开发者 UED
利用SwiftUI构建动态列表:iOS开发的新范式
【4月更文挑战第22天】在本文中,我们将深入探讨如何使用SwiftUI来创建动态列表。SwiftUI是苹果最新推出的用户界面工具集,它允许开发者以声明式的方式描述用户界面,从而简化了代码的复杂性。我们将通过具体的代码实例,展示如何利用SwiftUI的List和ForEach视图来创建动态列表,并讨论其在实际开发中的应用。
18 2