objc非主流代码技巧

最用心的转载了…

我是前言

看开源代码时,总会看到一些大神级别的代码,给人眼前一亮的感觉,多数都是被淡忘的C语言语法,总结下objc写码中遇到的各类非主流代码技巧和一些妙用:

  1. [娱乐向]objc最短的方法声明
  2. [C]结构体的初始化
  3. [C]三元条件表达式的两元使用
  4. [C]数组的下标初始化
  5. [objc]可变参数类型的block
  6. [objc]readonly属性支持扩展的写法
  7. [C]小括号内联复合表达式
  8. [娱乐向]奇葩的C函数写法
  9. [Macro]预处理时计算可变参数个数
  10. [Macro]预处理断言
  11. [多重]带自动提示的keypath宏

    [娱乐向]objc最短的方法声明

    先来个娱乐向的。

    方法声明时有一下几个trick:

返回值的- (TYPE)如果不写括号,编译器默认认为是- (id)类型:

 1. init;
 2. (id)init; // 等价于

同理,参数如果不写类型默认也是id类型:

 3. (void)foo:arg;
 4. (void)foo:(id)arg; // 等价于

还有,有多参数时方法名参数提示语可以为空

 5. (void):(id)arg1 :(id)arg2;
 6. (void)foo:(id)arg1 bar:(id)arg2; // 省略前

综上,最短的函数可以写成这样:

 7. _;   // 没错,这是一个oc方法声明
 8. :_;  // 这是一个带一个参数的oc方法声明
// 等价于
 9. (id)_;
 10. (id) :(id)_;

PS: 方法名都没的方法只能靠performSelector来调用了,selector是”:”

[C]结构体的初始化

// 不加(CGRect)强转也不会warning
CGRect rect1 = {1, 2, 3, 4};
CGRect rect2 = {.origin.x=5, .size={10, 10}}; // {5, 0, 10, 10}
CGRect rect3 = {1, 2}; // {1, 2, 0, 0}

[C]三元条件表达式的两元使用

三元条件表达式?:是C中唯一一个三目运算符,用来替代简单的if-else语句,同时也是可以两元使用的:

NSString *string = inputString ?: @"default";
NSString *string = inputString ? inputString : @"default"; // 等价

利用这个特性,我们还脑洞出了一个一行代码的 block 调用,平时我们的 block 是这样调用:

if (block0) {
  block0();
}
// or
if (block1) {
  int result = block1(1, 2);
}

居然可以简化成下面的样子:

!block0 ?: block0();
int result = !block1 ?: block1(1, 2);

[C]数组的下标初始化

const int numbers[] = {
    [1] = 3,
    [2] = 2,
    [3] = 1,
    [5] = 12306
};
// {0, 3, 2, 1, 0, 12306}

这个特性可以用来做枚举值和字符串的映射

typedef NS_ENUM(NSInteger, XXType){
    XXType1,
    XXType2
};
const NSString *XXTypeNameMapping[] = {
    [XXType1] = @"Type1",
    [XXType2] = @"Type2"
};

[objc]可变参数类型的block

一个block像下面一样声明:

void(^block1)(void);
void(^block2)(int a);
void(^block3)(NSNumber *a, NSString *b);

如果block的参数列表为空的话,相当于可变参数(不是void)

void(^block)(); // 返回值为void,参数可变的block
block = block1; // 正常
block = block2; // 正常
block = block3; // 正常
block(@1, @"string");  // 对应上面的block3
block(@1); // block3的第一个参数为@1,第二个为nil

这样,block的主调和回调之间可以通过约定来决定block回传回来的参数是什么,有几个。如一个对网络层的调用:

 11. (void)requestDataWithApi:(NSInteger)api block:(void(^)())block {
    if (api == 0) {
        block(1, 2);
    }
    else if (api == 1) {
        block(@"1", @2, @[@"3", @"4", @"5"]);
    }
}

主调者知道自己请求的是哪个Api,那么根据约定,他就知道block里面应该接受哪几个参数:

[server requestDataWithApi:0 block:^(NSInteger a, NSInteger b){
    // ...
}];
[server requestDataWithApi:1 block:^(NSString *s, NSNumber *n, NSArray *a){
    // ...
}];

这个特性在Reactive Cocoa-combineLatest:reduce:等类似方法中已经使用的相当好了。

 12. (RACSignal *)combineLatest:(id<NSFastEnumeration>)signals reduce:(id (^)())reduceBlock;

