如何在Swift中从字符串实例化class和init?

我可以在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];
    }

现在我需要将代码迁移到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()
    }
    }

但是如何实例化类并调用相应的init函数呢?

Swift更加静态类型化,这使得它在运行时更加安全.例如,如果任何字符串都是错误的,你将崩溃(或者最好悄悄地做什么,这可能比崩溃更糟)并且编译器无法为你捕获它.

好的,但你如何处理这种模式?在Swift中,IMO的最佳方式是显式并创建映射.

let cellMapping = [
  "Dbcell": Dbcell.self,]

if let cellClass = cellMapping[cellClassName] {
  let cell = cellClass(style: cellStyle,reuseIdentifier: cellClassName)
  // use cell
}

现在,当您重构并更改Dbcell的名称,但错过了Storyboard中的字符串时,一切都将按预期继续工作,而不是像在ObjC中那样崩溃或悄然失败.

Swift对这些映射也非常聪明.上面的cellMapping类型是[String:Dbcell.Type],所以如果你有一个特殊的初始化Dbcell,它就可用了.但如果你的字典看起来像:

let cellMapping = [
  "Dbcell": Dbcell.self,"OtherCell": UITableViewCell.self
]

那么类型是[String:UITableViewCell.Type]. Swift自动计算出涵盖所有值的最具体的类.如果你在这个列表中添加一个没有实际使用的方法的类型,Swift可以在编译时捕获它.如果你犯了一个错误添加了一些甚至不是表格视图的单元格:

let cellMapping = [
  "Dbcell": Dbcell.self,"OtherCell": UITableViewCell.self,"BadCell": UICollectionViewCell.self
]

然后Swift会给你一个编译时警告,你已经创建了一个AnyObject.这对编写安全但灵活的代码非常好,所有这些都是为了字典的成本.

相关文章

软件简介:蓝湖辅助工具,减少移动端开发中控件属性的复制和粘...
现实生活中,我们听到的声音都是时间连续的,我们称为这种信...
前言最近在B站上看到一个漂亮的仙女姐姐跳舞视频,循环看了亿...
【Android App】实战项目之仿抖音的短视频分享App(附源码和...
前言这一篇博客应该是我花时间最多的一次了,从2022年1月底至...
因为我既对接过session、cookie,也对接过JWT,今年因为工作...