我有一个 VehicleModels
框架,其中包含 Car
、Bike
、Plane
等类。
在另一个框架 VehicleInventory
中,我需要在表格中打印自定义描述(特定于第二个框架)。所以我添加了协议(protocol) DescriptableVehicle
和方法 describe()
。
然后我为所有车辆添加了协议(protocol)扩展,例如:
extension Car: DescriptableVehicle {
func describe() -> String {
return "Car: \(self.vin)" // returns formatted vehicle number
}
}
但是,假设已经改变,现在我没有公开我的车辆框架中的具体类。相反,我公开了像 CarProtocol
、BikeProtocol
这样的协议(protocol),因此通常我有相同的信息。
问题是我不能再使用协议(protocol)扩展了(或者至少不是那种形式)因为 协议(protocol)扩展 与类扩展 不能有继承条款。
知道如何解决这个问题,而不是过多地修改我的用法吗?
最初,我认为协议(protocol)上的几个 where
子句加上几个强制转换会达成协议(protocol),但是如果无法访问类,它也无济于事。
我也尝试过 Adapters 和类型删除,但要么我使用不当,要么它有不同的用途。
为了说明问题,我准备了存储库:https://github.com/wedkarz/SwiftProtocolExtensionsProblem
有两个 Playground 。 V1 是我曾经拥有的并且可以正常工作的。 V2 包含我现在拥有的以及我正在尝试使其发挥作用的内容。
在现实生活中,PrivateFramework
类是一个独立的框架和协议(protocol):VehicleProtocol
、CarProtocol
、BikeProtocol
、 PlaneProtocol
是其中的一部分,但是DescriptableVehicle
不是PrivateFramework
的一部分,所以不能在里面使用。该示例说明了访问具体类型的问题及其扩展的问题。
你可以看到有一些扩展被注释掉了,因为我不能再使用它们了。目标是使 [VehicleProtocol]
的集合以与之前类似的方式打印其内容。
最佳答案
这是否回答了您的问题?
class PrivateFramework {
private class AbstractVehicle: VehicleProtocol {
var name: String { return "Abstract vehicle name" }
}
private class Car: AbstractVehicle, CarProtocol {
override var name: String { return "Car vehicle name" }
let vin: String = "ABCDEFGH VIN"
}
private class Bike: AbstractVehicle, BikeProtocol {
override var name: String { return "Bike vehicle name" }
let saddle: String = "Comforable saddle name"
}
private class Plane: AbstractVehicle, PlaneProtocol {
override var name: String { return "Plane vehicle name" }
let numberOfEngines: Int = 4
}
func produceVehicles() -> [DescriptableVehicle] {
let car = Car()
let bike = Bike()
let plane = Plane()
return [car, bike, plane]
}
}
protocol VehicleProtocol {
var name: String { get }
}
protocol DescriptableVehicle: VehicleProtocol {
func describe() -> String
}
protocol CarProtocol: DescriptableVehicle {
var vin: String { get }
}
protocol BikeProtocol: DescriptableVehicle {
var saddle: String { get }
}
protocol PlaneProtocol: DescriptableVehicle {
var numberOfEngines: Int { get }
}
extension DescriptableVehicle where Self: CarProtocol {
func describe() -> String { return "Car, \(self.name), \(self.vin)" }
}
extension DescriptableVehicle where Self: BikeProtocol {
func describe() -> String { return "Bike, \(self.name), \(self.saddle)" }
}
extension DescriptableVehicle where Self: PlaneProtocol {
func describe() -> String { return "Plane, \(self.name), \(self.numberOfEngines)" }
}
let vehicles: [DescriptableVehicle] = PrivateFramework().produceVehicles()
vehicles.forEach { print($0.describe()) }
关于ios - 将类扩展继承子句迁移到协议(protocol)扩展,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47436106/