以下代码打印nil
,尽管ListCell
是有效的类。
var lCellClass : AnyClass! = NSClassFromString("ListCell");
println(lCellClass);
文档说该方法返回
由 aClassName 命名的类对象,如果当前没有加载该名称的类,则为
nil
。如果 aClassName 为nil
,则返回nil
。
我还尝试获取当前viewcontroller的NSClassFromString()
,它已加载但仍然获得nil
可能是什么问题?
更新
即使尝试这样做NSClassFromString("Array")
我仍然会nil
答案 0 :(得分:61)
NSClassFromString
函数 适用于(纯的和Objective-C派生的)swift类,但仅限于使用完全限定名称。
例如,在名为MyApp
的模块中,有一个名为Person
的纯swift类:
let personClass: AnyClass? = NSClassFromString("MyApp.Person")
模块名称由“产品模块名称”(PRODUCT_MODULE_NAME
)构建设置定义,通常与目标名称相同(应用了一些规范化,例如删除空格)。
这是非典型的,但如果您要加载的类在当前模块中,并且您在编译时不知道模块名称,例如因为代码是作为多个目标的一部分编译的,所以您可以动态查找包名称,通常对应于模块名称:
let moduleName = Bundle.main.infoDictionary!["CFBundleName"] as! String
let personClass: AnyClass? = NSClassFromString(moduleName + "." + "Person")
除非你在编译时真的不知道,否则我不建议这样做。
有关详情,请参阅Using Swift With Cocoa and Objective-C中的使用带有Objective-C API的Swift类名。
答案 1 :(得分:11)
还可以在Objective-C中指定符号的名称,该名称通过使用@objc注释省略了目标C中的名称空间。如果您更喜欢使用不带模块的名称,只需使用相同的类名:
@objc(Foobar)
class Foobar{
}
NSClassFromString(“Foobar”)将返回Foobar类。
答案 2 :(得分:0)
这可能是您的类没有从NSObject扩展的问题。默认情况下,swift没有超类。
答案 3 :(得分:0)
例如:
仅以 UIViewController 为例,
func getVc(from stringName: String) -> UIViewController? {
let appName = Bundle.main.infoDictionary!["CFBundleName"] as! String
guard let vc = NSClassFromString(appName + "." + stringName) as? UIViewController.Type else {
return nil
}
return vc.init()
}
如果您的应用名称包含“-”或数字,只需将其替换为“_”