IOS中的Block与C++11中的lambda

ios中的block 可以说是一种函数指针,但更确切的讲,其实际上其应该算是object-c对C++11中lambda的支持或者说是一个语言上的变体,其实际内容是一样的,C++的lambda我已经有简介过,现在说下ios中的block

Block的实际行为和Function很像,最大的差别是在可以存取同一个Scope的变量值。Block实体形式如下:

^(传入参数列){行为主体};

Block实体开头是“^”,接着是由小括号所包起来的参数列(比如 int a, int b, int c),return a*a;};  
这是代表Block会回传输入值的平方值(int a 就是参数列, return a*a; 就是行为主体)。记得行为主体里最后要加“;”,因为是叙述,而整个{}最后也要加“;”,因为Block是物件实体。用法如下:

  1. int result = ^(int a){return a*a;}(5);
  2. NSLog(@"%d", result);

很奇怪吧?后面的小括号里面的5会被当成a的输入值,然后经由Block输出 5*5 = 25指定给result这个变量。

有没有简单一点的方法嗯?不然每次都写这么长?有。接下来介绍一个叫做Block Pointer的东西来简化我们的写法。

Block Pointer是这样定义的:

回传值(^名字)(参数列);

比如下面的例子:

  1. //声明一个square的Block Pointer,其所指向的Block有一个int输入和int输出
  2. int (^square)(int);
  3. //将Block实体指定给square
  4. square = ^(int a){ return a*a ; };
  5. //调用方法,感觉是是不是很像function的用法?
  6. int result = square(5);
  7. NSLog(@"%d", result);

是不是变的简单了?

也可以吧Block Pointer当成参数传递给一个function,比如:

  1. void myFunction(int (^mySquare)(int));     //function的定义,将Block作为参数
  2. int (^mySquare)(int) = ^(int a){return a*a;};   //定义一个mySquare的Block pointer变量
  3. myFunction(mySquare);    //把mySquare作为myFunction的参数

上面的三行代码其实等价于下面这行代码:

  1. myFunction( ^int(int a){return a*a;} );

当其作为Object-C method的传入值的话,需要把类型写在变量前面,然后加上小括号。比如下面这种写法:

  1. -(void)objcMethod:(int(^)(int))square;  //square参数的类型是int(^)(int)

存取变量

1、可以读取和Block pointer同一个Scope的变量值:

  1. {
  2. int outA = 8;
  3. int (^myPtr)(int) = ^(int a){ return outA + a;};
  4. //block里面可以读取同一类型的outA的值
  5. int result = myPtr(3);  // result is 11
  6. NSLog(@"result=%d", result);
  7. }

下面来看一段很有意思的代码:

  1. {
  2. int outA = 8;
  3. int (^myPtr)(int) = ^(int a){ return outA + a;}; //block里面可以读取同一类型的outA的值
  4. outA = 5;  //在调用myPtr之前改变outA的值
  5. int result = myPtr(3);  // result的值仍然是11,并不是8
  6. NSLog(@"result=%d", result);
  7. }

为什么result 的值仍然是11?而不是8呢?事实上,myPtr在其主体中用到的outA这个变量值的时候做了一个copy的动作,把outA的值copy下来。所以,之后outA即使换成了新的值,对于myPtr里面copy的值是没有影响的。

需要注意的是,这里copy的值是变量的值,如果它是一个记忆体的位置(地址),换句话说,就是这个变量是个指针的话,

它的值是可以在block里被改变的。如下例子:

  1. {
  2. NSMutableArray *mutableArray = [NSMutableArray arrayWithObjects:@"one", @"two", @"three", nil];
  3. int result = ^(int a){[mutableArray removeLastObject]; return a*a;}(5);
  4. NSLog(@"test array :%@", mutableArray);
  5. }

原本mutableArray的值是{@"one",@"two",@"three"},在block里面被更改mutableArray后,就变成{@"one", @"two"}了。

2、直接存取static类型的变量

  1. {
  2. static int outA = 8;
  3. int (^myPtr)(int) = ^(int a){return outA + a;};
  4. outA = 5;
  5. int result = myPtr(3);  //result的值是8,因为outA是static类型的变量
  6. NSLog(@"result=%d", result);
  7. }

甚至可以直接在block里面修改outA的值,例如下面的写法:

  1. {
  2. static int outA = 8;
  3. int (^myPtr)(int) = ^(int a){ outA = 5; return outA + a;};
  4. int result = myPtr(3);  //result的值是8,因为outA是static类型的变量
  5. NSLog(@"result=%d", result);
  6. }

3、Block Variable类型的变量

在某个变量前面如果加上修饰字“__block”的话(注意,block前面有两个下划线),这个变量就称作block variable。

那么在block里面就可以任意修改此变量的值,如下代码:

  1. {
  2. __block int num = 5;
  3. int (^myPtr)(int) = ^(int a){return num++;};
  4. int (^myPtr2)(int) = ^(int a){return num++;};
  5. int result = myPtr(0);   //result的值为5,num的值为6
  6. result = myPtr2(0);      //result的值为6,num的值为7
  7. NSLog(@"result=%d", result);
  8. }

