美文网首页iOS DeveloperiOS开发
利用Associated Object给Swift扩展添加属性及

利用Associated Object给Swift扩展添加属性及

作者: 周二可 | 来源:发表于2017-06-16 11:46 被阅读409次

    Associated Object的使用

    熟悉OC的人估计都知道如何通过Category给已有的类添加属性,那就是通过runtime的Associated Object这种方式。而在Swift中这种方法依然有效,对应的API如下

    C

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

    Swift

    public func objc_getAssociatedObject(_ object: Any!,
                                         _ key: UnsafeRawPointer!) -> Any!
    
    public func objc_setAssociatedObject(_ object: Any!,
                                         _ key: UnsafeRawPointer!,
                                         _ value: Any!,
                                         _ policy: objc_AssociationPolicy)
    
    

    需要注意的是在Swift3.0之前对于一些C语言的API需要传void *的指针时,swif中对应的是UnsafePointer<Void>。3.0之后的版本中有了一个新的类型来处理这些指针:UnsafeRawPointer

    另外贴一个书中具体实现的例子

    // MyClass.swift
    class MyClass {
    }
    
    // MyClassExtension.swift
    private var key: Void?
    
    extension MyClass {
        var title: String? {
            get {
                return objc_getAssociatedObject(self, &key) as? String
            }
            
            set {
                objc_setAssociatedObject(self,
                                         &key, newValue,
                                         .OBJC_ASSOCIATION_RETAIN_NONATOMIC)
            }
        }
    }
    
    // 测试
    func printTitle(_ input: MyClass) {
        if let title = input.title {
            print("Title: \(title)")
        } else {
            print("没有设置")
        }
    }
    
    let a = MyClass()
    printTitle(a)
    
    a.title = "tips"
    printTitle(a)
    

    这里需要注意的是private var key: Void?这种写法。根据上面runtime里的函数我们能看出来key这个参数的类型应该是const void *就是一个指向任意类型常量的指针。在OC中这个key有多种写法

    1、static void *key = &key;
    2、static NSString *key = @"key"; 
    3、static char key;
    

    第一种写法可能不太好理解(C语言够呛的我掩面走过),我查了下了解到这是一个静态无类型指针key然后用指针的地址赋给key就是一个初始化了。也就是一个指向指针的指针。

    然后我们可以根据上面OC中的第二种写法写出对应的Swift版本private var key = "key",个人感觉这种更加直观可读。
    另外这篇文章中的处理方法是不正确的。。

    Swift中使用C语言指针

    具体使用方法可以参考这篇文章以及书中UnsafePointer这章的相关内容,这里记录下一些需要注意的地方。

    1. Swift中大体上分UnsafePointer<Type>, UnsafeMutablePointer<Type>两种数据类型对应C中的指针。这个Type可以是CInt,CBool,CChar这些类型,分别对应C中的相关类型。
      UnsafePointer<Type>对应C中的const Type *,不允许修改指针的值
      UnsafeMutablePointer<Type>对应Type *,允许修改
    2. Swift3.0之前存在UnsafePointer<Void>这个类型,在3.0之后使用UnsafeRawPointer这个类型来处理。
    3. 对于&这个操作来说,它需要一个var变量,比如上面例子中的private var key: Void?,把var换成let就会报错。
    cannot pass immutable value as inout argument: 'key' is a 'let' constant
    
    1. 当需要在Swift3中打印地址的时候可以参考这篇文章的下面这些方法
    // 打印引用类型
    func addressHeap<T: AnyObject>(o: T) -> Int {
        return unsafeBitCast(o, to: Int.self)
    }
    // 打印值类型
    func address(o: UnsafeRawPointer) -> Int {
        return Int(bitPattern: o)
    }
    

    unsafeAddressOf()这个方法被移除了。

    相关文章

      网友评论

        本文标题:利用Associated Object给Swift扩展添加属性及

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