[objc]readonly属性支持扩展的写法

@interface Sark : NSObject
@property (nonatomic, readonly) NSArray *friends;
@end

.m中可以使用_friends来使用自动合成的这个变量,但假如:

习惯使用self.来set实例变量时(只合成了getter) 希望重写getter进行懒加载时(重写getter时则不会生成下划线的变量,除非手动@synthesize) 允许子类重载这个属性来修改它时(编译报错属性修饰符不匹配)

这种readonly声明方法就行不通了,所以下面的写法更有通用性:

@interface Sark : NSObject
@property (nonatomic, readonly, copy/*加上setter属性修饰符*/) NSArray *friends;
@end

如想在.m中像正常属性一样使用:

@interface Sark ()
@property (nonatomic, copy) NSArray *friends;
@end

子类化时同理。iOS SDK中很多地方都用到了这个特性。

[C]小括号内联复合表达式

A compound statement enclosed in parentheses原谅我的渣翻译- -,来自《gcc官方对此的说明》,源自gcc对c的扩展,如今被clang继承。

RETURN_VALUE_RECEIVER = {(
    // Do whatever you want
    RETURN_VALUE; // 返回值
)};

于是乎可以发挥想象力了:

self.backgroundView = ({
    UIView *view = [[UIView alloc] initWithFrame:self.view.bounds];
    view.backgroundColor = [UIColor redColor];
    view.alpha = 0.8f;
    view;
}); // 我的这篇博客就用到这个写法[iOS 一步一步带你实现引导页]

有点像block和内联函数的结合体,它最大的意义在于将代码整理分块,将同一个逻辑层级的代码包在一起;同时对于一个无需复用小段逻辑,也免去了重量级的调用函数,如:

self.result = ({
    double result = 0;
    for (int i = 0; i <= M_2_PI; i+= M_PI_4) {
        result += sin(i);
    }
    result;
});

这样使得代码量增大时层次仍然能比较明确。

PS: 返回值和代码块结束点必须在结尾

[娱乐向]奇葩的C函数写法

int sum(a,b)
int a; int b;
{
    return a + b;
}

[Macro]预处理时计算可变参数个数

#define COUNT_PARMS2(_a1, _a2, _a3, _a4, _a5, RESULT, ...) RESULT
#define COUNT_PARMS(...) COUNT_PARMS2(__VA_ARGS__, 5, 4, 3, 2, 1)
int count = COUNT_PARMS(1,2,3); // 预处理时count==3

[Macro]预处理断言

下面的断言在编译前就生效

#define C_ASSERT(test) \
    switch(0) {          case 0:          case test:;    }

如断言上面预处理时计算可变参数个数:

C_ASSERT(COUNT_PARMS(1,2,3) == 2);

[多重]带自动提示的keypath宏

源自Reactive Cocoa中的宏:

