http://blog.csdn.net/menxu_work/article/details/8762848

block(代码块)的介绍以及使用方法和变量之间的关系

标签: block
2013-04-05 22:52 5613人阅读 评论(0) 收藏 举报
 分类:
ios(72)   object-c(66) 
 

目录(?)[+]

 

Block是什么?

Block是C语言的一个语法特性,同时也是C语言的运行时特性,它很像C中的函数指针,因为你可以像使用函数指针一样的去使用block对象;它也很像C++中的函数对象,因为除了要执行的代码,block还可以携带和block绑定的状态信息。

因此,block是一个对象,这个对象里包含了要执行的代码片段以及一些状态信息。

MacOSX 10.6和iOS 4.0以上版本的Xcode开发包提供了对block的支持。

Block的功能

block是一片具有以下特性的内联代码片段集合:

  • 可以像函数一样有类型参数;
  • 可以声明或推算出一个返回类型;
  • 可以访问和block定义在同一个词法范围里的变量(即Status);
  • 可以修改同一个词法范围里的变量;
  • 同一个词法范围的block之间可以共享变量和变量的修改结果;
  • 当栈被摧毁后,栈里的block依旧可以保持状态信息;

Block的用法

作为一个自包含的代码片段,由于以下特性,block很适合作为回调函数的替代方案:

  • 你可以在方法的下上文中,调用block的地方直接编写构成block的代码片段;
  • block可以访问局部变量;

如何声明和定义Block

你可以通过^操作符定义一个block类型的变量,用{}来圈定block的代码片段,如下图所示:

再次声明:block可以访问和block定义在同一个词法范围里的变量。

int multiplier = 7;
int (^myBlock)(int) = ^(int num) { return num * multipiler; } printf("%d", myBlock(3)); // prints "21"

直接使用Block

在更多的时候,你并不需要定义自己的Block类型,而是在API中直接编写block代码片段,例如:qsort_b。

char *myCharacter[3] = { "safari", "ie", "chrome" };
qsort_b(myCharacter, 3, sizeof(char *), ^(const void *l, const void *r) {
    char *left = *(char **)l;
    char *right = *(char **)r;
    return strncmp(left, right, 1);
});

Block和Cocoa

Cocoa framework中很多方法使用了Block作为其参数(尽管也有对应的callback版本,但还是推荐使用 block版本)。在动画以及集合遍历方面,block很常见。

NSArray *stringsArray = [NSArray arrayWithObjects:
                                 @"string 1",
                                 @"String 21",
                                 @"string 12",
                                 @"String 11",
                                 @"String 02", nil];
static NSStringCompareOptions comparisonOptions = NSCaseInsensitiveSearch |
    NSNumericSearch |NSWidthInsensitiveSearch | NSForcedOrderingSearch; NSLocale *currentLocale = [NSLocale currentLocale];
NSComparator finderSortBlock = ^(id string1, id string2) {
    NSRange string1Range = NSMakeRange(0, [string1 length]);
    return [string1 compare:string2
                    options:comparisonOptions
                      range:string1Range
                     locale:currentLocale];
}; NSArray *finderSortArray = [stringsArray
    sortedArrayUsingComparator:finderSortBlock]; NSLog(@"finderSortArray: %@", finderSortArray);

block和变量之间的关系

这一部分涉及内存管理相关的内容,为了正确的使用block,理解并记住它们,很重要:

在一个block代码片段的内部,你可以使用三种不同类型的变量(就像你在函数里一样):

  • 全局变量(包括static locals);
  • 全局函数(尽管这并不是变量=。=);
  • 在包含block的词法范围内的局部变量;

当在一个Block里使用变量时,应遵循以下规则:

  • 在包含block的词法范围里的栈局部变量,在block内部是常量,只能只读访问。这些局部变量的值即block执行时,局部变量的值。在多层内嵌的block中,

局部变量的值,取最内层词法范围里,局部变量的值;

  • 被声明为__block存储类型的局部变量通过引用传递给block,因此是mutable的;在__block类型的局部变量的有效范围内,对该局部变量的修改,会在该有效范围内的所有block范围内生效;
  • 定义在block内部的局部变量,和函数内部的局部变量法则相同;

