ios - 将类扩展继承子句迁移到协议(protocol)扩展

标签 ios swift generics swift-protocols protocol-extension

我有一个 VehicleModels 框架,其中包含 CarBikePlane 等类。 在另一个框架 VehicleInventory 中,我需要在表格中打印自定义描述(特定于第二个框架)。所以我添加了协议(protocol) DescriptableVehicle 和方法 describe()。 然后我为所有车辆添加了协议(protocol)扩展,例如:

extension Car: DescriptableVehicle {
  func describe() -> String {
    return "Car: \(self.vin)" // returns formatted vehicle number
  }
}

但是,假设已经改变,现在我没有公开我的车辆框架中的具体类。相反,我公开了像 CarProtocolBikeProtocol 这样的协议(protocol),因此通常我有相同的信息。

问题是我不能再使用协议(protocol)扩展了(或者至少不是那种形式)因为 协议(protocol)扩展类扩展 不能有继承条款。

知道如何解决这个问题,而不是过多地修改我的用法吗? 最初,我认为协议(protocol)上的几个 where 子句加上几个强制转换会达成协议(protocol),但是如果无法访问类,它也无济于事。 我也尝试过 Adapters 和类型删除,但要么我使用不当,要么它有不同的用途。

为了说明问题,我准备了存储库:https://github.com/wedkarz/SwiftProtocolExtensionsProblem

有两个 Playground 。 V1 是我曾经拥有的并且可以正常工作的。 V2 包含我现在拥有的以及我正在尝试使其发挥作用的内容。

在现实生活中,PrivateFramework 类是一个独立的框架和协议(protocol):VehicleProtocolCarProtocolBikeProtocolPlaneProtocol是其中的一部分,但是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/

相关文章:

swift - 如何知道 AVSpeechSynthesizer 何时说完所有排队的话语。 swift 4.2

ios - tableView + CoreData 上的空单元格

java - 设计帮助!枚举工厂变压器中的java泛型!

Java:不能在 switch 语句中使用通用枚举

ios - Swift NSURLSession 和身份验证

ios - 在 SWIFT 中为 JSON 格式制作字符串(包括 TextView 中的换行符)的最佳方法是什么?

ios - 如何从TextField中获取十进制值并在Objective C中进行计算

ios - 难倒重新 : Chrome iOS Behavior

ios - 单击时 UILabel 点击手势不起作用

java - 在非泛型类中实现泛型接口(interface)