>>定义并使用一个block    返回值(^名字)(参数列表) =^返回值类型(参数列表){表达式};  其中返回值和参数列表可以神略 ,最简单的block是  ^{xxxx};

void(^add)(int a , int b ) = ^(int a , int b ){

NSLog(@"a+b=%d",a+b);

};

add(1,3);

>>假设有如下代码

- (void)viewDidLoad {

int f = 0;

void(^add)(int a , int b ) = ^(int a , int b ){

NSLog(@"a+b=%d",a+b);

f = 1;

};

add(1,3);

}

这个时候,会提示f=1;这里出现错误,如果需要修改block以外的局部变量(在这个viewDidLoad函数内的)就需要使用 __block int f = 0;

不需要加入__block就可以直接修改的变量(从下面看出,应该就是变量的生命周期问题)(注释1)

全局变量

类的成员变量

static变量(函数体里面也可以的)

>>有如下代码

int a = 1;

void(^OutPut)() = ^(){

NSLog(@"a=%d",a);

};

a = 2;

OutPut();

上面的代码输出 a=1;

如果修改为 __block int a = 1;   这样会输出 a=2;(所有的可改变的变量都会输出2,在上面注释1处说明的变量)

>>再看一个例子

- (void)viewDidLoad {

NSMutableArray * arr = [NSMutableArray arrayWithObject:@"ddd"];

void(^OutPut)() = ^(){

[arr removeLastObject];

arr = [NSMutableArray new];

};

OutPut();

}

上面提到比如不加入__block的变量是不能在block里面修改的,这里的arr因为没有加__block 所以在执行 arr = [NSMutableArray new];的时候就出错了,但是[arr removeLastObject];并没有任何问题。

>>定义一个 Block Pointer

上面的 void(^add)(inta,int b) 我们其实可以使用一个typedef来实现

typedef void(^AddMethod)(int , int );

