5.OC语法糖总结-@[],@{},@()

语法糖(Syntactic sugar),也译为糖衣语法,是由英国计算机科学家彼得·约翰·兰达(Peter J.Landin)发明的一个术语,指计算机语言中添加的某种语法,这种语法对语言的功能并没有影响,但是更方便程序员使用。通常来说使用语法糖能够增加程序的可读性,从而减少程序代码出错的机会。实际上没有对代码功能没有影响,但是增强我们程序的可读性的语法。

1.枚举绑定数据类型

enum {
        ObjectiveC,
        Java,
        Ruby,
        Python,
        Erlang
};
typedef NSUInteger Language;

2.点语法。OC里面所有的点语法都是。

  1. 属性的@synthesize,我们知道现在连@synthesize也可以省略了,那就是说@property本身已经连带使用了语法糖。
    4.语法简化
    4.1 NSNumber
NSNumber *value;
value  = @12345;
value  = @123.45f;

4.2NSArray

NSArray *array;
array    = @[];     // 空数组
array    = @[a];// 一个对象的数组
array    = @[q,w,e,r]; // 多个对象的数组
// @[] 只能是不可变数组,可变数组NSMutableArray *mutablePlanets = [@[q,w,e,r] mutableCopy];

4.3NSDictionary

NSDictionary  *dict;
dict  = @{};       // 空字典
dict  = @{@"a":@"s"}; // 包含一个键值对的字典
dict   = @{@"a":@"a",@"s":@"s"}; // 多个键值对的字典
// @{}只能是不可变字典,可变字典NSMutableDictionary *dic = [[@{@"a":@"a"}] mutableCopy];

5.对象下标
可以通过下标方式存取数组和字典的数据,类似C语言结构
5.1 NSArray

NSArray *array = @[a,s,d];
id obj  = array[i]; // 通过下标方式获取数组对象,替换原有写法:[array objectAtIndex:i];
array[i]  = newObj; // 也可以直接为数组队形赋值。替换原有写法:[array replaceObjectAtIndex:i withObject:newObj];

5.2NSDictionary

NSDictionary *dict = @{@”b”:b,@”a”:a,@”c”:c};
id obc = dict[@”b”]; // 获取o1队形,替换原有写法:[dic objectForKey:@”b”];
dic[@”c”] = newObj; // 重新为键为@”c”的对象赋值,替换原有写法:[dic setObject:newObj forKey:@”c”];


6.@()用法
在Objective-C中我们可以用@"foo"来创建一个NSString常量,看起来似乎平淡无奇。
但它背后其实比想象的精彩,@可以被理解成一个特殊的宏,其接受一个C字符串作为参数,也可写作@("foo")。
之所以说@是一个特殊的宏,是因为其能根据传入的C字符串类型不同-- C字符串常量或C字符串--在运行时构建返回不同类型的NSString,参见下面的代码:
```Objective-C
char *obtain_c_string(void)
{
  return "c_string";
}
NSLog(@"%@",@"foo".class);
NSLog(@"%@", @("bar").class);
NSLog(@"%@", @(obtain_c_string()).class);
NSLog(@"%@",@"aaa");
NSLog(@"%@",@("aaa"));

输出结果:

__NSCFConstantString
 __NSCFConstantString
 __NSCFString
 aaa
 aaa

可见,如果传入的是C字符串常量,运行时构建的则为NSConstantString;如果传入的是C字符串则创建是NSString。
众所周知,Objective-C 代码里有很多地方需要我们把代码中的一些文法串写成字符串再作为传入参数,比如KVO中的keyPath 参数往往就要传入形如propertyA.propertyB的字符串,从实用角度出发这有两个弊端:写字符串的时候没有代码提示,很容易写错
即便一开始写对了,如果后来相关类重构了,keyPath的参数便失效了,而Xcode Refactor 无法扫描字符串
当我们理解了@(),再加上自定义的宏,上述两个问题便可迎刃而解。

/**
 * # 将宏的参数字符串化,c函数 strchr 返回字符串中第一个'.'字符的位置
*/
#define Keypath(keypath) (strchr(#keypath,'.')+ 1)
// 有代码提示,可以被重构扫描到
[objA addObserver:objB forKeyPath:@Keypath(objA.property1.property2) options:nil context:nil];
点赞