因为myPtr和myPtr2都有用到num这个block variable,最终num的值为7.

IOS中的Block与C++11中的lambda

时间: 2024-10-21 01:35:33

IOS中的Block与C++11中的lambda的相关文章

在c++11中你最吃惊的新feature是什么?

对我来说,我最吃惊居然把对于多线程的支持加到标准中了.真的想不明白,对于c++这样一种语言,怎么会加进这个东西. 1. 因为各个平台的不同,对于多线程的支持会有许多平台独有的特色,这样c++标准的定义必然会受到很大限制. 2. 各个平台对于多线程的支持已经形成了各自的标准,比如pthread,或是windows上的标准api,现在去推c++的多线程标准,有意义吗?是不是太晚了? 但是仍然看到c++11中加入了很多很棒的feature.相信其中很多的feature会在以后的c++编程中大行其道.

如何在iOS中使用Block

如何在iOS中使用Block Block可以帮助我们组织独立的代码段,并提高复用性和可读性.iOS4在UIKit中引入了该特征.超过100个的Apple API都使用了Block,所以这是一个我们必须开始熟悉的知识. Block是什么样的? 你可以使用^操作符来声明一个Block变量,它表示一个Block的开始. int num1 = 7; int(^aBlock)(int) = ^)int num2) { return num1+nunm2; }; 在如上代码中我们将Block声明为一个变量,

iOS中为什么block用copy属性

1. Block的声明和线程安全Block属性的声明,首先需要用copy修饰符,因为只有copy后的Block才会在堆中,栈中的Block的生命周期是和栈绑定的,可以参考之前的文章(iOS: 非ARC下返回Block).另一个需要注意的问题是关于线程安全,在声明Block属性时需要确认“在调用Block时另一个线程有没有可能去修改Block?”这个问题,如果确定不会有这种情况发生的话,那么Block属性声明可以用nonatomic.如果不肯定的话(通常情况是这样的),那么你首先需要声明Block

iOS中使用block进行网络请求回调

iOS中使用block进行网络请求回调 HttpRequest.h // // HttpRequest.h // UseBlockCallBack // // Created by Michael on 2/13/14. // Copyright (c) 2014 EIMS. All rights reserved. // #import <Foundation/Foundation.h> typedef void (^FinishBlock)(NSString *dataString); @

ios 中的block应用

在这个大冬天里默默敲着键盘,勿喷.今天学习swift过程中,学习到闭包,发现闭包和oc的block中有很多的相同之处,又重新学习了一下并且学习了一些高级点的用法,内容如下: 1.block格式说明:(返回类型)(^块名称)(参数类型) = ^(参数列表) {代码实现};//如果没有参数,等号后面参数列表的()可以省略 例子: void(^demoBlock)() = ^ { NSLog(@"demo Block"); }; int(^sumBlock)(int, int) = ^(in

block使用小结、在arc中使用block、如何防止循环引用

引言 使用block已经有一段时间了,感觉自己了解的还行,但是几天前看到CocoaChina上一个关于block的小测试主题 : [小测试]你真的知道blocks在Objective-C中是怎么工作的吗?,发现竟然做错了几道, 才知道自己想当然的理解是错误的,所以抽时间学习了下,并且通过一些测试代码进行测试,产生这篇博客. Block简介(copy一段) Block作为C语言的扩展,并不是高新技术,和其他语言的闭包或lambda表达式是一回事.需要注意的是由于Objective-C在iOS中不支

iOS开发读取plist文件、iphone中plist文件的

在Xcode中建立一个iOS项目后,会自己产生一个.plist文件,点击时会看见它显示的是类似于excel表格: 但是,如果打开方式选择Source Code,你会看见它其实是一个xml文件. 我们会做一个小例子,在这个例子中我们自己建立一个plist文件并填入数据,然后运行时读取这个plist文件,并将数据填写在界面上. 首先要知道读取plist文件的方法,一般来说,使用代码 NSString *plistPath = [[NSBundle mainBundle] pathForResourc

Objective-c中的Block(块)详解

Block初探 在Objective-c中NSArray是很常用的容器之一,很多时候我们需要对数组中的数据进行排序,因此与下面类似的代码会经常碰到: NSArray *sortedArray = [array sortedArrayUsingComparator: ^(id obj1, id obj2) { if ([obj1 integerValue] > [obj2 integerValue]) { return (NSComparisonResult)NSOrderedDescending

Objective-C中的Block

1.相关概念 在这篇笔记开始之前,我们需要对以下概念有所了解. 1.1 操作系统中的栈和堆 注:这里所说的堆和栈与数据结构中的堆和栈不是一回事. 我们先来看看一个由C/C++/OBJC编译的程序占用内存分布的结构: 栈区(stack):由系统自动分配,一般存放函数参数值.局部变量的值等.由编译器自动创建与释放.其操作方式类似于数据结构中的栈,即后进先出.先进后出的原则. 例如:在函数中申明一个局部变量int b;系统自动在栈中为b开辟空间. 堆区(heap):一般由程序员申请并指明大小,最终也由