ios宏展开

本文深入探讨了RAC中复杂的宏命令,通过一个具体的小demo展示了如何使用JK_Prepare_Subviews宏来遍历并添加多个子视图。解析了宏命令的工作原理,包括参数计数、宏拼接和递归调用等高级技巧。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言:近期看了一下RAC中的宏命令,发现其中的宏命令一环扣一环,环环相扣,把我都给整懵逼了...

 

 

通过深入学习,终于能看懂了,写个小demo秀一波操作.

 


源码:

#define JK_Prepare_Subviews(supperView, ...)\

JK_metamacro_foreach(JK_Add_Subview, supperView, __VA_ARGS__)

#define JK_metamacro_foreach(Marco, supperView, ...)\

JK_metamacro_concat(JK_metamacro_foreach, JK_metamacro_argcount(__VA_ARGS__))(Marco, supperView, __VA_ARGS__)

#define JK_metamacro_foreach20(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18, _19)\

JK_metamacro_foreach19(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18)\

Marco(Parameter, _19, 19)

#define JK_metamacro_foreach19(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18)\

JK_metamacro_foreach18(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17)\

Marco(Parameter, _18, 18)

#define JK_metamacro_foreach18(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17)\

JK_metamacro_foreach17(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16)\

Marco(Parameter, _17, 17)

#define JK_metamacro_foreach17(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16)\

JK_metamacro_foreach16(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15)\

Marco(Parameter, _16, 16)

#define JK_metamacro_foreach16(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15)\

JK_metamacro_foreach15(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14)\

Marco(Parameter, _15, 15)

#define JK_metamacro_foreach15(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14)\

JK_metamacro_foreach14(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13)\

Marco(Parameter, _14, 14)

#define JK_metamacro_foreach14(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13)\

JK_metamacro_foreach13(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12)\

Marco(Parameter, _13, 13)

#define JK_metamacro_foreach13(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12)\

JK_metamacro_foreach12(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11)\

Marco(Parameter, _12, 12)

#define JK_metamacro_foreach12(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11)\

JK_metamacro_foreach11(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10)\

Marco(Parameter, _11, 11)

#define JK_metamacro_foreach11(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10)\

JK_metamacro_foreach10(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9)\

Marco(Parameter, _10, 10)

#define JK_metamacro_foreach10(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8, _9)\

JK_metamacro_foreach9(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8)\

Marco(Parameter, _9, 9)

#define JK_metamacro_foreach9(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7, _8)\

JK_metamacro_foreach8(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7)\

Marco(Parameter, _8, 8)

#define JK_metamacro_foreach8(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6, _7)\

JK_metamacro_foreach7(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6)\

Marco(Parameter, _7, 7)

#define JK_metamacro_foreach7(Marco, Parameter, _0, _1, _2, _3, _4, _5, _6)\

JK_metamacro_foreach6(Marco, Parameter, _0, _1, _2, _3, _4, _5)\

Marco(Parameter, _6, 6)

#define JK_metamacro_foreach6(Marco, Parameter, _0, _1, _2, _3, _4, _5)\

JK_metamacro_foreach5(Marco, Parameter, _0, _1, _2, _3, _4)\

Marco(Parameter, _5, 5)

#define JK_metamacro_foreach5(Marco, Parameter, _0, _1, _2, _3, _4)\

JK_metamacro_foreach4(Marco, Parameter, _0, _1, _2, _3)\

Marco(Parameter, _4, 4)

#define JK_metamacro_foreach4(Marco, Parameter, _0, _1, _2, _3)\

JK_metamacro_foreach3(Marco, Parameter, _0, _1, _2)\

Marco(Parameter, _3, 3)

#define JK_metamacro_foreach3(Marco, Parameter, _0, _1, _2)\

JK_metamacro_foreach2(Marco, Parameter, _0, _1)\

Marco(Parameter, _2, 2)

#define JK_metamacro_foreach2(Marco, Parameter, _0, _1)\

