iOS中block的定义与使用

概念

代码块block是苹果在iOS4开始引入的对C语言的扩展,用来实现匿名函数的特性,block是一种特殊的数据类型,其可以正常定义变量、作为参数、作为返回值,特殊地,block还可以保存一段代码,在需要的时候调用,目前block已经广泛应用于iOS开发中,常用于GCD、动画、排序及各类回调

block:我们称代码块,他类似一个方法。而每一个方法都是在被调用的时候从硬盘到内存,然后去执行,执行完就消失,所以,方法的内存不需要我们管理,也就是说,方法是在内存的栈区。所以,block不像OC中的类对象(在堆区),他也是在栈区的。如果我们使用block作为一个对象的属性,我们会使用关键字copy修饰他,因为他在栈区,我们没办法控制他的消亡,当我们用copy修饰的时候,系统会把该 block的实现拷贝一份到堆区,这样我们对应的属性,就拥有的该block的所有权。就可以保证block代码块不会提前消亡。

定义与使用

block 变量的声明

block 变量的声明格式:返回值类型(^block名字)(参数列表);

block 变量的声明格式:返回值类型(^block名字)(参数列表);
 //声明一个无返回值,两个参数的 block
 void(^block1)(NSString *a,NSString *b);
 //省略写法
 void(^block2)(NSString *,NSString *);

block变量的赋值

 //block变量的赋值
 //block变量名 = ^(参数列表){函数体};
 block1 = ^(NSString *x,NSString *y){
 NSLog(@"%@--%@",x,y);
 };
 block1(@"123123",@"QWEQWEQWE");

声明block变量的同时进行赋值

 //声明 block 变量的同时进行赋值
 int(^block3)(int) = ^(int a){
 return a*3;
 };
 NSLog(@"%d",block3(3));

使用typedef定义block类型

在实际使用Block的过程中,我们可能需要重复地声明多个相同返回值相同参数列表的Block变量,如果总是重复地编写一长串代码来声明变量会非常繁琐,所以我们可以使用typedef来定义Block类型

#pragma mark 使用 typedef 定义 block 类型
 //定义一个无返回值类型 无参数列表的 block
 typedef void (^Block4)();
 Block4 block4 = ^(){
  NSLog(@"i am block4");
 };
 block4();

block作为函数参数

#pragma mark block作为函数参数
 int(^block5)(int,int) = ^(int a,int b){
  return a+b;
 };
 [self useBlock5:block5];

 //简化书写
 //typedef int (^Block6)(int,int);(全局声明)
 Block6 block6 = ^(int a,int b){
  return a+b;
 };
 [self useBlock6:block6];

- (void)useBlock5:(int(^)(int,int))block5 {
 NSLog(@"block5:%d",block5(3,5));
}
- (void)useBlock6:(Block6 )block6{
 NSLog(@"block6:%d",block6(4,5));
}

block内访问局部变量

  • 在Block中可以访问局部变量
  • 在声明Block之后、调用Block之前对局部变量进行修改,在调用Block时局部变量值是修改之前的旧值
  • 在Block中不可以直接修改局部变量
#pragma mark block -----------访问局部变量
 //block 中不可以直接修改局部变量
 int value1 = 100;
 void (^block7)(void) = ^(){
   NSLog(@"value1:%d",value1);
 };
 value1 = 200;
 block7();//输出100 
  • 在局部变量前使用下划线下划线block修饰,在声明Block之后、调用Block之前对局部变量进行修改,在调用Block时局部变量值是修改之后的新值
  • 在局部变量前使用下划线下划线block修饰,在Block中可以直接修改局部变量
#pragma mark block -----------访问局部变量
 //block 中不可以直接修改局部变量
 //在局部变量前使用__block修饰,在Block中可以直接修改局部变量
 __block int value1 = 100;
 void (^block7)(void) = ^(){
   value1++;
  NSLog(@"value1:%d",value1);
 };
 value1 = 200;
 block7();//输出100 __block 修饰之后输出200

block内访问全局变量

在声明Block之后、调用Block之前对全局变量进行修改,在调用Block时全局变量值是修改之后的新值

#pragma mark ------------------block 访问全局变量
 //在Block中可以访问全局变量
 value2 = 100;
 void(^block8)(void) = ^(){
  //在Block中可以直接修改全局变量
  self->value2++;
  NSLog(@"value2:%d",self->value2);
 };
 value2 = 200;
 block8();

