Black 全面分析 如果有Block语法不懂的,可以参考fuckingblocksyntax,里面对于Block

为了方便对比,下面的代码我假设是写在ViewController子类中的 1、第一部分

定义和使用Block, ?

1 - (void)viewDidLoad 2{

3 4 5 6 7 8 9

10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
    [super viewDidLoad];

//(1)定义无参无返回值的Block void (^printBlock)() = ^(){

printf("no number"); };

    printBlock();
printBlock(9);

int mutiplier = 7; //(3)定义名为myBlock的代码块,返回值类型为int int (^myBlock)(int) = ^(int num){

return num*mutiplier; }

//使用定义的myBlock
int newMutiplier = myBlock(3);
printf("newMutiplier is %d",myBlock(3));

}

//定义在-viewDidLoad方法外部
//(2)定义一个有参数,没有返回值的Block
void (^printNumBlock)(int) = ^(int num){

printf("int number is %d",num);
};

定义Block变量,就相当于定义了一个函数。但是区别也很明显,因为函数肯
定是在-viewDidLoad方法外面定义,而Block变量定义在了viewDidLoad
方法内部。当然,我们也可以把Block定义在-viewDidLoad方法外部,例如
上面的代码块printNumBlock的定义,就在-viewDidLoad外面。

再来看看上面代码运行的顺序问题,以第(3)个myBlock距离来说,在定
义的地方,并不会执行Block{}内部的代码,而在myBlock(3)调用之后才
会执行其中的代码,这跟函数的理解其实差不多,就是只要在调用Block(函
数)的时候才会执行Block体内(函数体内)的代码。所以上面的简单代码示
例,我可以作出如下的结论,

(1)在类中,定义一个Block变量,就像定义一个函数;

(2)Block可以定义在方法内部,也可以定义在方法外部;

(3)只有调用Block时候,才会执行其{}体内的代码;

(PS:关于第(2)条,定义在方法外部的Block,其实就是文件级别的全局
变量)

那么在类中定义一个Block,特别是在-viewDidLoad方法体内定义一个
Block到底有什么意义呢?我表示这时候只把它当做私有函数就可以了。我之
前说过,Block其实就相当于代理,那么这时候我该怎样将其与代理类比以了
解呢。这时候我可以这样说:本类中的Block就相当于类自己服从某个协议,
然后让自己代理自己去做某个事情。很拗口吧?看看下面的代码,

?

2
3
4
5
6
7
8
9

10

1 //定义一个协议

@protocol ViewControllerDelegate<NSObject>
- (void)selfDelegateMethod;
@end

//本类实现这个协议ViewControllerDelegate
@interface ViewController ()<ViewControllerDelegate>
@property (nonatomic, assign) id<ViewControllerDelegate> delegate;

@end

接着在-viewDidLoad中的代码如下,
?

