swift associatedtype和typealias

typealias

1、可以用来对已有的类型进行重命名,比如在表示两点之间的距离的时候,可以使用typealias将x和y轴的距离Double表示为Distance。

    func distance(p1: CGPoint, p2: CGPoint) -> Double {
        let x = Double(p1.x - p2.x)
        let y = Double(p1.y - p2.y)
        return sqrt(x * x + y * y)
    }
    
    typealias Distance = Double
    func distance2(p1: CGPoint, p2: CGPoint) -> Double {
        let x = Distance(p1.x - p2.x)
        let y = Distance(p1.y - p2.y)
        return sqrt(x * x + y * y)
    }

通过上面的表述,可以使语义更加清晰,特别是在做一些数据计算的时候,不会出现很多Int,Double的数字处理,可以使用更加语义明确的表达需要计算的数据。
2、可以对闭包进行重新命名,这样在做参数传递的时候更加清晰

    typealias Success = (_ result: String) -> Void
    typealias Failure = (_ error: String) -> Void

    public func excuteNetworking(_ successBlock: Success, failBlock: Failure) {
        
    }

3 协议使用associatedtype的时候,可以用来对关联类型重定义,协助协议实现泛型功能,下面会说到

1、关联类型作为协议实现泛型的一种方式,可以在协议中预先定义一个占位符,实现协议的时候再确定这个占位符具体的类型。
下面是一个通用的协议,以及实现协议的类

protocol Food {
    func desc() -> String
}

class Apple: Food {
    func desc() -> String {
        return "apple"
    }
}

protocol Animal {
    func eat(food: Food)
}

class People: Animal {
    func eat(food: Food) {
        print("eat \(food.desc())")
    }
}

let p = People()
p.eat(food: Apple())

上面的例子中,人继承自动物,实现eat方法,方法中的Food是代表食物,也是一个协议,现在是假设人可以吃所有的食物。
再看下面的例子,如果把动物吃的食物换成associatedType类型(下面的代码编译不过)

protocol Animal2 {
    associatedtype F: Food
    func eat(food: F)
}

class People2: Animal2 {
    typealias F = Food
    func eat(food: F) {
        print("eat \(food.desc())")
    }
}

编译器会报错,大致意思是人没有实现动物中的关联类型,关联类型需要是一个确定的类型,而不是一个协议,我们将上面的People2改成如下形式,让F是一个具体的类型,比如Apple,就可以了

protocol Animal2 {
    associatedtype F: Food
    func eat(food: F)
}

class People2: Animal2 {
    typealias F = Apple
    func eat(food: F) {
        print("eat \(food.desc())")
    }
}

let p = People2()
p.eat(food: Apple())

对于上面的写法,虽然用这个例子不太好解释,上面是限定了人吃东西只能吃苹果,这样显然是不合适的,但是如果是熊猫或者是病人,我们就可以限定他吃的东西只能是某种具体的食物。

在实际应用中,如果协议使用了关联类型,那么这个协议就失去了动态派发的特性,比如上面的例子人就只能吃苹果了。如果你给他吃一个梨,那他就消化不了

class Orange: Food {
    func desc() -> String {
        return "orange"
    }
}
let p = People2()
p.eat(food: Orange())

当然,我们也可以将People定义为泛型,这样在确认人的类型的时候需要确定他只能吃什么

class People2<F: Food>: Animal2 {
    typealias MF = F
    
    func eat(food: MF) {
        print("eat \(food.desc())")
    }
}

let p: People2<Apple> = People2()
p.eat(food: Apple())

swift associatedtype和typealias

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值