AddMethod add = ^(int a , int b ){ NSLog(@“a+b=%d”,a+b};

下面涉及到的内容可以看看如下网址

http://www.cocoachina.com/bbs/read.php?tid=152222

>>NSStackBlock  NSGlobalBlock NSMallocBlock 如何区分(可以通过NSLog来打印出一个block属于哪个类型)

NSGlobalBlock:

block内不访问任何变量的

block除了方位全局变量跟static变量的(其中类的成员变量等不在这2种类型里面)

剩下的基本上都是StackBlock

比如:

void(^outPut)() = ^{

NSLog(@"outPut");

};

是GlobalBlock,因为block内部没有访问任何变量

比如:

int a = 0;

void(^outPut)() = ^{

NSLog(@"outPut:%d",a);

};

是StackBlock,因为方位了非(全局,static)变量

比如:

static int a = 0;

void(^outPut)() = ^{

NSLog(@"outPut:%d",a);

};

是GlobalBlock,因为只访问了static变量

>> 有了上面3个Block存储区域的概念,我们就比较容易理解了下面的内容了。

例子:

  • void exampleB_addBlockToArray(NSMutableArray *array) {
  • char b = 'B';
  • [array addObject:^{
  • printf("%c\n", b);
  • }];
  • }
  • void exampleB() {
  • NSMutableArray *array = [NSMutableArray array];
  • exampleB_addBlockToArray(array);
  • void (^block)() = [array objectAtIndex:0];
  • block();
  • }

首先看addObject这个block是一个StackBlock

在ARC情况下,执行正确 :因为ARC下会自动将StackBlock拷贝到MallocStack中

在非ARC情况下,执行错误:因为在exampleB_addBlockToArray函数返回之后,这个block失效了

例子

  • void exampleC_addBlockToArray(NSMutableArray *array) {
  • [array addObject:^{
  • printf("C\n");
  • }];
  • }
  • void exampleC() {
  • NSMutableArray *array = [NSMutableArray array];
  • exampleC_addBlockToArray(array);
  • void (^block)() = [array objectAtIndex:0];
  • block();
  • }

首先看这里的addObject这个block没有访问任何的外部变量,所以这是一个GlobalBlock,不管在ARC还是非ARC下,都是正确的。

>>  为什么在使用property属性修饰block的时候,需要用到copy操作?

在非ARC下,需要将block从栈中copy到堆中,避免被自动释放。对block 进行retain操作是无效的

在ARC下,block会自动copy到堆中。

>> Block循环引用问题(一定要是双方的)

使用block的时候,经常会由于循环引用的问题导致内存无法释放,(对象拥有block,block又用到对象)

比如

self.myBlock = ^{ [self xxxxx];};

这里myBlock是self的属性,而在myBlock里面又使用到了self;self在释放的时候必定要先释放myBlock,但是myBlock需要先释放里面的self,这样就导致死循环了。

要打破这个格局,可以使用如下

在ARC下:

__weak typeof(self) weakSelf = self;

self.myBlock = ^{ [weakSelf xxxxx];};

如果在block里面会多次用到weakSelf 那应该使用如下的strongSelf

__weak typeof(self) weakSelf = self;

self.myBlock = ^{

__strong typeof(self)strongSelf = weakSelf;

[strongSelf xxxxx];

[strongSelf bbbbb];

};

在非ARC下,直接将__weak改为__block就可以(__block typeof(self)weakSelf=self;)。

所以__block的意思在ARC下只表示可修改;在非ARC下,表示可修改、不要retain这个变量.

>> 在一个类中,并不是直接使用了self才会导致强引用self的。比如类中有一个实例变量 int value;

那么在  dispatch_async(queue,^{    doSomthingWith(value); } );   这个时候用到了类的实例变量value,实际上也是强应用了 self。如果queue是类的成员变量,那就会出现死循环的问题。如果是这样  dispatch_async(dispatch_get_global_queue(0,0),^{    doSomthingWith(value); } ); 这个就是单向引用,不会造成死循环。

>>  block与函数的区别

block是对象,它封装了一段代码,这段代码可以在任何时候执行。block可以作为函数参数或者函数的返回值,而其本身又可以带输入参数或者返回值。

block跟传统的函数指针很类似,但是block是内联的(inline),并且它对局部变量只是只读的。

block可以访问局部变量,但是不能修改

在需要修改的局部变量前加入__block

定义一个block看上去像定义一个函数差不多,void(^bbb)(void)=^(int a){…..};

但是还是有很大的区别的,block的定义可以在一个函数的内部实现。

>> __block 与 __weak的区别

__block 在ARC以及MRC模式下都可以使用。

__weak是ARC模式下使用的,代表的是弱引用,防止循环引用。

__block在非ARC下表示可以修改、如果修饰的是对象,表示不要retain(防止循环引用)

__block对象在block中可以被重新赋值,__weak不可以。

BLOCK专题的更多相关文章

  1. ios专题 -block用法

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

  2. [.NET领域驱动设计实战系列]专题八:DDD案例:网上书店分布式消息队列和分布式缓存的实现

    一.引言 在上一专题中,商家发货和用户确认收货功能引入了消息队列来实现的,引入消息队列的好处可以保证消息的顺序处理,并且具有良好的可扩展性.但是上一专题消息队列是基于内存中队列对象来实现,这样实现有一 ...

  3. Block Chain, a protocol view

    我做了个区块链的文档,给自己扫盲用的,有兴趣的可以看下,主要是自己画示意图比较好理解,示意图之后的专题部分,内容直接取自参考链接.网上的资料都是谈区块链有什么性质.有什么能力.有什么应用之类的,我主要 ...

  4. block的内部实现

    主要内容: 一.block相关的题目 二.block的定义 三.block的实现 四.捕获自动变量值 五.block存储区域 六.截获对象 一.block相关的题目 这是一篇比较长的博文,前部分是bl ...

  5. 【转】css学习专题-BFC

    css学习专题-BFC 转自:原文链接:css学习专题-BFC文章目录 BFC:Block Formatting Context. BFC就是一种布局方式,在这种布局方式下,盒子们自所在的 conta ...

  6. HBase 专题技术收录

    HBase系列: 博客地址:http://www.cnblogs.com/panfeng412/tag/HBase/ 技术专题文章: HBase中MVCC的实现机制及应用情况 HBase在单Colum ...

  7. CSS浮动专题!

    在css中,浮动问题可能是很多刚入门的小白比较头疼的问题. 1,首先先来介绍一下两种浮动类型:左浮动和右浮动 1) float:left;左浮动,后面的内容会流向对象的右侧 2) float:righ ...

  8. mysql优化专题」90%程序员都会忽略的增删改优化(2)

    补充知识点:操作数据语句优化的认识 通常情况下,当访问某张表的时候,读取者首先必须获取该表的锁,如果有写入操作到达,那么写入者一直等待读取者完成操作(查询开始之后就不能中断,因此允许读取者完成操作). ...

  9. Oracle之SQL优化专题01-查看SQL执行计划的方法

    在我2014年总结的"SQL Tuning 基础概述"中,其实已经介绍了一些查看SQL执行计划的方法,但是不够系统和全面,所以本次SQL优化专题,就首先要系统的介绍一下查看SQL执 ...