正确的:

int x = 123;
void (^printXandY)(int) = ^(int y) {
    printf("%d %d\n", x, y);
}; printXandY(456);

错误的:

int x = 123;

void (^printXandY)(int) = ^(int y) {
    x = x + y; // ERROR HERE!!! x should be __block
    printf("%d %d\n", x, y);
}; printXandY(456);

关于__block的进一步讨论

  1. __block是只针对局部变量生效的一种描述变量存储类型的关键字,因此__block类型的变量都是栈变量;
  2. __block类型的变量在其定义的语法范围里,和该范围内的所有block共享存储空间,当block在被复制到heap区域时,同区域内的__block变量占用的内存不会随着退栈而销毁;
  3. 出于优化的考虑,栈中的block对象最开始和一般的栈局部变量是相同的,当使用Block_copy对block进行复制时,才被拷贝到heap区域;
  4. __block变量不能是一个可变长数组;

下面这个例子用于展示,各种类型的变量与__block之间的交互:

extern NSInteger CounterGlobal;
static NSInteger CounterStatic;
{
    NSInteger localCounter = 42;
    __block char localCharacter;
    void (^aBlock)(void) = ^(void) {
        ++CounterGlobal;
        ++CounterStatic;
        CounterGlobal = localCounter; // localCounter fixed at block creation
        localCharacter = 'a'; // sets localCharacter in enclosing scope
    };
    ++localCounter; // unseen by the block
    localCharacter = 'b';
    aBlock(); // execute the block
    // localCharacter now 'a'
}

__block和Object C对象之间的关系

如果,你在一个方法的实现里,使用了__block,则:

  • 通过引用的方式访问对象的,self被retain;

    dispatch_async(queue, ^{
        // instanceVariable is used by reference, self is retained
        doSomethingWithObject(instanceVariable);
    });
  • 通过值访问对象的,被访问的对象被retain;

    id localVariable = instanceVariable;
    dispatch_async(queue, ^{
        // localVariable is used by value, localVariable is retained (not self)
        doSomethingWithObject(localVariable);
    });

__block和C++对象之间的关系

需要注意两点:

  • 把一个基于栈的C++对象变成一个__block类型的时候,要调用类的copy constructor;
  • 在block内部使用栈中的C++对象时,要调用栈的const copy constructor;
Comment by project member SuPeiqi....@gmail.com, Mar 10, 2011

change :

int (myBlock)(int) = (int num) { return num ^ multipiler; }

to :

int (myBlock)(int) = (int num) { return num *  multipiler; }

来自:http://code.google.com/p/nevel-mercury/wiki/GetFamiliarWithBlock
 
 

block(代码块)的介绍以及使用方法和变量之间的关系的更多相关文章

  1. block代码块介绍

    关于block的简单介绍 什么是block? Block是C语言的一个语法特性,同时也是C语言的运行时特性,它很像C中的函数指针,因为你可以像使用函数指针一样的去使用block对象:它也很像C++中的 ...

  2. IOS Block代码块的定义与使用

    代码块的本质是和其他的变量类似,不同的是,代码块存储的数据是一个函数体.使用代码块,你可以像调用其他标准函数一样的调用,可以传入参数,并得到返回值.     脱字符是代码块的语法标记.下图表示代码块的 ...

  3. Block代码块中使用局部变量注意点

    第一次写代码遇到报这个错,实在是想不通为什么,按常理应该是不会有问题,报错的呀??纠结了一会之后只好仔细查看报错原因咯,原来是: 当我们在block代码块中使用局部变量时,就会很容易出现如图的错误. ...

  4. IOS学习4——block代码块

    本文转载自:iOS开发-由浅至深学习block 一.关于block 在iOS 4.0之后,block横空出世,它本身封装了一段代码并将这段代码当做变量,通过block()的方式进行回调.这不免让我们想 ...

  5. OC Block(代码块)

    #import "ViewController.h" @interface ViewController () @end @implementation ViewControlle ...

  6. java synchronized修饰普通方法,修饰静态方法,修饰代码块,修饰线程run方法 比较

    synchronized用于多线程设计,有了synchronized关键字,多线程程序的运行结果将变得可以控制.synchronized关键字用于保护共享数据. synchronized实现同步的机制 ...

  7. iOS - OC Block 代码块

    前言 Block 是一段预先准备好的代码,可以在需要的时候执行,可以当作参数传递.Block 可以作为函数参数或者函数的返回值,而其本身又可以带输入参数或返回值.Block 是 C 语言的,类似于一个 ...

  8. iOS - Block 代码块

    1.Block Block 是一段预先准备好的代码,可以在需要的时候执行,可以当作参数传递.Block 可以作为函数参数或者函数的返回值,而其本身又可以带输入参数或返回值.Block 是 C 语言的, ...

  9. 构造方法、封装、关键字(this、static)和代码块的介绍

    1.构造方法 1.1 构造方法与成员方法的区别 构造方法分为无参构造和有参构造,其中有参构造方法和无参构造方法为方法的重载关系. 构造方法在初始化一个类的对象时进行调用,它没有返回值,方法名与类名相同 ...

