use*_*178 8 class instantiation swift
我可以在ObjC中从string中实例化类.例如,我通过子类化UITableViewCell定义了新的DBCell类,并使用这些代码从name实例化类:
DBCell *cell=[tableView dequeueReusableCellWithIdentifier:cellClassname];
if (cell==nil) {
Class cellClass=NSClassFromString(cellClassname);
cell=[cellClass alloc];
cell=[cell initWithStyle:cellStyle reuseIdentifier:cellClassname];
}
Run Code Online (Sandbox Code Playgroud)
现在我需要将代码迁移到Swift,我在Swift中重新定义了DBCell类:
class DBCell: UITableViewCell {
var label:String?
var key:String?
var managedObject:NSManagedObject?
var delegate:AnyObject?
convenience override init(style: UITableViewCellStyle, reuseIdentifier: String!) {
self.init(style: style, reuseIdentifier: reuseIdentifier)
self.textLabel.font=UIFont.preferredFontForTextStyle(UIFontTextStyleCaption1)
self.backgroundColor=UIColor.blackColor()
self.textLabel.backgroundColor=UIColor.blackColor()
}
}
Run Code Online (Sandbox Code Playgroud)
但是我如何实例化类并调用相应的init函数?
Rob*_*ier 17
Swift更加静态类型化,这使得它在运行时更加安全.例如,如果任何字符串错误,你将崩溃(或者最好悄悄地做什么,这可能比崩溃更糟)并且编译器无法为你捕获它.
好的,但你如何处理这种模式?最好的方式,IMO,在Swift中是明确的并创建一个映射.
let cellMapping = [
"DBCell": DBCell.self,
]
if let cellClass = cellMapping[cellClassName] {
let cell = cellClass(style: cellStyle, reuseIdentifier: cellClassName)
// use cell
}
Run Code Online (Sandbox Code Playgroud)
现在,当你重构并更改名称DBCell,但错过了Storyboard中的字符串时,一切都将按预期继续工作,而不是像在ObjC中那样崩溃或悄然失败.
Swift对这些映射也非常聪明.cellMapping上面的类型是[String: DBCell.Type],所以如果你有一个特殊init的DBCell,它将是可用的.但如果你的字典看起来像:
let cellMapping = [
"DBCell": DBCell.self,
"OtherCell": UITableViewCell.self
]
Run Code Online (Sandbox Code Playgroud)
然后类型是[String: UITableViewCell.Type].Swift自动计算出涵盖所有值的最具体的类.如果你在这个列表中添加了一个没有实际使用的方法的类型,Swift可以在编译时捕获它.如果你犯了一个错误并添加了一些甚至不是表视图单元格:
let cellMapping = [
"DBCell": DBCell.self,
"OtherCell": UITableViewCell.self,
"BadCell": UICollectionViewCell.self
]
Run Code Online (Sandbox Code Playgroud)
然后Swift会给你一个编译时警告,你已经创建了一个AnyObject.这对编写安全但灵活的代码非常好,所有这些都是为了字典的成本.