[转]blocks编程
原文地址:http://geeklu.com/2012/01/block/
- 介绍
- 声明创建和调用
- Block和变量
- Block实际应用
1.介绍
Block是一个C Level的语法以及运行时的一个特性,非常像标准C中的函数(函数指针),但是其运行需要编译器和运行时支持,目前LLVM+Clang可以很好的支持Block(苹果修改过的GCC也可以)。Block和函数不同的是其语义 闭包 特性,以及可以有匿名block的存在。 你可以在LLVM的官方网站查看Block语言规范.
你可以通过^
运算符来声明一个block变量,或用来表明block定义的开始,而block的代码块则是包含在一对花括号{}
内的.
int multiplier = 2;
int (^myBlock)(int) = ^(int num){
return num * multiplier;
};
printf("%d",myBlock(4));
上面代码中的myBlock
就是Block的变量名,由myBlock变量的声明可以看出,它返回值为int类型,且存在一个int型的参数。 等于号后面就是Block的定义并将其赋值给myBlock . Block的调用就和C函数的使用类似.
2. 声明创建和调用
声明Block变量
Block变量保存着指向Block的指针,声明一个Block变量就和声明一个函数指针变量类似,只是将*改成了^.
其他的就和C的类型系统保持一致了。
void (^blockReturningVoidWithVoidArgument)(void);
int (^blockReturningIntWithIntAndCharArguments)(int, char);
void (^arrayOfTenBlocksReturningVoidWithIntArgument[10])(int);
另 Block还支持可变参数variadic (...) ,没有参数的话,变量列表的地方必须写上void关键字.
声明Block类型 你可以通过typedef声明Block的类型,这样多个地方需要使用同种类型的Block的时候会比较方便,
typedef float (^MyBlockType)(float, float);
MyBlockType myFirstBlock = // ... ;
MyBlockType mySecondBlock = // ... ;
Block创建 声明了一个Block变量之后,可以为这个变量赋值
blockReturningVoidWithVoidArgument = ^{
printf("%s","Block Returing Void With Void Argument.");
}
当然你可以在声明变量的同时赋值
void (^blockReturningVoidWithVoidArgument)(void) = ^{
printf("%s","Block Returing Void With Void Argument.");
}
Block调用 Block的调用和函数的调用是非常相似的,如上面定义的blockReturningVoidWithVoidArgument,调用的时候则直接 blockReturningVoidWithVoidArgument();
便可.
匿名Block 当一个Block作为函数参数的时候,一般实参都是以匿名Block的方式传过去的。
void callVoidVoid(void (^closure)(void)) {
closure();
}
int main(int argc, char *argv[]) {
__block int i = 10;
callVoidVoid(^{ ++i; });
if (i != 11) {
printf("*** %s didn't update i\n", argv[0]);
return 1;
}
printf("%s: success\n", argv[0]);
return 0;
}
当然你也可以直接调用匿名Block,如
^{ ++i; }();
3.Block和变量
一个Block的内部是可以引用自身作用域外的变量的,包括static变量,extern变量或自由变量(定义一个变量的时候,如果不加存储修饰 符,默认情况下就是自由变量auto,auto变量保存在stack中的.除了auto之外还存在register,static等存储修饰符) ,对于自由变量,在Block中是只读的。在引入block的同时,还引入了一种特殊的变量存储修饰符__block
,通过它 的变量叫做block变量,block变量在block内部可以进行写操作的。这些变量中,自由变量是最特殊的,在Block声明的时候,自由变量在 Block内部只读且其值被固定住(自由变量被拷贝了一份,且限定为const),即使在block调用前改变了这个自由变量的值,block调用的时 候,看到的却还是block声明的时候的那个值。
代码示例 3.1
__block int blockValue = 0;
int autoValue = 0;
void(^printValue)(void) = ^{
printf("blockValue = %d\n",blockValue);
printf("autoValue = %d\n",autoValue);
};
blockValue ++;
autoValue ++;
printValue();
3.1中的代码,输出的值为
blockValue = 1
autoValue = 0
可以看到自由变量尽管自增了,但是在调用printValue这个Block的时候,看到的还是其定义的时候看到的那个autoValue的值,autoValue的值在Block的内部示无法修改的,要不然编译器会报错:
Semantic Issue: Variable is not assignable (missing __block type specifier)
Block定义时内存是分配在stack上的,当其作用域结束,就会被自动释放,所以你不需要去担心它的内存情况,我们可以对一个Block进行Block_copy()
操作,Block_copy()
之后,Block会被拷贝到heap中的内存中,且其所有的引用到的自由变量也将会被拷贝,当然你得记得通过Block_release()
释放heap的内存空间哦。 在objc中Block是和对象一样被看作一等公民的(其实这是objc的Block扩展的功劳),你可以像使用对象那样对Block进行retain(retain只对heap中的Block起作用),copy以及release操作.
在Block内部如果引用到对象或者对象的成员变量,那么当Block被拷贝Block_copy()
之后,这个对象的引用计数会增加。
代码示例 3.2
NSObject *testObject = [[NSObject alloc] init];
NSLog(@"%lu",[testObject retainCount]); //1
NSLog(@"%lu",[self retainCount]); //1
void(^testBlock)(void) = ^{
NSLog(@"The Test String : %@", testObject);
NSLog(@"The Window Object : %@", _window);
};
NSLog(@"%lu",[testObject retainCount]); //1
NSLog(@"%lu",[self retainCount]); //1
void(^testBlock2)(void) = Block_copy(testBlock); //testBlock会被拷贝到heap中,所以用完了要自己调用Block_release进行释放
NSLog(@"%lu",[testObject retainCount]); //2
NSLog(@"%lu",[self retainCount]); //2
testBlock2();
Block_release(testBlock2);
NSLog(@"%lu",[testObject retainCount]); //1
NSLog(@"%lu",[self retainCount]); //1
[testObject release];
Block的闭包特性使得Block可以脱离其定义的作用域进行运行,所以你可以在一个函数中返回一个Block,在别的线程或者当前线程的RunLoop中进行运行,而不用担心那些引用到的外部变量是否被释放掉了。
4.Block实际应用
那么我们一般什么时候会用到Block呢? Blocks通常是一小段自包含的代码片段.所以它经常被用于多线程运行的代码单元(如GCD),或用于处理聚合类元素单元,或者作为某个函数调用完成后的回调函数.
Block用作回调函数比传统的回调函数有以下的优越性:
- 在函数调用的时候,将Block作为一个参数传给函数
- 允许访问本地变量,这样可以避免通过结构体将本地变量封装后传递给回调函数
应用1: Animations & Completion Handler
[UIView animateWithDuration:2
animations:^{
self.view.backgroundColor = [UIColor redColor];
}
completion:^(BOOL finished){
if (finished){
self.view.backgroundColor = [UIColor blueColor];
}
}];
应用2: Enumeration
对数据集合类中的每一个元素进行遍历,每次传入一个对象,进行处理
NSArray *cards = [NSArray arrayWithObjects:@"Jack", @"Queen", @"King", @"Ace", nil];
[cards enumerateObjectsUsingBlock:^(id object, NSUInteger index, BOOL *stop) {
NSLog(@"%@ card at index %d", object, index);
}];
应用3: Notification Handler
[[NSNotificationCenter defaultCenter]
addObserverForName:@"TestNotification"
object:nil
queue:aNSOperationQueue
usingBlock:^(NSNotification *notification){
NSLog(@"Notification: %@",notification);
}];
应用4: GCD
dispatch_queue_t imageDownloadQueue = dispatch_queue_create("Image Download Queue", NULL);
dispatch_async(imageDownloadQueue, ^{
NSURL *imageURL = [NSURL URLWithString:@"http://xxx.xx.com/a.png"];
NSData *imageData = [NSData dataWithContentsOfURL:imageURL];
UIImage *image = [UIImage imageWithData:imageData];
dispatch_async(dispatch_get_main_queue(), ^{
[imageView setImage:image];
});
});
[转]blocks编程的更多相关文章
- Blocks编程要点
[老狼推荐]Blocks编程要点原文:Blocks Programming Topics链接:http://developer.apple.com/library/ios/#documentation ...
- .Net中的并行编程-5.流水线模型实战
自己在Excel整理了很多想写的话题,但苦于最近比较忙(其实这是借口).... 上篇文章<.Net中的并行编程-4.实现高性能异步队列>介绍了异步队列的实现,本篇文章介绍我实际工作者遇到了 ...
- Instruments 使用指南
Instruments 用户指南 http://cdn.cocimg.com/bbs/attachment/Fid_6/6_24457_90eabb4ed5b3863.pdf 原著:Apple Inc ...
- 反汇编objc分析__block
"You can specify that an imported variable be mutable—that is, read-write— by applying the __bl ...
- objc反汇编分析,手工逆向libsystem_blocks.dylib
上一篇<block函数块为何物?>介绍了在函数中定义的block函数块的反汇编实现,我在文中再三指出__block变量和block函数块自始还都是stack-based的,还不完全适合在离 ...
- 体验Code::Blocks下的C++编程
0.前言 在当前的行业发展和国际形势下,让更多的程序员思考跨平台编程问题.在众多的跨平台开发环境中,Code::Blocks具有独特的优势. 近二十年来,跨平台开发环境曾经如雨后春笋般产生,但是,由于 ...
- 体验Code::Blocks下的Windows GUI编程(32 bit and 64 bit)
0. 前言 不知道为什么,几乎所有的C++编程教程都是从命令行版(控制台应用程序)的Hello World开始的.然而,对于现在的年轻人来说,从小就用鼠标点击窗口或图标操作电脑,对于那个神秘黑色的窗口 ...
- (译文)IOS block编程指南 4 声明和创建blocks
Declaring and Creating Blocks (声明和创建blocks) Declaring a Block Reference (声明一个block引用) Block variable ...
- Python(七)Socket编程、IO多路复用、SocketServer
本章内容: Socket IO多路复用(select) SocketServer 模块(ThreadingTCPServer源码剖析) Socket socket通常也称作"套接字" ...
随机推荐
- ng2 quickstart-primeng
1.导入quickstart-angular项目 2.安装primeng npm install primeng 3.安装@angular/animations npm install @angula ...
- Linux基础知识点
1.目录结构 bin 存放二进制可执行文件(ls,cat,mkdir等) boot 存放用于系统引导时使用的各种文件 dev 用于存放设备文件 etc 存放系统配置文件 home 存放所有用户文件的根 ...
- 一个带展示的jsp上传控件模型
带展示上传控件的基本模型,无样式 jsp部分: <td> <form id="form1" enctype="multipart/form-data&q ...
- arm-linux-gcc安装使用教程
arm-linux-gcc如何下载安装2(转) [转]ubuntu下交叉编译环境构建(arm-linux-gcc-3.4.1.tar.bz2 ) 2009-03-03 10:05 1.下载arm-li ...
- Python基础笔记系列十:模块
本系列教程供个人学习笔记使用,如果您要浏览可能需要其它编程语言基础(如C语言),why?因为我写得烂啊,只有我自己看得懂!! 模块 #1.类比于java中的jar包,模块能让你能够有逻辑地组织你的Py ...
- Nordic老版官网介绍(2018-11-30停止更新)
1. Nordic官网及资料下载 Nordic官网主页:https://www.nordicsemi.com/,进入官网后,一般点击“Products”标签页,即进入Nordic产品下载首页,其独立链 ...
- php redis 秒杀demo
$redis = new Redis(); $redis->connect("127.0.0.1", "6379"); $redis->select ...
- 设计模式--享元模式C++实现
1定义 使用共享对象可有效的支持大量细粒度的对象 2类图 角色分析 Flyweight抽象享元角色,一个产品的抽象,定义内部状态和外部状态的接口或者实现 ConcreteFlyweight具体享元角色 ...
- 十四 web爬虫讲解2—Scrapy框架爬虫—豆瓣登录与利用打码接口实现自动识别验证码
打码接口文件 # -*- coding: cp936 -*- import sys import os from ctypes import * # 下载接口放目录 http://www.yundam ...
- 原生javascript-日历插件编写
在线实例:http://lgy.1zwq.com/calendar/ 按照我们常用的日历格式,是7*6的格子,所以生成格子的总数就确定为42 例子:(如:2013年8月,这个时间为例子) /*---- ...