Object-c 中的block就好像一段C函数般,由函数名,有返回值,有参数,由函数体等

1.简单的block

 ^(int A ,int B)
{
int C=A*B;
return C;
};

上述代码表示block有两个整形参数A和B.在block体中进行A和B的相乘,将结果作为block的返回值返回出去。

2.将block作为参数的API

在程序开发时,当需要一个NSArray对象的所有元素进行遍历时,除了for循环,开发者可以使用block进行遍历,代码如下:

  NSArray *arrChar=[@"A/B/C/D/E/F" componentsSeparatedByString:@"/"];
//遍历元素
[arrChar enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
NSLog(@"\nindex:[%d], value:[%@]",idx,obj);
if(idx == )
{
*stop=YES;
}
}];

这段代码运行后,在控制台上留下了如下日志:

index:[0], value:[A]
index:[1], value:[B];
index:[2], value:[C];
index:[3], value:[D];
index:[4], value:[E];

这里,NSArray对象所使用的API主要是一个对自己元素的遍历功能,enumerateObjectsUsingBlock方法的参数需要是一个block对象,在NSArray的帮助文档中,我们找到了enumerateObjectsUsingBlock的声明,内容如下:

-(void)enumerateObjectsUsingBlock:(void)(^(id obj,NSInterger idx,BOOL *stop))block

所传入的3个参数中,前两个是表明当前元素指针和序号的输入参数,开发者直接拿来使用即可。最后一个参数讲一个布尔类型的指针传递进来,显然是属于一个典型的输入参数。这个参数让开发者可以灵活的控制遍历的继续执行是否。

3.block的声明

block在声明时,需要遵循如下所示的格式结构。

<返回值类型> + (^<block名字>) + (<参数类型1>, <参数类型2>...)

根据enumerateObjectsUsingBlock的结构,我们声明的block变量对象如下:

void (^arrayEnumerateBlock)(id,NSUInteger,BOOL *);

对于声明完的block变量,可以直接进行初始化,也可以进行赋值。block的初始化格式如下:

void (^arrayEnumerateBlock)(id,NSUInteger,BOOL *)=^(id anObject, NSUInteger index, BOOL *isStop)
{
NSLog(@"\nindex:[%d], value:[%@]", index, anObject); //序号4, 则停止遍历
if(index == )
{
*isStop=YES;
}
};

上述代码中对于变量的初始化,开发者同样需要加上“^”符号表明这是一个block类型。不过block的返回值和block名不用写明,block名作为声明的一部分已经在声明时写明,至于返回值,只需要在block体中严格遵守返回类型即可。

4.block 的 typedef

在平时开发中,我们习惯于使用typedef定义属于自己框架的东西,本质上其实是为了类型使用更加统一。

而block由于有着相对复杂的声明方式,不如可以考虑将block的格式进行typedef,之后所有需要遵循这个block格式的对象声明再也不需要考虑格式的撰写,并且由于block有着函数指针般的接口协议特征,使用typedef定义过的block格式,当作为接口提供外部调用时,调用者可以明确具体需要给与的参数格式。

同样是enumerateObjectsUsingBlock所用的那个block格式,typedef定义示例如下

typedef void (^ArrayEnumerateBlockType)(id ,NSUInteger, BOOL *);

虽然typedef后接着的格式和前一小节中block声明格式相当相似,不过typedef所书写的内容是类型名字,而block声明时所书写的是变量对象的名字。所以在这里,我们特意以ArrayEnumerateBlockType定义类型名和之前的arrayEnumerateBlock变量类型名加以区分。

一旦有了typedef过得block类型后,到哪里都能够简单的使用这个block了,初始化代码如下:

ArrayEnumerateBlockType aEnumerateBlock=^(id aObject, NSUInteger index, BOOL *isStop)
{
//跟前面一样
}

5.block体的外部变量使用的奇怪之处

首先,让我们来看看以下代码。

     NSUInteger result   = ;
NSUInteger changeValue=;
//block变量声明
NSUInteger (^testReturnValueBlock)(NSUInteger, NSUInteger) = ^(NSUInteger param1, NSUInteger param2)
{
return param1+param2+changeValue;
}; result=testReturnValueBlock(,);
NSLog(@"1.[%d]",result); changeValue++;
result=testReturnValueBlock(,);
NSLog(@"2.[%d]",result);

执行结果却是

1.[3]
2.[3]

是不是觉得相当奇怪,我明明在第一次日志打印后对changeValue变量执行了++操作,其实打印的结果并没有错,我们需要更深入的了解block内部机制才能够看懂这其中的奥妙。

在block体中,我们不仅能够访问到block声明的传入参数,也能够正常访问到block以外的变量,不过,对于block以外的变量来说,如果把它放在block内进行访问也罢赋值也罢,一旦进入了block体中,基本类型变量会被block进行一次copy后以一个临时变量存放在block体中,而指针变量会被block进行一次retain后也以一个临时变量存放起来,无论是基本数据类型还是指针,被block使用了,就表明他的生命周期除了自己本身所在的作用域,又多了一个block体的作用域,也就是说:

(1)基本数据类型在block中的地址已经发生变化,所以block体外对于此数据类型的值修改对于体内的值毫无影响。

(2)block所copy或者retain的变量,一旦block结束,也就一起跟着被释放和销毁了。

(3)所谓的block会进行retain的指针类型,也包含Object-c中的所有对象。

