- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我尝试使用通用协议(protocol)来实现面向对象的代码。假设我有两个协议(protocol)
protocol Executable: class {
func execute()
}
protocol Dockable: class {
associatedtype T
func dock(object: T)
}
我已经为可执行文件实现了一个装饰器:
final class DockableExecutable: Executable, Dockable {
typealias T = Executable
private let decorated: Executable
private var docked: Executable?
init(_ decorated: Executable) {
self.decorated = decorated
}
// from Executable
func execute() {
decorated.execute()
docked?.execute()
}
// from Dockable
func dock(object: Executable) {
docked = object
}
}
现在我不想在这样的类中使用它:
final class MyViewController: UIViewController {
init(save: Executable, uiUpdateConnector: Dockable<Executable>) {}
}
但这不可能,因为协议(protocol)本身不是通用的,只有函数。编译器告诉我:
Cannot specialize non-generic type 'Dockable'
想法是这样使用它:
let dockableExecutable = DockableExecutable(
SQLUpdateExecutable(/** dependencies **/)
)
let controller = MyViewController(save: dockableExecutable, uiUpdateConnector: dockableExecutable)
Swift 3 中的正确语法如何让编译器满意?
更新 1
我通过以下代码取得了一些进展:
final class MyViewController: UIViewController {
init<DOCKABLE: Dockable>(save: Executable, uiUpdateConnector: DOCKABLE) where DOCKABLE.T: Executable {}
}
看起来很奇怪,也许有人有更好的主意?现在使用类(class)时我得到:
Generic parameter 'DOCKABLE' could not be inferred
所以我的问题没有改变:
Swift 3 中的正确语法如何让编译器满意?
更新 2
似乎不可能在基于协议(protocol)的面向对象编程中使用快速通用(或更好:类型关联)协议(protocol)。
因此我们必须将它们包装到某种容器中并放松基于协议(protocol)的方法,或者我们必须为每种情况定义不同的协议(protocol)。
因为在没有协议(protocol)的情况下工作对我来说是不可能的,我必须在没有泛型的情况下编写不同的协议(protocol)。对 swift 感到羞耻😒
最佳答案
我不确定您的问题是否可以在不进行类型删除的情况下得到解决,因为您不能将协议(protocol)与关联类型用于变量类型或函数参数。
尝试使用类型删除的解决方案:
final class AnyDockable<U>: Dockable {
typealias T = U
let _dock: (U) -> Void
let _execute: () -> Void
init<Base: Dockable & Executable>(base: Base) where Base.T == U {
_dock = base.dock
_execute = base.execute
}
func dock(object: U) {
_dock(object)
}
func execute() {
_execute()
}
}
你的编译器会很高兴,我已经检查过了:
class Exe: Executable {
func execute() {
print("")
}
}
let executable: Executable = Exe()
let de = AnyDockable(base: DockableExecutable(executable))
final class MyViewController: UIViewController {
init(save: Executable, uiUpdateConnector: AnyDockable<Executable>) {
super.init(nibName: "", bundle: nil)
}
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
}
MyViewController(save: executable, uiUpdateConnector: de)
关于Swift 泛型 : "Cannot specialize non-generic type",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42887010/
我是一名优秀的程序员,十分优秀!