2

我有一个自定义类(VotingOption),它继承自NSManagedObject,有时我想检查数组中的某些投票选项是否重复。我试图使我的代码尽可能通用。这是我为扩展 CollectionType 协议所做的:

extension CollectionType where Self.Generator.Element : Equatable {

    var duplicates: [Self.Generator.Element]{
        return = self.filter { element in
            return self.filter { $0 == element }.count != 1
        }
    }

    var hasDuplicates: Bool {
        return (self.duplicates.count != 0)
    }
}

这就像一个魅力,除了它没有使用全局函数:

func ==(lhs: VotingOption, rhs: VotingOption) -> Bool {
     return (lhs.location?.title.lowercaseString == rhs.location?.title.lowercaseString) && (lhs.startDate == rhs.startDate)
}

当我做这样的事情时:

let temp: [VotingOption] = votingOptions?.array as? [VotingOption]
if temp.hasDuplicates {
     //do something
}

当我像这样在 VotingOption 中扩展 isEqual 时:

class VotingOption: NSManagedObject {

    override func isEqual(object: AnyObject?) -> Bool {

        if let rhs = object as? VotingOption {
            return (self.location?.title.lowercaseString == rhs.location?.title.lowercaseString) && (self.startDate == rhs.startDate)
        } else {
            return false
        }
    }
    ...
    ...
    ... rest of class
}

应用程序崩溃并指向 AppDelegate 并显示“libc++abi.dylib:以 NSException 类型的未捕获异常终止”错误

如何告诉 CollectionType 中的“==”使用 VotingOption 的全局函数?

4

2 回答 2

1

这是一个实现duplicatesandhasDuplicates两次的解决方案,一次用于Equatable元素,一次用于您的VotingOptions类。为了尽可能减少代码重复,我定义了一个用于查找重复的通用实现,它允许您传递一个比较两个元素的函数/闭包:

extension CollectionType {

    func findDuplicates(checkEqual: (Self.Generator.Element, Self.Generator.Element) -> Bool) -> [Self.Generator.Element]{
        return self.filter { element in
            return self.filter { checkEqual($0, element) }.count != 1
        }
    }
}

extension CollectionType where Self.Generator.Element : Equatable {

    var duplicates: [Self.Generator.Element]{
        return self.findDuplicates(==)
    }

    var hasDuplicates: Bool {
        return (self.duplicates.count != 0)
    }
}

extension CollectionType where Self.Generator.Element : VotingOption {

    var duplicates: [Self.Generator.Element]{
        return self.findDuplicates {lhs, rhs in
            return (lhs.location?.title.lowercaseString == rhs.location?.title.lowercaseString) && (lhs.startDate == rhs.startDate)
        }
    }

    var hasDuplicates: Bool {
        return (self.duplicates.count != 0)
    }
}
于 2016-02-10T07:13:34.483 回答
0

感谢所有帮助过的人。我将在这里发布最终实现:基本上创建了一个新协议“Identity”,其方法“equals”基本上是“==”,最后为实现“Identity”的 NSManagedObjects 添加了一个通用扩展,该扩展基本上在“等于”方法。但是在“VotingOption”中,我已经根据我的特定需求覆盖了这个函数。最后,在“元素”为“身份”的地方扩展了集合类型。而不是 "==" 它调用 "equals" 方法:)

protocol Identity {
     func equals(rhs: Self) -> Bool
}

extension Identity where Self: NSManagedObject {
     func equals(rhs: Self) -> Bool {
          return self == rhs
     }
}

extension CollectionType where Self.Generator.Element : Identity {

    var duplicates: [Self.Generator.Element] {

        return self.filter { element in
            return self.filter { $0.equals(element) }.count != 1
        }
    }

    var hasDuplicates: Bool {
        return (self.duplicates.count > 0)
    }
}

class VotingOption: NSManagedObject, Identity {

    func equals(rhs: VotingOption) -> Bool {

    //... implementation here
    }
}
于 2016-02-24T19:23:47.583 回答