1 - (void)viewDidLoad
2{

  1. 3  [super viewDidLoad];

  2. 4  // Do any additional setup after loading the view from its nib.

  3. 5  self.delegate = self;

  4. 6  if (self.delegate && [self.delegate respondsToSelector:@selector

7
8}
9}

[self.delegate selfDelegateMethod];

10
11 #pragma mark - ViewControllerDelegate method

//实现协议中的方法
- (void)selfDelegateMethod
{

NSLog(@"自己委托自己实现的方法");
}

看出这种写法的奇葩地方了吗?自己委托自己去实现某个方法,而不是委托
别的类去实现某个方法。本类中定义的一个Block其实就是闲的蛋疼,委托自
己去字做某件事情,实际的意义不大,所以你很少看见别人的代码直接在类
中定义Block然后使用的,Block很多的用处是跨越两个类来使用的,比如作
为property属性或者作为方法的参数,这样就能跨越两个类了。

2、第二部分
__block关键字的使用

在Block的{}体内,是不可以对外面的变量进行更改的,比如下面的语句,
?

1 - (void)viewDidLoad
2{

3 //将Block定义在方法内部

12
13
14
15
16

4
5
6
7
8

int x = 100;
void (^sumXAndYBlock)(int) = ^(int y){
x = x+y;
printf("new x value is %d",x);
};

9 sumXAndYBlock(50);
10 }

这段代码有什么问题呢,Xcode会提示x变量错误信息:Variable is not
assigning (missing __block type),这时候给int x = 100;语句前面加上
__block关键字即可,如下,

?

这样在Block的{}体内,就可以修改外部变量了。
3、第三部分:Block作为property属性实现页面之间传值

需求:在ViewController中,点击Button,push到下一个页面
NextViewController,在NextViewController的输入框TextField中输入
一串字符,返回的时候,在ViewController的Label上面显示文字内容,

(1)第一种方法:首先看看通过“协议/代理”是怎么实现两个页面之间传值
的吧,

?

2
3
4
5
6
7
8
9

  10
11
12
13
14
15
16
17
18

1 __block int x = 100;

1 //NextViewController是push进入的第二个页面

//NextViewController.h 文件 //定义一个协议,前一个页面ViewController要服从该协议,并且实现协议中的方

@protocol NextViewControllerDelegate <NSObject> - (void)passTextValue:(NSString *)tfText;
@end

@interface NextViewController : UIViewController
@property (nonatomic, assign) id<NextViewControllerDelegate> delegat

@end

//NextViewController.m 文件
//点击Button返回前一个ViewController页面
- (IBAction)popBtnClicked:(id)sender {

if (self.delegate && [self.delegate respondsToSelector:@selector
//self.inputTF是该页面中的TextField输入框

19 [self.delegate passTextValue:self.inputTF.text];
20 }

21

接下来我们在看看ViewController文件中的内容,
?

//ViewController.m 文件
@interface ViewController ()<NextViewControllerDelegate>
@property (strong, nonatomic) IBOutlet UILabel *nextVCInfoLabel;

@end

//点击Button进入下一个NextViewController页面
- (IBAction)btnClicked:(id)sender
{

}

[self.navigationController popViewControllerAnimated:YES];

1
2
3
4
5
6
7
8
9

  10
11
12
13
14
15
16
17
18
19
20

这是通过“协议/代理”来实现的两个页面之间传值的方式。 (2)第二种方法:使用Block作为property,实现两个页面之间传值,

先看看NextViewController文件中的内容, ?

//NextViewController.h 文件
@interface NextViewController : UIViewController
@property (nonatomic, copy) void (^NextViewControllerBlock)(NSString

1
2
3
4

NextViewController *nextVC = [[NextViewController alloc] initWit
nextVC.delegate = self;//设置代理
[self.navigationController pushViewController:nextVC animated:YE

}

//实现协议NextViewControllerDelegate中的方法
#pragma mark - NextViewControllerDelegate method
- (void)passTextValue:(NSString *)tfText
{

//self.nextVCInfoLabel是显示NextViewController传递过来的字符串Labe

    self.nextVCInfoLabel.text = tfText;
}

5 @end

//NextViewContorller.m 文件
- (IBAction)popBtnClicked:(id)sender {

6
7
8
9

10
11
12

再来看看ViewController文件中的内容,
?

1 - (IBAction)btnClicked:(id)sender
2{

  1. 3  NextViewController *nextVC = [[NextViewController alloc] initWit

  2. 4  nextVC.NextViewControllerBlock = ^(NSString *tfText){

  3. 5  [self resetLabel:tfText];

  4. 6  };

  5. 7  [self.navigationController pushViewController:nextVC animated:YE

8}

  1. 9  #pragma mark - NextViewControllerBlock method

  2. 10  - (void)resetLabel:(NSString *)textStr

  3. 11  {

  4. 12  self.nextVCInfoLabel.text = textStr;

  5. 13  }

好了就这么多代码,可以使用Block来实现两个页面之间传值的目的,实际上
就是取代了Delegate的功能。

另外,博客中的代码Sample Code可以再Github下载,如果因为Github
被墙了,可以在终端使用git clone + 完整链接,即可克隆项目到本地。

Github中的代码,可以开启两种调试模式,你需要在项目的配置文件
BlockSamp-Prefix.pch中注释或者解注释下面的代码,

?

if (self.NextViewControllerBlock) {
self.NextViewControllerBlock(self.inputTF.text);

}

[self.navigationController popViewControllerAnimated:YES];
}

1 #define Debug_BlcokPassValueEnable

即可开启两种调试的方式,如果注释了上面的语句就是使用Delegate进行调
试;否则使用Block进行调试。

Black 全面分析的更多相关文章

  1. alias导致virtualenv异常的分析和解法

    title: alias导致virtualenv异常的分析和解法 toc: true comments: true date: 2016-06-27 23:40:56 tags: [OS X, ZSH ...

  2. 火焰图分析openresty性能瓶颈

    注:本文操作基于CentOS 系统 准备工作 用wget从https://sourceware.org/systemtap/ftp/releases/下载最新版的systemtap.tar.gz压缩包 ...

  3. 一起来玩echarts系列(一)------箱线图的分析与绘制

    一.箱线图 Box-plot 箱线图一般被用作显示数据分散情况.具体是计算一组数据的中位数.25%分位数.75%分位数.上边界.下边界,来将数据从大到小排列,直观展示数据整体的分布情况. 大部分正常数 ...

  4. 应用工具 .NET Portability Analyzer 分析迁移dotnet core

    大多数开发人员更喜欢一次性编写好业务逻辑代码,以后再重用这些代码.与构建不同的应用以面向多个平台相比,这种方法更加容易.如果您创建与 .NET Core 兼容的.NET 标准库,那么现在比以往任何时候 ...

  5. UWP中新加的数据绑定方式x:Bind分析总结

    UWP中新加的数据绑定方式x:Bind分析总结 0x00 UWP中的x:Bind 由之前有过WPF开发经验,所以在学习UWP的时候直接省略了XAML.数据绑定等几个看着十分眼熟的主题.学习过程中倒是也 ...

  6. 查看w3wp进程占用的内存及.NET内存泄露,死锁分析

    一 基础知识 在分析之前,先上一张图: 从上面可以看到,这个w3wp进程占用了376M内存,启动了54个线程. 在使用windbg查看之前,看到的进程含有 *32 字样,意思是在64位机器上已32位方 ...

  7. ZIP压缩算法详细分析及解压实例解释

    最近自己实现了一个ZIP压缩数据的解压程序,觉得有必要把ZIP压缩格式进行一下详细总结,数据压缩是一门通信原理和计算机科学都会涉及到的学科,在通信原理中,一般称为信源编码,在计算机科学里,一般称为数据 ...

  8. ABP源码分析一:整体项目结构及目录

    ABP是一套非常优秀的web应用程序架构,适合用来搭建集中式架构的web应用程序. 整个Abp的Infrastructure是以Abp这个package为核心模块(core)+15个模块(module ...

  9. HashMap与TreeMap源码分析

    1. 引言     在红黑树--算法导论(15)中学习了红黑树的原理.本来打算自己来试着实现一下,然而在看了JDK(1.8.0)TreeMap的源码后恍然发现原来它就是利用红黑树实现的(很惭愧学了Ja ...

  10. ElasticSearch 5学习(9)——映射和分析(string类型废弃)

    在ElasticSearch中,存入文档的内容类似于传统数据每个字段一样,都会有一个指定的属性,为了能够把日期字段处理成日期,把数字字段处理成数字,把字符串字段处理成字符串值,Elasticsearc ...

随机推荐

  1. poj2074Line of Sight(直线相交)

    链接 几何细节题. 对于每一个障碍物可以求出它在地产线上的覆盖区间,如下图. 紫色部分即为每个障碍物所覆盖掉的区间,求出所有的,扫描一遍即可. 几个需要注意的地方:直线可能与地产线没有交点,可视区间可 ...

  2. unsigned 整型实现无溢出运算

    普通的 int 整型能表示的范围很有限,所以刷题时很多时候不得不用 long long 来存更大的数据.或者找出数列中某个只出现一次(或奇数次)的数(其余的数均出现两次 / 偶数次),用异或运算的经典 ...

  3. 线程池的原理及实现 (zhuan)

    http://blog.csdn.net/hsuxu/article/details/8985931 ************************************************* ...

  4. JPA基础

    目录 目录 1 一.JPA基础 2 1.1 JPA基础 2 1.2JPA开发过程 3 1.3 实体的生命周期及实体管理器常用方法 4 二.环境搭建 5 2.1 添加JPA支持 6 2.2 添加配置文件 ...

  5. javascript AOP实现

    参考:http://www.cnblogs.com/rubylouvre/archive/2009/08/08/1541578.html function Person(){ this.say = f ...

  6. No resource found that matches the given name 'Theme.AppCompat.Light'

    eclipse在新建andorid工程的时候出现找不到AppCompat.Light主题的问题,这是因为缺少支持低版本的v7包所致, 这个不是jar包,在sdk的extras->andorid- ...

  7. 创建和使用Windows静态链接库

    首先明确这篇文章的目的,我希望大家能够通过这篇文章了解一下如何在实际工作中创建和使用Windows平台下的静态链接库.关于链接库的概念,希望大家参考维基百科"Library"词条( ...

  8. input上传图片预览

    <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <t ...

  9. jmeter笔记5

    性能测试是任何分布式或Web应用程序测试计划的重要组成部分.在计划和开发周期中进行性能评价,可以保证交付给客户的应用程序满足客户对于高负载.可用性和可伸缩性的要求.提前确定软件的负载限制可以为适当地进 ...

  10. 例子:使用C++中的this

    在C++中很多的东西都传值的,. C++中的对象之间的copy是传值的 , 他不想java那样,对象之间传递的引用 , 或者说是java对指针进行了封装 , 禁止了一些不安全的操作 对于C++而言 , ...