#define keypath2(OBJ, PATH) \
    (((void)(NO && ((void)OBJ.PATH, NO)), # PATH))

《介绍RAC宏的文章》

逗号表达式

逗号表达式取后值,但前值的表达式参与运算,可用void忽略编译器警告

int a = ((void)(1+2), 2); // a == 2

逻辑最短路径

之前的文章没有弄清上面宏中NO&&NO的含义,其实这用到了编译器优化的特性:

if (NO && [self shouldDo]/*不执行*/) {
    // 不执行
}

编译器知道在NO后且什么的结果都是NO,于是后面的语句被优化掉了。也就是说keypath宏中这个NO && ((void)OBJ.PATH, NO)就使得在编译后后面的部分不出现在最后的代码中,于是乎既实现了keypath的自动提示功能,又保证编译后不执行多余的代码。

References

https://gcc.gnu.org/onlinedocs/gcc/Statement-Exprs.html

原文链接[email protected]

时间: 2024-11-11 03:39:33

objc非主流代码技巧的相关文章

vim 折叠代码技巧汇总

以下命令输入的方式: 如zo命令,先按z键,松开后按o键即可展开折叠. 一.打开.关闭折叠 zo 展开折叠,只展开最外层的折叠. zO 对所在范围内所有嵌套的折叠点展开,包括嵌套折叠. zc 折叠,只折叠最外层的折叠 zC 对所在范围内所有嵌套的折叠点进行折叠,包括嵌套的所有折叠. zm 这将折叠更多 (M-ore).你可以重复 “zr” 和 “zm” 来打开和关闭若干层嵌套的折叠,不然得一个一个的用zc来折叠. 如果你有一个嵌套了好几层深的折叠,你可以用这个命令把它们全部打开: zM 这将增加

移动平台前端开发之WebApp代码技巧

1.首先我们来看看webkit内核中的一些私有的meta标签,这些meta标签在开发webapp时起到非常重要的作用 <meta content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=0;" name="viewport" /> <meta content="yes" name="apple-mobile-web

优化PHP代码技巧的小结

优化PHP代码技巧的小结1. 如果一个方法能被静态,那就声明他为静态的,速度可提高 1/4;2. echo 的效率高于 print,因为 echo 没有返回值,print 返回一个整型;3. 在循环之前设置循环的最大次数,而非在在循环中;4. 销毁变量去释放内存,特别是大的数组;5. 避免使用像__get, __set, __autoload 等魔术方法;6. requiere_once()比较耗资源;7. 在 includes 和 requires 中使用绝对路径,这样在分析路径花的时间更少;

php优化代码技巧

1. 如果一个方法可静态化,就对它做静态声明.速率可提升至 4 倍. 2. echo 比 print 快. 3. 使用 echo 的多重参数(译注:指用逗号而不是句点)代替字符串连接. 4. 在执行 for 循环之前确定最大循环数,不要每循环一次都计算最大值对.于遍历同样一个数组,foreach速度最快,最慢的则是while.foreach比while大约快20"30左右. 5. 注销那些不用的变量尤其是大数组,以便释放内存. 6. 尽量避免使用 __get,__set,__autoload.

FireFox调试代码技巧

本文版权归 csdn DyncRole 所有,此处为技术收藏,如有再转请标明原创作者及出处,以示尊重! 作者:DyncRole 原文:http://blog.csdn.net/qqhjqs/article/details/41889413 这些天做的任务出现好多的错误,但是我也局限于再开发工具eclipse上调试,但是传到前台的数据没有办法调试,找经理来解决问题,被吵了,同时也学到一些小的技巧,在这里记下来,不是很全面,以后学到更多再来更新. 打开火狐浏览器,按"F12",页面下方会出

Javascript 优化项目代码技巧之语言基础(一)

1.全局变量污染与变量提升2.数据类型3.特殊值(NaN.undefined.null)4. === 与 ==5.没有真正的数组6.避免使用with与eval7.消除switch歧义8.不要省略块标志 { } Javascript的弱类型以及函数作用域等规则使用编写Javascript代码极为容易,但是编写可维护.高质量的代码却变得十分困难,这个系列的文章将总结在项目开发过程中,能够改善代码可读性.可维护性及优化运行性能的一系列技巧.     如有问题,请不吝指出,非常感谢:如果喜欢,右下角点个

CSS 代码技巧与维护 ★ Mozilla Hacks – the Web developer blog

原文链接:https://hacks.mozilla.org/2016/05/css-coding-techniques/ 译文链接 :http://www.zcfy.cc/article/css-coding-techniques-x2605-mozilla-hacks-8211-the-web-developer-blog-1244.html 最近,我发现许多人被CSS难倒,无论是新手还是有经验的开发者.自然地,他们就希望能有一种更好的语言来代替它,CSS预处理器就是从这种想法中诞生的.一些

从lighttpd学到的代码技巧

平时写开脚本,很多时候我们都可以不怎样注意效率,但是看c代码的时候,你会发现,才意思自己真的是一个coder啦 1,单位转换 (根据传入的数返回相应的kb,mb,gb等等) 可能我们直觉来想就会这样做啦 if(num>1024*1024*1024){ return num/1024/1024/1024+"gb" }elseif(num>1024*1024){ return num/1024/1024+"mb" }else{ return num/1024

使用DOS打Oracle代码技巧

初学者学习Oracle时一般会用SQP*Plus打Oracle代码,但是Oracle10的是不可以复制黏贴操作,所以需要用DOS来操作,更简单,当然SQL*Plus也有自己的优点 要使用DOS,先要配置环境变量,win7系统是我的电脑->高级系统设置->高级->环境变量,然后在系统变量里,找到Path,没有,就自己添加上去 如图,需要添加%oracle_home%/bin 配置好后,直接window键+R,跳出运行,输入cmd,回车,输入sqlplus/nolog 直接进入 然后其他数据