如何在Swift中列出符合协议的所有类?

mix*_*y87 16 reflection ios swift swift2

如何列出在Swift中实现给定协议的所有类?

假设我们有一个例子:

protocol Animal {
    func speak()
}

class Cat:Animal {
    func speak() {
        print("meow")
    }
}

class Dog: Animal {
    func speak() {
        print("Av Av!")
    }
}

class Horse: Animal {
    func speak() {
        print("Hurrrr")
    }
}
Run Code Online (Sandbox Code Playgroud)

这是我目前的(不可编译)方法:

func getClassesImplementingProtocol(p: Protocol) -> [AnyClass] {
    let classes = objc_getClassList()
    var ret = [AnyClass]()

    for cls in classes {
        if class_conformsToProtocol(cls, p) {
            ret.append(cls)
        }
    }
    return ret
}

func objc_getClassList() -> [AnyClass] {
    let expectedClassCount = objc_getClassList(nil, 0)
    let allClasses = UnsafeMutablePointer<AnyClass?>.alloc(Int(expectedClassCount))
    let autoreleasingAllClasses = AutoreleasingUnsafeMutablePointer<AnyClass?>(allClasses)
    let actualClassCount:Int32 = objc_getClassList(autoreleasingAllClasses, expectedClassCount)

    var classes = [AnyClass]()
    for i in 0 ..< actualClassCount {
        if let currentClass: AnyClass = allClasses[Int(i)] {
            classes.append(currentClass)
        }
    }

    allClasses.dealloc(Int(expectedClassCount))

    return classes
}
Run Code Online (Sandbox Code Playgroud)

但是在打电话时

getClassesImplementingProtocol(Animal.Protocol) 要么

getClassesImplementingProtocol(Animal) 要么

getClassesImplementingProtocol(Animal.self)

导致Xcode错误:无法将类型(Animal.Protocol).Type的值转换为预期的参数类型'Protocol'.

有人管理得到这个吗?

Col*_*aff 9

由于您使用Objective-C运行时来获取类型内省,因此需要以@objc这种方式添加到代码中:

@objc protocol Animal {
  func speak()
}

class Cat:Animal {
  @objc func speak() {
    print("meow")
  }
}

class Dog: Animal {
  @objc func speak() {
    print("Av Av!")
  }
}

class Horse: Animal {
  @objc func speak() {
    print("Hurrrr")
  }
}
Run Code Online (Sandbox Code Playgroud)

请注意,这种类型的内省可能非常慢.

  • 有什么方法可以获得与结构相似的东西吗? (2认同)