block内访问静态变量

#pragma mark --------block内访问静态变量
 static int value3 = 100;
 void(^block9)(void) = ^(){
  value3++;//在Block中可以直接修改静态变量
  NSLog(@"value3:%d",value3);
 };
 //在声明Block之后、调用Block之前对静态变量进行修改,在调用Block时静态变量值是修改之后的新值
 value3 = 200;
 block9();

block 造成的循环引用问题

如果对象内部有一个Block属性,而在Block内部又访问了该对象,那么会造成循环引用

情况一

@interface Person : NSObject

@property (nonatomic, copy) void(^myBlock)();

@end

@implementation Person

- (void)dealloc
{
 NSLog(@"Person dealloc");
}

@end

Person *p = [[Person alloc] init];

p.myBlock = ^{
 NSLog(@"------%@", p);
};
p.myBlock();

// 因为myBlock作为Person的属性,采用copy修饰符修饰(这样才能保证Block在堆里面,以免Block在栈中被系统释放),所以Block会对Person对象进行一次强引用,导致循环引用无法释放

情况二

@interface Person : NSObject

@property (nonatomic, copy) void(^myBlock)();

- (void)resetBlock;

@end

@implementation Person

- (void)resetBlock
{
 self.myBlock = ^{
  NSLog(@"------%@", self);
 };
}

- (void)dealloc
{
 NSLog(@"Person dealloc");
}

@end

Person *p = [[Person alloc] init];
[p resetBlock];

// Person对象在这里无法正常释放,在resetBlock方法实现中,Block内部对self进行了一次强引用,导致循环引用无法释放

解决循环引用的办法是使用一个弱引用的指针指向该对象,然后在Block内部使用该弱引用指针来进行操作,这样避免了Block对对象进行强引用

情况一

@interface Person : NSObject

@property (nonatomic, copy) void(^myBlock)();

@end

@implementation Person

- (void)dealloc
{
 NSLog(@"Person dealloc");
}

@end

Person *p = [[Person alloc] init];
__weak typeof(p) weakP = p;

p.myBlock = ^{
 NSLog(@"------%@", weakP);
};
p.myBlock();

// Person对象在这里可以正常被释放

情况二

@interface Person : NSObject

@property (nonatomic, copy) void(^myBlock)();

- (void)resetBlock;

@end

@implementation Person

- (void)resetBlock
{
 // 这里为了通用一点,可以使用__weak typeof(self) weakP = self;
 __weak Person *weakP = self;
 self.myBlock = ^{
  NSLog(@"------%@", weakP);
 };
}

- (void)dealloc
{
 NSLog(@"Person dealloc");
}

@end

Person *p = [[Person alloc] init];
[p resetBlock];

// Person对象在这里可以正常被释放

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对我们的支持。

(0)