JK_metamacro_foreach1(Marco, Parameter, _0)\

Marco(Parameter, _1, 1)

#define JK_metamacro_foreach1(Marco, Parameter, _0)\

Marco(Parameter, _0, 0)

#define JK_Add_Subview(supperView, suberView, Index)\

[supperView addSubview:suberView];

#define JK_metamacro_concat(A, B) \

JK_metamacro_concat_(A, B)

#define JK_metamacro_concat_(A, B) A ## B

#define JK_metamacro_argcount(...) \

JK_metamacro_at(20, __VA_ARGS__, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1)

#define JK_metamacro_at(N, ...) \

JK_metamacro_concat_(JK_metamacro_at, 20)(__VA_ARGS__, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1)

#define JK_metamacro_at20(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18, _19, ...) JK_metamacro_head(__VA_ARGS__)

#define JK_metamacro_head(...) \

JK_metamacro_head_(__VA_ARGS__, 0)

//得到结果: 参数个数

#define JK_metamacro_head_(FIRST, ...) FIRST

 


试用一下:

- (void)viewDidLoad {

[super viewDidLoad];

NSMutableArray *marr = [NSMutableArray array];

CGFloat margin = 5;

CGFloat y;

CGFloat height = 20;

for (int i = 0; i < 21; i++) {

y = height * i + margin * (i + 1);

UILabel *label = [[UILabel alloc] initWithFrame:CGRectMake(0, y, 375, height)];

label.text = [NSString stringWithFormat:@"第%d行",i];

label.backgroundColor = [UIColor colorWithRed:arc4random_uniform(255)/255.0 green:arc4random_uniform(255)/255.0 blue:arc4random_uniform(255)/255.0 alpha:1];

[marr addObject:label];

}

JK_Prepare_Subviews(self.view, marr[0], marr[1], marr[2], marr[3], marr[4], marr[5] ,marr[6], marr[7], marr[8], marr[9], marr[10], marr[11], marr[12], marr[13], marr[14], marr[15], marr[16], marr[17], marr[18], marr[19]);

}

 

效果图


是不是看不懂?

 

实例解析:传入三个参数:a, b, c

JK_Prepare_Subviews(supperView, a, b, c)

#define JK_Prepare_Subviews(supperView, ...)\

JK_metamacro_foreach(JK_metamacro_foreach, supperView, __VA_ARGS__)

第一次展开

JK_metamacro_foreach(JK_metamacro_foreach, supperView, a, b, c)

 


#define JK_metamacro_foreach(Marco, supperView, ...)\

JK_metamacro_concat(JK_metamacro_foreach, JK_metamacro_argcount(__VA_ARGS__))(Marco, supperView, __VA_ARGS__)

第二次展开

JK_metamacro_concat(JK_metamacro_foreach, JK_metamacro_argcount(a, b, c))(JK_Add_Subview, supperView, a, b, c)

参数中的宏展开

#define JK_metamacro_argcount(...) \

JK_metamacro_at(20, a, b, c, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1)

#define JK_metamacro_at(N, ...) \

JK_metamacro_concat_(JK_metamacro_at, 20)(a, b, c, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1)

截取之后就剩下了3, 2, 1  {_0 ~ _19这20个参数(a, b, c, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4)在这边没有传递到下一个宏中}

#define JK_metamacro_at20(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18, _19, ...) JK_metamacro_head(3, 2, 1)

#define JK_metamacro_head(...) \

JK_metamacro_head_(3, 2, 1, 0)

得到结果: 参数个数

#define JK_metamacro_head_(FIRST, ...) 3

 


#define JK_metamacro_concat(A, B) \

JK_metamacro_concat_(A, B)

第三次展开

JK_metamacro_concat_(JK_metamacro_foreach, 3)(JK_Add_Subview, supperView, a, b, c)

 


#define JK_metamacro_concat_(A, B) A ## B      //拼接宏,结果为AB

第四次展开