随机推荐

  1. iOS学习05C语言函数

    本次主要是学习和理解函数,函数树状图如下: 1.函数的声明和定义 函数定义的四要素分别为: 返回值类型 :函数的结果值类型,函数不能返回数组. 指定返回类型是void类型说明函数没有返回值. 函数名 ...

  2. BZOJ4504 : K个串

    从左往右枚举右端点,用一棵线段树维护每个左端点的去重后的区间和. 那么对于$a[r]$,需要在$[pre[a[r]]+1,r]$里区间加上$a[r]$. 将线段树可持久化,并维护区间最大值,就可以在线 ...

  3. BZOJ1103[POI2007]大都市meg 题解

    题目大意: 有一棵树,最先每条边的权值是1,然后给出n+m-1个操作,操作有两种:1.询问一个点到根的路径上的权值和:2.将一条边的权值改为0. 思路: 用dfs序将树化为序列,在dfs序中我们会保存 ...

  4. 20145308刘昊阳 《Java程序设计》第3周学习总结

    20145308刘昊阳 <Java程序设计>第3周学习总结 教材学习内容总结 第四章 认识对象 要产生对象必须先定义类,类是对象的设计图,对象是类的实例 类是从少数实例推广到大量相似实例的 ...

  5. Android AsyncTask

    AsyncTask类任务管理: 内部会创建一个进程作用域的线程池来管理要运行的任务,也就就是说当你调用了AsyncTask#execute()后,AsyncTask会把任务交给线程池,由线程池来管理创 ...

  6. Leetcode Unique Binary Search Trees

    Given n, how many structurally unique BST's (binary search trees) that store values 1...n? For examp ...

  7. 读书笔记:javascript高级技巧(一)

    一.安全的类型检测 javascript内置的类型检测机制并非完全可靠,由于浏览器或者作用域等原因,经常会发生错误.大家知道,在任何值调用toString()方法都会返回一个[object Nativ ...

  8. 【BZOJ】3093: [Fdu校赛2012] A Famous Game

    http://www.lydsy.com/JudgeOnline/problem.php?id=3093 题意:n个球(红和蓝两种),等概率有1~n个红球.首先取出p个球且这p个球里边有q个红球,问从 ...

  9. Java_java动态编译整个项目,解决jar包找不到问题

    java动态编译整个项目,解决jar包找不到问题原文:http://itzyx.com/index.php/javac/ 动态将java文件编译为class文件解决方案:将temp\sdl\src目录 ...

  10. JSch - Java实现的SFTP(文件下载详解篇)

    上一篇讲述了使用JSch实现文件上传的功能,这一篇主要讲述一下JSch实现文件下载的功能.并介绍一些SFTP的辅助方法,如cd,ls等. 同样,JSch的文件下载也支持三种传输模式:OVERWRITE ...