相关推荐

  • 浅谈iOS 对于block的一点理解

    block是对象,它封装了一段代码,这段代码可以在任何时候执行.block可以作为函数参数或者函数的返回值,而其本身又可以带输入参数或返回值.它和传统的函数指针很类似,但是有区别:block是inline的,并且它对局部变量是只读的. Block的定义: int (^myBlock) (int a,int b) = ^(int a,int b){ return a+b; }; 定义了一个名为myBlock的blocks对象,它带有两个int参数,返回int.等式右边就是blocks的具体实现,是

  • iOS面试中如何优雅回答Block导致循环引用的问题

    前言 说到循环引用问题,最最最常遇到的,不是在项目中,而是在面试中.如果面试官问你开发中是否遇到过retain cycle,你如果说没遇到过,估计已经很难跟面试官继续友好的沟通下去了. 但是这个问题怎么回答呢,网络上千篇一律的答案-->使用Block的时候遇到过,使用__weakSelf 代替 self 等等,可以说这个答案没啥错,但是所有人都回答的一样,并不能突出我们的逼格,无法让面试官知道我们在这方面有过研究,有闪光点. 对于开发者来说,喜欢探索,喜欢挖掘不懂的知识,在面试官眼里会加分不少.

  • iOS利用Block逆向传值的方式详解

    前言 在iOS通过代理逆向传值的方式详解一文中,分析了如何利用代理模式来逆向传值,其实还有一些其他的方式,如通知.Block等,相比较代理,我个人认为反而要简单些,但是需要处理好细节问题,如Block循环引用.还是用前文的案例,本次使用Block来实现,Block的基本知识本文不再赘述. 一.书写规范 Block传值,需要注意的是,谁传值就需要定义Block,捕获方仅仅需要传递Block给传值方,并处理捕获的值. 传值方 1.定义Block用于传值 2.声明一个上述Block属性,这个属性的具体

  • iOS中Block的回调使用和解析详解

    Block 回调实现 先跟着我实现最简单的 Block 回调传参的使用,如果你能举一反三,基本上可以满足了 OC 中的开发需求.已经实现的同学可以跳到下一节. 首先解释一下我们例子要实现什么功能(其实是烂大街又最形象的例子): 有两个视图控制器 A 和 B,现在点击 A 上的按钮跳转到视图 B ,并在 B 中的textfield 输入字符串,点击 B 中的跳转按钮跳转回 A ,并将之前输入的字符串 显示在 A 中的 label 上.也就是说 A 视图中需要回调 B 视图中的数据. 想不明白的同学

  • iOS中block变量捕获原理详析

    Block概述 Block它是C语言级别和运行时方面的一个特征.Block封装了一段代码逻辑,也用{}括起,和标准C语言中的函数/函数指针很相似,此外就是blokc能够对定义环境中的变量可以引用到.这一点和其它各种语言中所说的"闭包"是非常类似的概念.在iOS中,block有很多应用场景,比如对代码封装作为参数传递.这在使用dispatch并发(Operation中也有BlockOperation)和completion异步回调等处都广泛应用. Block是苹果官方特别推荐使用的数据类

  • iOS(闭包)block传值详解

    在iOSAPP开发的过程中 我们会用到很多需要传值的地方 传值的方式也多种多样 有:代理传值.通知传值.KVC.KVO.block.单例 等.其中block 因为其简洁实用规范的代码 无疑是大牛们传值的不二选择 但对于初学者来说要理解并能运用 起初确实有些难以理解 以下我将细细的介绍下block 首先我总结了一下block的公式: 步骤1.block 的声明 返回值类型(^block 的名字)(参数列表); 步骤2.block 实现 block的名字 = ^(参数列表)(){}; 步骤3.blo

  • iOS在Block中修改外部变量值的实现代码

    一,代码. - (void)viewDidLoad { [super viewDidLoad]; // Do any additional setup after loading the view, typically from a nib. // 第一种是可以修改 static 全局变量:第二种是可以修改用新关键字 __block 修饰的变量. __block int blockLocal = 100; static int staticLocal = 100; void (^aBlock)(

  • iOS Block解开多年以来一直的误解

    首先来了解下什么是Block (1)Block是OC中的一种数据类型,在iOS开发中被广泛使用 (2)^是Block的特有标记 (3)Block的实现代码包含在{}之间 (4)大多情况下,以内联inline函数的方式被定义和使用 (5)Block与C语言的函数指针有些相似,但使用起来更加灵活 这张图是我在2015年的时候发现的新大陆,那时候也知道block是一种特殊的数据类型.也是一种特殊的对象(不同于NSObject).在执行的时候务必要先if判断一下,否者crach.我给一个block变量赋

  • iOS通过逆向理解Block的内存模型

    前言 正常情况下,通过分析界面以及 class-dump 出来头文件就能对某个功能的实现猜个八九不离十.但是 Block 这种特殊的类型在头文件中是看不出它的声明的,一些有 Block 回调的方法名 dump 出来是类似这样的: - (void)FM_GetSubscribeList:(long long)arg1 pageSize:(long long)arg2 callBack:(CDUnknownBlockType)arg3; 因为这种回调看不到它的方法签名,我们无法知道这个 Block

  • iOS通过block在两个页面间传值的方法

    一.功能需求 在第一个页面中有一个button和一个label,label上默认显示"哈哈",点击button进入第二个页面.在第二个页面有一个UITextField和一个button2,点击button2回到第一个页面,但同时第一个页面的label上显示的文字修改为刚刚在UITextField写进去的文字. 二.先定义block 在要传值得那个页面定义含有block参数的方法,即在第二个页面的.h文件中定义: 重新定义:typedef void (^ReturnTextBlock)(

随机推荐