一句话笔记,某段时间内遇到或看到的某个可记录的点。 2016-8-10
- 一个获取KVO keyPath 的宏
- Category 可以声明属性,但是由于不能生成实例变量,才等同于不能添加属性
- 删除字符串的前后空格(NSCharacterSet)
- NSProxy 初探
1、一个获取KVO keyPath 的宏
#define YSGetKVOKeyPath(objc,keyPath) @(((void)objc.keyPath,#keyPath))
[self.tableView addObserver:self forKeyPath:YSGetKVOKeyPath(self.tableView, contentOffset) options:NSKeyValueObservingOptionNew context:nil];
[self.tableView removeObserver:self forKeyPath:YSGetKVOKeyPath(self.tableView, contentOffset)];
这样我们就不需要担心,写的麻烦或者说直接写路径名 出现拼写错误的的情况啦。
2、 Category 可以声明属性,但由于不生成实例变量,才等同于不能生成属性
通常我们常说 Category 内是不能直接添加属性的,我很好奇的去再尝试下,发现是可以声明属性的,问题来了,那为什么我们通常说的不能添加属性呢。
// .h
@property (nonatomic, copy) NSString *testName;
// .m
- (NSString *)testName {
return _testName;
}
上述这种情况,正常来说是可以return _testName
,但是在Category 里面是不可以哦,经查找资料发现,在动态添加成员变量的时候,是有限制的。
在Objective-C提供的runtime函数中,确实有一个class_addIvar()函数用于给类添加成员变量,但是文档中特别说明:
This function may only be called after objc_allocateClassPair and before objc_registerClassPair. Adding an instance variable to an existing class is not supported.
表面这个函数只能在“构建一个类的过程中”调用。一旦完成类定义,就不能再添加成员变量了。经过编译的类在程序启动后就runtime
加载,没有机会调用addIvar
。
再回过头来来,记住
@property = ivar + getter + setter;
“属性” (property) 作为 Objective-C 的一项特性,主要的作用就在于封装对象中的数据。 Objective-C 对象通常会把其所需要的数据保存为各种实例变量。实例变量一般通过“存取方法”(access method)来访问。其中,“获取方法” (getter
) 用于读取变量值,而“设置方法” (setter
)用于写入变量值。
详细的说:我们每次在增加一个属性,系统都会在 ivar_list
中添加一个成员变量的描述,在 method_list
中增加setter
与 getter
方法的描述,在属性列表中增加一个属性的描述,然后计算该属性在对象中的偏移量,然后给出 setter
与 getter
方法对应的实现,在setter
方法中从偏移量的位置开始赋值,在 getter
方法中从偏移量开始取值,为了能够读取正确字节数,系统对象偏移量的指针类型进行了类型强转.
来自@property 的本质是什么?ivar、getter、setter 是如何生成并添加到这个类中的
在类别中声明属性只是相等于声明了
setter
和getter
方法,但是没有实现,而实现需要实例变量配合。
所以这整个过程走不完,所以要真正的生成属性需要用到objc_setAssociatedObject
和objc_getAssociatedObject
关联对象啦。
3、删除字符串的前后空格(NSCharacterSet)
直接做法,同时还只是删除前后一个空格的
- (NSString *)cutLeadAndTrialOneBlankCharacter:(NSString *)string {
NSMutableString *resultString = [NSMutableString string];
if (string.length > 2) {
// 第一个 BlankCharacter
if ([[resultString substringToIndex:1] isEqualToString:@" "]) {
[resultString deleteCharactersInRange:NSMakeRange(0, 1)];
}
// 最后一个 BlankCharacter
if ([[resultString substringFromIndex:resultString.length - 1] isEqualToString:@" "]) {
[resultString deleteCharactersInRange:NSMakeRange(resultString.length - 1, 1)];
}
}
return resultString.copy;
}
NSCharacterSet 的做法,去除前后的空格 + 回车 或 去除前后的空格
[NSCharacterSet whitespaceAndNewlineCharacterSet];
[NSCharacterSet whitespaceCharacterSet];
- (NSString *)cutLeadAndTrialBlankCharacter:(NSString *)string {
NSString *resultString = [string stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]]; //去除前后的空格 + 回车
// NSString *resultString = [string stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]]; // 去除前后的空格
return resultString;
}
注意多了解些 NSCharacterSet 会有很多意想不到的效果哦。
4、NSProxy 初探
今天看到NSProxy——少见却神奇的类 这篇文章后,发现对 NSProxy 这个了解确实很模糊,虽说可能暂时不用,但是其用法还是有一个印象好。
言简意赅: NSProxy是NSObject的一个虚类,目的为了来“代理”一个真实对象的。我们可以让NSProxy的实现类能够很好地“代理”NSObject子类,并且把NSObject协议抽象出来,让他们能够共享某些行为。
@interface NSProxy
具体用处: 我们可以通过继承它,并重写这两个方法以实现消息转发到另一个实例。
用到的就是
// 方法参数代表未知的选择子,若当前接收者能找到备援对象,则将其返回,若找不到,就返回nil。
- (NSMethodSignature *)methodSignatureForSelector:(SEL)sel;
// 完整的消息转发
- (void)forwardInvocation:(NSInvocation *)anInvocation;
再次翻看之前的一个笔记有效方法12-理解消息转发机制,重新理解这个过程,另外具体的实现可以看看 NSProxy——少见却神奇的类 中的 Demo 的用法。