zoukankan      html  css  js  c++  java
  • typealias和泛型接口

    typealias 是用来为已经存在的类型重新定义名字的,通过命名,可以使代码变得更加清晰。使用的语法也很简单,使用 typealias 关键字像使用普通的赋值语句一样,可以将某个已经存在的类型赋值为新的名字。比如在计算二维平面上的距离和位置的时候,我们一般使用 Double 来表示距离,用 CGPoint 来表示位置:

    func distanceBetweenPoint(point: CGPoint, toPoint: CGPoint) -> Double {
        let dx = Double(toPoint.x - point.x)
        let dy = Double(toPoint.y - point.y)
        return sqrt(dx * dx + dy * dy)
    }
    
    let origin: CGPoint = CGPoint(x: 0, y: 0)
    let point: CGPoint = CGPoint(x: 1, y: 1)
    
    let distance: Double =  distanceBetweenPoint(origin, point)
    

    虽然在数学上和最后的程序运行上都没什么问题,但是阅读和维护的时候总是觉得有哪里不对。因为我们没有将数学抽象和实际问题结合起来,使得在阅读代码时我们还需要在大脑中进行一次额外的转换:CGPoint 代表一个点,而这个点就是我们在定义的坐标系里的位置Double 是一个数字,它代表两个点之间的距离

    如果我们使用 typealias,就可以将这种转换直接写在代码里,从而减轻阅读和维护的负担:

    import UIKit
    
    typealias Location = CGPoint
    typealias Distance = Double
    
    func distanceBetweenPoint(location: Location,
        toLocation: Location) -> Distance {
            let dx = Distance(location.x - toLocation.x)
            let dy = Distance(location.y - toLocation.y)
            return sqrt(dx * dx + dy * dy)
    }
    
    let origin: Location = Location(x: 0, y: 0)
    let point: Location = Location(x: 1, y: 1)
    
    let distance: Distance =  distanceBetweenPoint(origin, toLocation: point)
    

    同样的代码,在 typealias 的帮助下,读起来就轻松多了。可能单单这个简单例子不会有特别多的体会,但是当你遇到更复杂的实际问题时,你就可以不再关心并去思考自己代码里那些成堆的 Int 或者 String 之类的基本类型到底代表的是什么东西了,这样你应该能省下不少脑细胞。

    对于普通类型并没有什么难点,但是在涉及到泛型时,情况就稍微不太一样。首先,typealias 是单一的,也就是说你必须指定将某个特定的类型通过 typealias 赋值为新名字,而不能将整个泛型类型进行重命名。下面这样的命名都是无法通过编译的:

    这是错误代码

    class Person<T> {}
    typealias Worker = Person
    typealias Worker = Person<T>
    typealias Worker<T> = Person<T>
    

    一旦泛型类型的确定性得到保证后,我们就可以重命名了:

    class Person<T> {}
    
    typealias WorkId = String
    typealias Worker = Person<WorkId>
    

    另一个值得一提的是 Swift 中是没有泛型接口的,但是使用 typealias,我们可以在接口里定义一个必须实现的别名,这在一定范围内也算一种折衷方案。比如在 GeneratorType 和 SequenceType 这两个接口中,Swift 都用到了这个技巧,来为接口确定一个使用的类似泛型的特性:

    protocol GeneratorType {
        typealias Element
        mutating func next() -> Self.Element?
    }
    
    protocol SequenceType {
        typealias Generator : GeneratorType
        func generate() -> Self.Generator
    }
    

    在实现这些接口时,我们不仅需要实现指定的方法,还要实现对应的 typealias,这其实是一种对于接口适用范围的抽象和约束。

  • 相关阅读:
    178
    177
    176
    175
    To Do List
    洛谷 P4198 楼房重建
    斯特林数
    容斥原理
    组合数学笔记
    激光相机数据融合(6)--激光相机标定
  • 原文地址:https://www.cnblogs.com/weiboyuan/p/6102968.html
Copyright © 2011-2022 走看看