一句话笔记(05)

一句话笔记,某段时间内遇到或看到的某个可记录的点。 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中增加settergetter 方法的描述,在属性列表中增加一个属性的描述,然后计算该属性在对象中的偏移量,然后给出 settergetter方法对应的实现,在setter方法中从偏移量的位置开始赋值,在 getter 方法中从偏移量开始取值,为了能够读取正确字节数,系统对象偏移量的指针类型进行了类型强转.

来自@property 的本质是什么?ivar、getter、setter 是如何生成并添加到这个类中的

在类别中声明属性只是相等于声明了settergetter 方法,但是没有实现,而实现需要实例变量配合。

所以这整个过程走不完,所以要真正的生成属性需要用到objc_setAssociatedObjectobjc_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  

具体用处: 我们可以通过继承它,并重写这两个方法以实现消息转发到另一个实例。

一句话笔记(05)_第1张图片
NSProxy Methods

用到的就是

// 方法参数代表未知的选择子,若当前接收者能找到备援对象,则将其返回,若找不到,就返回nil。
- (NSMethodSignature *)methodSignatureForSelector:(SEL)sel;
// 完整的消息转发
- (void)forwardInvocation:(NSInvocation *)anInvocation;
一句话笔记(05)_第2张图片
消息转发全流程

再次翻看之前的一个笔记有效方法12-理解消息转发机制,重新理解这个过程,另外具体的实现可以看看 NSProxy——少见却神奇的类 中的 Demo 的用法。

你可能感兴趣的:(一句话笔记(05))