为了账号安全,请及时绑定邮箱和手机立即绑定

如何在NSManagedObject Swift扩展中创建托管对象子类的实例?

如何在NSManagedObject Swift扩展中创建托管对象子类的实例?

HUWWW 2019-08-19 15:46:38
如何在NSManagedObject Swift扩展中创建托管对象子类的实例?当创建一个扩展助手来NSManagedObject创建一个新的托管对象子类时,swift提供了Self模仿的类型,instancetype这很好,但我似乎无法进行类型转换AnyObject。下面的代码没有编译错误'AnyObject'不能转换为'Self'救命?extension NSManagedObject{     class func createInContext(context:NSManagedObjectContext) -> Self {         var classname = className()         var object: AnyObject = NSEntityDescription.insertNewObjectForEntityForName(classname, inManagedObjectContext: context)         return object    }     class func className() -> String {         let classString = NSStringFromClass(self)         //Remove Swift module name        let range = classString.rangeOfString(".", options: NSStringCompareOptions.CaseInsensitiveSearch, range: Range<String.Index>(start:classString.startIndex, end: classString.endIndex), locale: nil)         return classString.substringFromIndex(range!.endIndex)     }}
查看完整描述

3 回答

?
慕沐林林

TA贡献2016条经验 获得超9个赞

(现在更新为Swift 3/4。可以在编辑历史中找到早期Swift版本的解决方案。)

您可以使用unsafeDowncast到的返回值转换NSEntityDescription.insertNewObject()Self (这是该方法实际上是所谓的类型):

extension NSManagedObject {
    class func create(in context: NSManagedObjectContext) -> Self {
        let classname = entityName()
        let object = NSEntityDescription.insertNewObject(forEntityName: classname, into: context)
        return unsafeDowncast(object, to: self)
    }

    // Returns the unqualified class name, i.e. the last component.    // Can be overridden in a subclass.    class func entityName() -> String {
        return String(describing: self)
    }}

然后

let obj = YourEntity.createInContext(context)

工作和编译器推断obj正确的类型YourEntity


查看完整回答
反对 回复 2019-08-19
?
手掌心

TA贡献1942条经验 获得超3个赞

这是通过实现初始化方法(使用Xcode 7.1测试)来解决问题的不同方法:

extension NSManagedObject {

    // Returns the unqualified class name, i.e. the last component.    // Can be overridden in a subclass.    class func entityName() -> String {
        return String(self)
    }

    convenience init(context: NSManagedObjectContext) {
        let eName = self.dynamicType.entityName()
        let entity = NSEntityDescription.entityForName(eName, inManagedObjectContext: context)!
        self.init(entity: entity, insertIntoManagedObjectContext: context)
    }}

Init方法具有隐式返回类型,Self并且不需要强制转换技巧。

let obj = YourEntity(context: context)

创建该YourEntity类型的对象。


Swift 3/4更新:

extension NSManagedObject {

    // Returns the unqualified class name, i.e. the last component.    // Can be overridden in a subclass.    class func entityName() -> String {
        return String(describing: self)
    }

    convenience init(context: NSManagedObjectContext) {
        let eName = type(of: self).entityName()
        let entity = NSEntityDescription.entity(forEntityName: eName, in: context)!
        self.init(entity: entity, insertInto: context)
    }}


查看完整回答
反对 回复 2019-08-19
?
慕的地10843

TA贡献1785条经验 获得超8个赞

在Swift 2中,有一个使用协议和协议扩展的非常智能的解决方案

protocol Fetchable{
  typealias FetchableType: NSManagedObject

  static var entityName : String { get }
  static func createInContext(context: NSManagedObjectContext) ->  FetchableType}extension Fetchable where Self : NSManagedObject, FetchableType == Self{
  static func createInContext(context: NSManagedObjectContext) -> FetchableType
  {
    return NSEntityDescription.insertNewObjectForEntityForName(entityName, inManagedObjectContext: context) as! FetchableType
  }}

在每个NSManagedObject子类中添加协议Fetchable并实现该属性entityName

现在该函数MyEntity.createInContext(…)将返回正确的类型而无需进一步的类型转换。


查看完整回答
反对 回复 2019-08-19
  • 3 回答
  • 0 关注
  • 531 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信