随机推荐

  1. 重写equals()方法时,需要同时重写hashCode()方法

    package com.wangzhu.map; import java.util.HashMap; /** * hashCode方法的主要作用是为了配合基于散列的集合一起正常运行,<br/&g ...

  2. C语言中volatile关键字的作用

    http://blog.csdn.net/tigerjibo/article/details/7427366#comments 一.前言 1.编译器优化介绍: 由 于内存访问速度远不及CPU处理速度, ...

  3. Win32 SDK程序创建一些控件(简单调用InitCommonControlsEx,并指定ICC_LISTVIEW_CLASSES控件就可以了)

    在Win32 SDK中创建一些控件的时候需要注意一下(具体是哪些控件请参看MSDN文档中列出来的) /* MSDN:Carries information used to load common co ...

  4. Spring的父子容器问题

    在ssm框架搭建的时候 配置了一个Spring容器,又配置了一个前端控制器 <!-- 初始化spring容器 --> <context-param> <param-nam ...

  5. NSDictionary 键值对查找

           NSDictionary *dic1=[NSDictionary dictionaryWithObjectsAndKeys: @"1",@"a", ...

  6. 4、JPA table主键生成策略(在JPA中table策略是首推!!!)

    用 table 来生成主键详解 它是在不影响性能情况下,通用性最强的 JPA 主键生成器.这种方法生成主键的策略可以适用于任何数据库,不必担心不同数据库不兼容造成的问题. initialValue不起 ...

  7. IOS判断手机型号

    #define iPhone5 ([UIScreen instancesRespondToSelector:@selector(currentMode)] ? CGSizeEqualToSize(CG ...

  8. gulp.watch监听文件

    Gulp.watch()会返回我们熟知的watcher.我们可以利用watcher来监听额外的事件或者向watch中添加文件. 例如,在执行一系列任务和调用一个函数时,你就可以在返回的watcher中 ...

  9. Hadoop集群(第7期)_Eclipse开发环境设置

    1.Hadoop开发环境简介 1.1 Hadoop集群简介 Java版本:jdk-6u31-linux-i586.bin Linux系统:CentOS6.0 Hadoop版本:hadoop-1.0.0 ...

  10. hdu 3501 Calculation 2 (欧拉函数)

    题目 题意:求小于n并且 和n不互质的数的总和. 思路:求小于n并且与n互质的数的和为:n*phi[n]/2 . 若a和n互质,n-a必定也和n互质(a<n).也就是说num必定为偶数.其中互质 ...