Block简介:

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

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

Block实体开头是“^”,接着是由小括号所包起来的参数列(比如 int a, int b, int c),行为主体由大括号包起来,专有名字叫做block literal。行为主体可以用return回传值,类型会被compiler自动辨别。如果没有参数列要写成: ^(void)。

例如下面的一个例子:

[cpp] view plaincopy

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

这是代表Block会回传输入值的平方值(int a 就是参数列, return a*a; 就是行为主体)。记得行为主体里最后要加“;”,因为是叙述,而整个{}最后也要加“;”,因为Block是物件实体。用法如下:

[cpp] view plaincopy

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

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

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

Block Pointer是这样定义的:

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

比如下面的例子:

[cpp] view plaincopy

 
  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,比如:

[cpp] view plaincopy

 
  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的参数

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

[cpp] view plaincopy

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

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

[cpp] view plaincopy

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

存取变量

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

[cpp] view plaincopy

 
  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. }

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

[cpp] view plaincopy

 
  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里被改变的。如下例子:

[cpp] view plaincopy

 
  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类型的变量

[cpp] view plaincopy

 
  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的值,例如下面的写法:

[cpp] view plaincopy

 
  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里面就可以任意修改此变量的值,如下代码:

[cpp] view plaincopy

 
  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 用法 (1)-once again的更多相关文章

  1. iOS block 用法

    1.定义Block /* 回传void ,参数也是void 的block*/ void (^blockReturningVoidWithVoidArgument)( void ); /* 回传整数,两 ...

  2. ios Block详细用法

    ios Block详细用法 ios4.0系统已开始支持block,在编程过程中,blocks被Obj-C看成是对象,它封装了一段代码,这段代码可以在任何时候执行.Blocks可以作为函数参数或者函数的 ...

  3. iOS Block界面反向传值

    在上篇博客 <iOS Block简介> 中,侧重解析了 iOS Block的概念等,本文将侧重于它们在开发中的应用. Block是iOS4.0+ 和Mac OS X 10.6+ 引进的对C ...

  4. iOS block从零开始

    iOS block从零开始 在iOS4.0之后,block横空出世,它本身封装了一段代码并将这段代码当做变量,通过block()的方式进行回调. block的结构 先来一段简单的代码看看: void ...

  5. IOS NSInvocation用法简介

    IOS NSInvocation用法简介 2012-10-25 19:59 来源:博客园 作者:csj007523 字号:T|T [摘要]在 iOS中可以直接调用某个对象的消息方式有两种,其中一种就是 ...

  6. iOS block 机制

    本文要将block的以下机制,并配合具体代码详细描述: block 与 外部变量 block 的存储域:栈块.堆块.全局块 定义 块与函数类似,只不过是直接定义在另一个函数里,和定义它的那个函数共享同 ...

  7. iOS Block的本质(四)

    iOS Block的本质(四) 上一篇文章iOS Block的本质(三)中已经介绍过block变量的捕获,本文继续探寻block的本质. 1. block内修改变量的值 int main(int ar ...

  8. # iOS Block的本质(三)

    iOS Block的本质(三) 上一篇文章iOS Block的本质(二)中已经介绍过block变量的捕获,本文继续探寻block的本质. 1. block对对象变量的捕获,ARC 环境 block一般 ...

  9. iOS Block的本质(二)

    iOS Block的本质(二) 1. 介绍引入block本质 通过上一篇文章Block的本质(一)已经基本对block的底层结构有了基本的认识,block的底层就是__main_block_impl_ ...

随机推荐

  1. ASIHttpRequest或者SDWebImage给UIImageView加载图片的逻辑是什么样子的

    非常非常喜欢SDWebImage,就算后来AF也做了图片缓存,我也还是一直使用.图片缓存真的是一件令人头疼的事,但是SDWebImage封装了一切,你所需要的只是简单的set一个URL:1.设置一个展 ...

  2. 浅谈异步IO各模型优缺点

    本文只讨论OverLapped I/O的三种异步模型及完成端口,像select.SWASelect不作讨论,讨论顺序从劣到优,方便于循序渐进地对比,更容易区分各模型之间的差别. 1. OverLapp ...

  3. Apache Kylin

    日前,eBay公司隆重宣布已经正式向开源业界推出分布式分析引擎:Kylin(http://kylin.io).作为一套旨在对Hadoop环境下分析流程进行加速.且能够与SQL兼容性工具顺利协作的解决方 ...

  4. Asp.Net Mvc后台数据验证自测小Demo

    使用过MVC的同学一定不陌生MVC的模型绑定和模型校验,使用起来非常方便,定义好Entity之后,在需要进行校验的地方可以打上相应的Attribute,在Action开始时检查ModelState的I ...

  5. windows7实用快捷键 分类: windows常用小技巧 2014-04-18 14:34 169人阅读 评论(0) 收藏

    几个比较实用的快捷键 windows键:简写成win win+   方向键上下,可以使当前窗体放大或缩小 win+   方向键左右,可以使当前窗体悬靠在左边或右边 win+Home   仅保留当前窗体 ...

  6. HTTP协议和WEB应用

    一.应用层协议原理 1.套接字(Socket):主机地址+端口地址.(通常为32位IP地址和16位端口号组成,总长度为48位) 2.进程通过套接字来接收和发送报文.因特网运输层将所提供的服务整合成两种 ...

  7. Textview 文本旋转,倾斜

    有时候Android自带的控件无法满足我们的某些要求,这时就需要我们自定义控件来实现这些功能.比如需要一个TextView里的字倾斜一定的角度,就需要自定义TextView. 代码如下: ? 1 2 ...

  8. 【LeetCode】Swap Nodes in Pairs

    Given a linked list, swap every two adjacent nodes and return its head. For example, Given 1->2-& ...

  9. July收集荷兰国旗问题之三路partition

    这道题目和分成两块的partition的扩展.比如有一堆0 1 2 数字组成的数组,要分成 00 00  11 1 1  222 2这样的顺序的. 利用lumoto版的partition能够非常好的解 ...

  10. 第一篇:数据库需求与ER建模

    前言 在数据库建设过程中,哪一步最重要?绝大多数资料会告诉你,是需求分析阶段.这一步的好坏甚至直接决定数据库项目的成败. 需求分析阶段,也被称为ER建模(entity-relationship mod ...