6.克服外部变量的魔咒

(1)可以把外部变量修改成  static NSUInteger changeVaue=0;

static 关键字意味着changeValue的地址不再被我们放置于栈中,不过也并不在堆中,而是放在全局数据区,拥有一个永远不会改变的地址。

(2)也可以在外部变量前加上 __block   如:  __block NSUInteger changeValue=0;

当一个变量被__block所修饰时,block体中就会知道,即使使用到这个外部变量,也坚决不会去进行retain或者copy,这样就能够保证内外统一了

Object-c Block的使用及说明的更多相关文章

  1. iOS通知中心升级 -可设置按优先级执行block

    简单介绍下,这是需求驱动中发现iOS的NotificationCenter有很多功能无法实现,于是对其进行了一层包装.相当于手动管理观察者栈和监听者期望执行的事件,因此可以为其添加了很多新增的功能,将 ...

  2. block的语法

    主要内容: 1. 开始使用block(Getting Started with Blocks) 2. block概念综述(Conceptual Overview) 3. 声明和创建block(Decl ...

  3. Core Java Volume I — 4.6. Object Construction

    4.6. Object ConstructionYou have seen how to write simple constructors that define the initial state ...

  4. ios专题 -block用法

    what is block Blocks are a language-level feature added to C, Objective-C and C++, which allow you t ...

  5. (译文)IOS block编程指南 3 概念总览

    Conceptual Overview(概览) Block objects provide a way for you to create an ad hoc function body as an ...

  6. 简洁的KVO -- 使用Block响应事件

    涉及内容: KVO,Runtime,Category,Block 首先创建NSObject的Category 举个例子是这样的: 随后定义你需要响应的Block结构 我简单一点就这样咯 typedef ...

  7. 【原】iOS学习之PINCache第三方缓存框架

    在项目中总是需要缓存一些网络请求数据以减轻服务器压力,业内也有许多优秀的开源的解决方案.通常的缓存方案都是由内存缓存和磁盘缓存组成的,内存缓存速度快容量小,磁盘缓存容量大速度慢可持久化. 1.PINC ...

  8. Functional Programming without Lambda - Part 2 Lifting, Functor, Monad

    Lifting Now, let's review map from another perspective. map :: (T -> R) -> [T] -> [R] accep ...

  9. 相克军_Oracle体系_随堂笔记005-Database buffer cache

    本章主要阐述SGA中的Database buffer cache. Buffer cache { 1.pin (读写瞬间的状态) 2.clean 3.dirty 4.unused } --Databa ...

  10. IOS开发基础知识--碎片28

    1:通用的weakify和strongify /** * 强弱引用转换,用于解决代码块(block)与强引用self之间的循环引用问题 * 调用方式: `@weakify_self`实现弱引用转换,` ...

随机推荐

  1. Android中实现ListView圆角效果[转]

    本文演示如何Android中实现ListView圆角效果. 无论是网站,还是APP,人们都爱看一些新颖的视图效果.直角看多了,就想看看圆角,这几年刮起了一阵阵的圆角设计风:CSS新标准纳入圆角元素,特 ...

  2. Ubuntu下用glade和GTK+开发C语言界面程序(三)——学习make的使用方法

    makefile的规则 makefile的规则例如以下: target ... : prerequisites ... command ... ... target能够是一个object file(目 ...

  3. 【iOS】Plist-XML-JSON数据解析

    网络上数据传输通用的有XML.JSON等,iOS中也能够用Plist. 要进行传输数据.就要首先进行序列化: 1.序列化.  对象转换成二进制流.(这个一句话即可) .反序列化. 二进制流转换为对象等 ...

  4. xcode xib 加载 、注意点

    加载xib2中方式 NSArray *array = [[NSBundle mainBundle] loadNibNamed:@"xib名称" owner:nil options: ...

  5. HTML之学习笔记(六)添加链接

    html添加链接所用的标签为<a>标签 语法: 定义:从当前页面,跳转到指定页面或文件的一个标签            <a href="URL">热点文字 ...

  6. 前端新人学习笔记-------html/css/js基础知识点

    即将毕业的软件工程大学生一枚,秋季招聘应聘的是Android,今年来到公司实习,要求做前端开发,所以一切只有现学,现在根据视频来学习,然后开这个博客记录一下自己的学习过程,废话不多说,开写. 4月6日 ...

  7. 编写可维护的JS 04

    4.变量.函数和运算符 变量 变量声明提前,单var 函数声明 先声明fn再执行 函数声明不应出现在语句块中 函数调用间隔 函数名与左括号间无间隔 立即调用函数 (fuction(){}) 严格模式  ...

  8. MySQL学习笔记(3)

    约束 作用:保证数据的完整性,唯一性 根据字段:分为表级约束(针对2个或者2个以上字段使用),列级约束(针对1个字段使用) 约束类型:NOT NULL 非空约束 PRIMARY KEY  主键约束 U ...

  9. [学习笔记]viewport定义,弹性布局,响应式布局

    一,移动端宽度设置viewport视图窗口,<meta name="viewport" content="width=device-width,initial-sc ...

  10. ASPxComboBox-通过回车过滤结果集

    Dev ASP.NET组件中的ASPxComboBox可以方便的根据输入内容进行过滤,不过对于数据量较大或者用户数较多的情况下,这个功能会给服务器带来严重的负担,因此我们应该输入自己想要查询的字符串时 ...