JK_metamacro_foreach3(JK_Add_Subview, supperView, a, b, c)

#define JK_metamacro_foreach3(Marco, Parameter, _0, _1, _2)\

JK_metamacro_foreach2(Marco, Parameter, _0, _1)\

Marco(Parameter, _2, 2)

 


第五次展开

JK_metamacro_foreach2(JK_Add_Subview, supperView, a, b)

JK_Add_Subview(supperView, c)

#define JK_metamacro_foreach2(Marco, Parameter, _0, _1)\

JK_metamacro_foreach1(Marco, Parameter, _0)\

Marco(Parameter, _1, 1)

 


第六次展开

JK_metamacro_foreach1(JK_Add_Subview, supperView, a)

JK_Add_Subview(supperView, b)

JK_Add_Subview(supperView, c)

第七次展开

JK_Add_Subview(supperView, a)

JK_Add_Subview(supperView, b)

JK_Add_Subview(supperView, c)

 


结果:

宏定义其实就是是替换,

JK_Prepare_Subviews(supperView, a, b, c)

层层展开,最后替换之后就变成了

JK_Add_Subview(supperView, a)

JK_Add_Subview(supperView, b)

JK_Add_Subview(supperView, c)

最后附上Demo地址:戳我 

### 结构体定义中 `typedef` 与宏展开的作用及区别 在 C 语言中,`typedef` 和宏定义(`#define`)都可以用于为结构体定义别名,从而简化结构体类型的使用。然而,两者在实现机制、类型安全性和作用范围等方面存在显著差异。 #### 使用 `typedef` 定义结构体别名 `typedef` 是 C 语言的关键字,用于为已有数据类型定义新的名称。它可以为结构体创建别名,使代码更简洁、可读性更强。例如: ```c typedef struct { char name[20]; int age; } Student; Student stu1; ``` 上述代码中,`Student` 是 `struct { char name[20]; int age; }` 的别名。与宏定义不同,`typedef` 在编译阶段处理,编译器可以识别其类型,因此提供了更强的类型安全性[^4]。 #### 使用宏定义(`#define`)定义结构体别名 宏定义是预处理器指令,用于在编译前进行文本替换。可以通过宏定义为结构体类型起别名,例如: ```c #define Student struct student Student { char name[20]; int age; }; Student stu1; ``` 这种方式虽然也能简化结构体的使用,但宏定义仅在预处理阶段进行简单的文本替换,不具备类型检查功能。如果宏定义不当,可能导致难以调试的类型错误[^1]。 #### `typedef` 与宏定义在结构体定义中的区别 | 特性 | `typedef` | `#define` | |------|-----------|-----------| | 处理阶段 | 编译阶段 | 预处理阶段 | | 类型检查 | 支持,编译器可识别类型 | 不支持,仅文本替换 | | 作用范围 | 限定于当前作用域 | 全局有效 | | 可读性 | 更高,结构清晰 | 易引发歧义,尤其在复杂宏中 | `typedef` 的类型检查机制使其在结构体定义中更具优势。例如在函数参数传递或结构体嵌套定义中,`typedef` 能确保类型一致性,而宏定义则可能因替换错误导致类型不一致问题[^1]。 #### 示例对比 使用 `typedef` 定义结构体: ```c typedef struct { int x; int y; } Point; Point p1; ``` 使用宏定义定义结构体: ```c #define Point struct point Point { int x; int y; }; Point p1; ``` 两者在使用上看似相似,但 `typedef` 提供了编译器可识别的类型信息,而宏定义仅在预处理阶段进行替换,可能导致类型冲突[^2]。 #### 实际应用场景 在系统级编程、驱动开发或嵌入式开发中,`typedef` 被广泛用于定义固定宽度的类型别名,如 `uint8_t`、`size_t` 等,这些类型别名增强了代码的可移植性和可读性[^3]。而宏定义则更多用于条件编译、常量定义或简单的文本替换,较少用于结构体别名定义。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值