iOS Blocks 第一弹|基础知识

Block 是在 iOS 开发过程中经常出现的角色。它是由 Apple 在 OS X Snow Leopard / iOS 4 上引入的,属于对标准 C 的拓展。Block 可以视为是「带有局部变量的匿名函数(anonymous functions together with automatic variables)」

iOS Blocks

Blocks 系列是基于学习《Pro Multithreading and Memory Management for iOS and OS X with ARC, Grand Central Dispatch, and Blocks》 一书过程中的笔记整理而来的。

Block 是在 iOS 开发过程中经常出现的角色。它是由 Apple 在 OS X Snow Leopard / iOS 4 上引入的,属于对标准 C 的拓展。Block 可以视为是「带有局部变量的匿名函数(anonymous functions together with automatic variables)」。

我们首先需要区分两个概念:BlockBlock 类型的变量。Block 是咱们要讲述的主角,而 Block 类型的变量则是一个值类型为 Block 的变量。类似于 int a = 0 中的 0 和 a 的区别。

如何声明 Block 和 Block 类型的变量

声明一个 Block:

^ return_type (argument list) { ... }

返回值 return_type 可以省略,省略后由编译器推断返回值类型,没有返回值时会使用 void

^ (argument list) { ... }

如果 Block 执行时不需要参数,(argument list) 也可以省略:

^ { ... }

// 对应的完整形式
^ void (void) { ... }

声明一个 Block 类型的变量:

return_type (^block_variable_name) (argument list)

当然,为了减少多次声明同一种类型变量的工作量,咱们也可以使用 typedef

typedef return_type (^block_variable_name)(argument list);

使用 typedef 之后,就能让函数也能返回 Block 类型的值了:

typedef int (^blk_t)(int);

blk_t func() {
    return ^(int count){return count + 1;};
}

C 语言中通过函数返回一个 Block 类型的值,必须使用 typedef ,而 Objc 则不需要:

- (void (^)(void)) myFunctionReturningABlock {
    return ^ { ... };
}

声明一个 Block,并将其赋值给 Block 类型的变量:

int (^block) (int) = ^ (int param) { return param + 1; };

把 Block 作为函数的参数:

void func(int (^block)(int)) {
		// ...
}

// 或者
typedef int (^blk_t)(int);
void func(blk_t block) {
		// ...
}

Block 和函数指针

C 语言中,函数指针的声明以及使用方式为:

#include "stdio.h"

int test(int arg) {
    return arg;
}

int main() {
    int (*funcptr)(int) = &test;
    printf("%d", funcptr(100));
}

// output: 100

对比一下 Block 类型的变量和函数指针类型的变量,两者非常相似,只不过函数指针使用 * 而 Block 使用 ^

int (*funcptr) (int) 

int (^block) (int)

大部分函数只需要关心函数体内的逻辑,而 Block 既需要关心其内部的处理逻辑,还需要关心 Block 外部的上下文。在接下来的学习过程中,我们会发现 Block 的底层实现里面少不了函数指针的身影,函数指针在 Block 里充当着处理 Block 内部计算逻辑的角色。

Block 对变量的捕获

前面提到,Block 可以视为是「带有局部变量的匿名函数」。这是因为 Block 可以捕获局部变量,这也是 Block 和函数指针的最大区别——能够存储上下文信息。

捕获局部变量的值

先看一段代码:

int val = 10;
const char *fmt = "val = %d\n";
void (^blk)(void) = ^{printf(fmt, val);};
val = 2;
fmt = "These values were changed. val = %d\n";
blk();

// output
// val = 10

这段代码首先初始化了一个 int 型的变量 val 以及一个字符串 fmt 。然后声明一个 Block 类型的变量 blk,并在这个 Block 中捕获了变量 val 以及 fmt

最后修改变量 val 以及 fmt 的值,并执行 blk 后发现输出结果依旧使用了两个变量修改前的值。这说明 Block 对局部变量的捕获就是直接捕获变量的值,不是通过内部的指针指向原来的变量,而是内部有一个新的变量直接存了变量的值,两者没有「纠缠」关系。

self 的捕获也是类似,self 在方法作用域里,其实也可以看作一个局部变量或者方法的参数。类似 python 中的实例方法,默认第一个参数就是 self ,只不过 OC 或者 C++ 里面默认把 self 或者 this 给隐藏了,后面的文章里会具体说这个。

class MyClass:
    def method(self, param: int):
				pass

如果 Block 捕获了 self ,也就是其内部有变量被赋值为 self 的值,那么 Block 会强持有 self 。这时候 self 再持有 Block 的话,就会出现循环引用问题。

Block 内修改局部变量的值

Block 捕获变量的值后,一般情况下,编译器是不允许在 Block 内部修改变量的值。因为 Block 内部有一个新的变量(假设为 A)存储了被捕获变量(假设为 B)的值,A 和 B 之间没有连接关系,修改其中一方不会影响另一方的值。编译器会阻止这种试图通过修改 A 的值来影响 B 的行为。

例如下面代码会报错:

int val = 0;
void (^blk)(void) = ^{ val = 1; };
// error: Variable is not assignable (missing __block type specifier)

但是既然说了这是一般情况,那肯定还有特殊情况。

如果想要在 Block 内部修改外界变量的值,需要使用 __block 关键字。给局部变量加上 __block 关键字之后,编译器就允许在 Block 内修改外面变量的值:

__block int val = 0;
void (^blk)(void) = ^{ val = 1; };
// 此时编译器不再报错

下一篇文章会讲到,这时候的 val 在底层已经不是一个 int 类型的变量,而是一个结构体。

Block 无法捕获 C 数组

下面代码尝试在 Block 内捕获一个 C 数组,但编译器会报错。

