美文网首页
Objective-C进阶

Objective-C进阶

作者: again_onceagain | 来源:发表于2020-06-01 11:31 被阅读0次

    分类

    Q1: 你用分类都做了那些事?

    • 声明私有方法
    • 分解体积庞大的类文件
    • 把Framework的私有方法公开化

    Q2: 特点

    • 运行时决议
    • 可以为系统类添加分类

    Q3: 分类中可以添加哪些内容?

    • 实例方法
    • 类方法
    • 协议
    • 属性(只定义了getter/setter方法)

    分类数据结构:

    image.png
    1. 分类添加的方法可以“覆盖”原类方法
    2. 同名分类方法谁能生效取决于编译顺序(最后编译的类最先生效)
    3. 名字相同的分类会引起编译报错

    关联对象

    Q1: 能否给分类添加“成员变量”?

    id objc_getAssociatedObject(id object, const void * key)
    void objc_setAssociatedObject(id object, const void * key, id value, objc_AssociationPolicy policy)
    voi objc_removeAssociatedObjects(id object)
    

    Q2: 成员变量添加到哪呢?

    关联对象的本质:
    关联对象由AssociationsManager管理并在AssociationHashMap存储。
    所有对象的关联内容都在同一个全局容器中。

    image.png image.png

    扩展(Extension)

    Q1: 一般用扩展做什么?

    • 声明私有属性
    • 声明私有方法
    • 声明私有成员变量

    Q2:特点

    • 编译时决议
    • 只以声明的形式存在,多数情况下寄生于宿主类的.m中。
    • 不能为系统类添加扩展。

    代理(Delegate)

    • 准确的说是一种软件设计模式

    • ios当中以@protocol形式提现

    • 传递方式一对一

      image.png
    • 一般声明以weak以规避循环引用

      image.png

    通知(NSNotification)

    • 是使用观察者模式来实现的用于跨层传递消息的机制。
    • 传递方式为一对多
      image.png

    Q1: 如何实现通知机制?


    image.png

    KVO

    • KVO是Key-value observing的缩写。
    • KVO是Objective-C对观察者设计模式的又一实现。
    • Apple使用了isa混写(isa-swizzling)来实现KVO。
    - (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
    - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(nullable void *)context API_AVAILABLE(macos(10.7), ios(5.0), watchos(2.0), tvos(9.0));
    - (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath;
    
    image.png

    重写的setter添加的方法

    - (void)willChangeValueForKey:(NSString *)key;
    - (void)didChangeValueForKey:(NSString *)key;
    
    - (void)setValue:(int)value {
        //直接为成员变量赋值
        [self willChangeValueForKey:@"value"];
        _value += 1;
        [self didChangeValueForKey:@"value"];
    }
    

    Q1: 通过KVC设置value, KVO能否生效?
    YES.(KVC实现原理)

    Q2: 通过成员变量直接赋值value, KVO能否生效?
    NO. 但是可以手动触发KVO, 代码如下

    - (void)increase {
        //直接为成员变量赋值
        [self willChangeValueForKey:@"value"];
        _value += 1;
        [self didChangeValueForKey:@"value"];
    }
    

    总结

    • 使用setter方法改变值KVO才会生效。
    • 使用setValue:forKey:改变值KVO才会生效。
    • 成员变量直接修改需手动添加KVO才会生效。

    KVC

    KVC是Key-value coding的缩写

    - (nullable id)valueForKey:(NSString *)key;
    - (void)setValue:(nullable id)value forKey:(NSString *)key;
    

    KVC破坏了面向对象的编程思想(封装性)

    实现流程

    image.png

    Accessor Method

    • <getKey>
    • <key>
    • <isKey>

    Instance var

    • _key
    • _isKey
    • key
    • isKey
    image.png

    属性关键字

    • 读写权限
      • readonly
      • readwrite
    • 原子性
      • atomic
      • nonatomic
    • 引用计数
      • retain/strong
      • assign/unsafe_unretained
      • weak
      • copy

    assgin

    • 修饰基本数据类型, 如int, BOOL等。
    • 修饰对象类型时,不改变其引用计数。
    • 会产生悬悬垂指针。

    weak

    • 不改变被修饰对象的引用计数。
    • 所知对象在呗释放之后会自动置为nil。

    copy

    • 浅拷贝


      image.png

    浅拷贝就是对内存地址的复制,让目标对象指针和源对象指向同一片内存空间。

    • 深拷贝
      image.png
      深拷贝让目标对象指针和源对象指针指向两片内容相同的内存空间。

    如何区分深拷贝vs浅拷贝

    • 是否开辟了新的内存空间
    • 是否影响了引用计数
    image.png
    • 可变对象的copy和mutableCopy都是深拷贝。
    • 不可变对象的copy是浅拷贝, mutableCopy是深拷贝。
    • copy方法返回的都是不可变对象。
    • mutableCopy方法返回的都是可变对象。

    @property(copy) NSMutableArray *array?

    • 如果赋值过来的是NSMutableArray, copy之后还是NSArray
    • 如果赋值过来的NSArray, copy之后是NSArray

    MRC下如何重写retain修饰变量的Setter方法?

    if (_obj != obj) {
      [_obj release];
      _obj = [obj retain];
    }
    

    请简述分类的实现原理。
    KVO的实现原理是怎么样的?
    能否为分类添加成员变量?

    相关文章

      网友评论

          本文标题:Objective-C进阶

          本文链接:https://www.haomeiwen.com/subject/wjsfzhtx.html