const char text[] = "hello";
void (^blk)(void) = ^{ 
    printf("%c\n", text[2]);
};

// error: cannot refer to declaration with an array type inside block
//         printf("%c\n", text[2]);
//                       ^
// note: 'text' declared here
//    const char text[] = "hello";
//               ^

这是因为 Block 无法捕获 C 数组,具体原因会在下一篇的文章中讲述。如果想要在 Block 内读取外部的 C 数组,可以让 Block 捕获一个 C 数组的指针:

const char *text = "hello"; 
void (^blk)(void) = ^{
    printf("%c\n", text[2]); 
};

Read more

碎碎念——投资,不确定性沟通定语

碎碎念——投资,不确定性沟通定语

投资理财 最近因为关税的冲击,美股正在经历一波大跌行情。我个人比较看好纳斯达克,也在一直定投纳斯达克。我是长期主义者,没有精力和时间在短期波动中挣钱,只想在下跌调整中「进货」。 定投分左侧定投和右侧定投。左侧定投是在下跌的过程中定投,而右侧定投是在上涨的过程中定投。左侧定投无法确认底部在哪里,需要源源不断往里投入金钱(行内成为「子弹」);右侧定投无法确认反弹是诱多还是形势已经逆转。我采用的是左侧定投,大跌大加,小跌小加,反弹时停止定投。不论采用哪种定投,殊途同归,都是尽量降低投资成本。 目前网上看衰美股的声音不少,不少人因为恐慌割肉卖出股票。但我们要知道目前美国仍旧是世界第一大国,消费潜力巨大,大型科技公司(苹果、英伟达等)的基本面并没有出现大问题。只是因为特朗普的「量子态」关税政策,导致市场恐慌抛售。我们无需担心纳斯达克、标普指数从此一蹶不振。恰恰相反,现在是买入美股的绝佳时机。苹果、英伟达等大型公司的 PE 值已经降到了合理位置,只要不买妖股,不投机,只关注纳斯达克、标普指数,只买大型公司股票,迟早会取得丰厚盈利的。

By Gray
怀念小时候吃过的食物

怀念小时候吃过的食物

前两天下班骑车回家的路上听到了路旁有人在讨论泡馍。他们口中的泡馍应该是类似西安羊肉泡馍之类的食物。但是我却想起来了小时候吃的不一样的泡馍以及其他吃食。 不一样的泡馍 小时候我们那里普遍比较贫穷,家家户户除了过年过节基本上很难吃到大块肉。小孩子饭量时小时大,中午吃的饭,半晌就又饿了。家里有大葱或者豆糁的话,可以拿着一个馍就着就吃了。整根的葱是最下馍的,葱白部分甜又辣,葱叶里面会有像鼻涕一样的粘液,要把它挤出来才下得嘴吃。豆糁是黄豆的发酵产物,煮熟的大豆加盐发酵几天,黏丝丝的时候团成球,放到发黑就能吃了。吃的时候从球上掰下来几小块就行。豆糁是咸的,因而也能下饭。不过最妙的吃法是将豆糁和鸡蛋一起炒。鸡蛋的香气和豆糁稍微发臭的味道混在一起,形成一种独特的香味。像北京的臭豆腐一样,闻着臭,吃着香。 如果家里没葱没豆糁了,馍又很干,那泡馍就是解决饿肚子的绝好办法。将干硬的馍掰成几瓣,不能太碎小,放到瓷碗里。倒入炒菜的肉味王佐料,或者是平时攒下来的方便面调料。再提溜着暖水瓶,倒进去冒着热气的水。当然香油是少不了的,拿着油光光的瓶子,滴进去几滴喷香的香油。最后用大碗盖住,或者干脆啥也不盖,静等

By Gray
Swift Server Push Notification 配置

Swift Server Push Notification 配置

获取证书 在 Apple Developer 开发者账号 Certificates, Identifiers & Profiles 里选择 Keys。新增一个 key, configure 里选择 Sandbox & Production。下载该 p8 证书,并且保存好(只能下载一次)。 终端 cd 到证书所在路径,输入下面指令。 openssl pkcs8 -nocrypt -in AuthKey_XXXXXXXXX.p8 -out ~/Downloads/key.pem cat key.pem 得到 PRIVATE KEY 字符串,复制好。 服务端配置 服务端有多种技术栈方案,包括 Java、

By Gray
香港游记——一个传统而又现代的城市

香港游记——一个传统而又现代的城市

这是 2024 年的最后一场旅行,从北京到香港,跨越了大半个中国。去香港,一方面是想领略一下它的文化和风光,另一方面是想办一个香港银行卡,买港美股以及海外收付款。 从北京到香港,动卧是一个不错的选择。乘坐 D903 次动车,晚上八点登车,睡一觉,第二天一早就到深圳北了。再从深圳北坐高铁过口岸到香港西九龙,差不多上午九点多就能到达香港。深圳北到西九龙的高铁车次非常多,不用担心买不到票。 密集的建筑 香港给我的初印象就是——这里的楼房真的很密集。不光是住宅区又高又密,商业区的建筑物与建筑物之间也几乎只有街道相隔,很少见到大型的公园或者绿化带。土地利用率很高。这一点和北京差别还是挺大的。北京虽然也是寸土寸金,但是市内绿化面积很高,大型公园也很常见。 街上密集的建筑,让人第一眼看就知道这是香港。 旧与新,传统与现代 在香港,不同地区的风格面貌会相差很多。你既能见到破旧不堪、需要修缮的古老楼房,也能见到银光闪闪、科技感十足的现代化大厦。这种新与旧的切换,传统和现代的反差,总是能给人强烈的震撼。这正是香港的魅力所在。 维多利亚港和中环摩天轮 维多利亚港是香港的中心,是香港旅游

By Gray