Swift --- 扩展(Extention)

  • Swift这种的扩展,有点类似于OC中的分类(Category)
  • 扩展可以为枚举、结构体、类、协议添加新功能

可以添加方法、计算属性、下标、(便捷)初始化器、嵌套类型、协议等等

  • 扩展不能办到的事情

1.不能覆盖原有的功能
2.不能添加存储属性,不能向已有的属性添加属性观察器
3.不能添加父类
4.不能添加指定初始化器,不能添加反初始化器

扩展计算属性

extension Double {
    var km: Double { self * 1_000.0 }
    var m: Double { self }
    var dm: Double { self / 10.0 }
    var cm: Double { self / 100.0 }
    var mm: Double { self / 1_000.0 }
}
var d = 100.0
print(d.km)
print(d.m)
print(d.dm)
print(100.0.dm)

扩展下标

  • 不能覆盖原有功能
  • 可以使用原来的泛型
subscript(nullable idx: Int) -> Element?
var arr: Array<Int> = [10, 20, 30]
extension Array {
		subscript(nullable idx: Int) -> Element? {
				if (startIndex..<endIndex).contains(idx) {
					return self[Idx]
				}	
				return nil
		}
}
print(arr[nullable: 10] as Any)
print(arr[nullable: 1] ?? 0)

扩展方法,嵌套类型,下标

extension Int {
    func repeats(task: () -> Void) {
        for _ in 0..<self { task() }
    }
    mutating func square() -> Int {
        self = self * self
        return self
    }
    enum Kind { case negative, zero, positive }
    var kind: Kind {
        switch self {
        case 0: return .zero
        case let x where x > 0: return .positive
        default: return .negative
        }
    }
    subscript(digitIndex: Int) -> Int {
        var decimalBase = 1
        for _ in 0..<digitIndex { decimalBase *= 10 }
        return (self / decimalBase) % 10
    }
}

扩展协议、初始化器(便捷)

  • 只有类才区分指定初始化器还是便捷初始化器,结构体中不区分
  • required初始化器,不能写在扩展中
  • 如果希望自定义初始化器的同时,编译器也能够生成默认初始化器,可以在扩展中便捷自定义初始化器
class Person {
    var age: Int
    var name: String
    init(age: Int, name: String) {
        self.age = age
        self.name = name
    }
}
extension Person : Equatable {
    static func == (left: Person, right: Person) -> Bool {
       return left.age == right.age && left.name == right.name
    }
    convenience init() {
        self.init(age: 0, name: "")
    }
}
struct Point {
    var x: Int = 0
    var y: Int = 0
}
extension Point {
    init(_ point: Point) {
        self.init(x: point.x, y: point.y)
    }
}
var p1 = Point()
var p2 = Point(x: 10)
var p3 = Point(y: 20)
var p4 = Point(x: 10, y:20)
var p5 = Point(p4)
  • 如果一个类型已经实现了协议的所有要求,但是还没有声明它遵守了这个协议,可以通过扩展来让它遵守这个协议
protocol TestProtocol {
    func test()
}
class TestClass {
    func text() {
        print("test")
    }
}
extension TestClass : TestProtocol {}
extension BinaryInteger {
    func isOdd() -> Bool { self % 2 != 0 }
}
var age: Int8 = 10

print(age.isOdd())
print(10.isOdd())
print((-3).isOdd())
  • 扩展可以给协议提供默认实现,也间接实现可选协议的效果
  • 扩展可以给协议扩充协议中从未声明过的方法
protocol TestProtocol {
    func test1()
}
extension TestProtocol {
    func test1() {
        print("TestProtocol test1")
    }
    func test2() {
        print("TestProtocol test2")
    }
}
class TestClass: TestProtocol {}
var cls = TestClass()
cls.test1() // TestProtocol test1
cls.test2() // TestProtocol test2
  • 下面这种情况:
  • 对象显示遵守某协议,协议中没有声明的方法会优先调用扩展实现,协议中声明的方法会优先调用类中实现方法
protocol TestProtocol {
    func test1()
}
extension TestProtocol {
    func test1() {
        print("TestProtocol test1")
    }
    func test2() {
        print("TestProtocol test2")
    }
}
class TestClass: TestProtocol {
    func test1() {
        print("TestClass test1")
    }
    func test2() {
        print("TestClass test2")
    }
}
var cls: TestProtocol = TestClass()
cls.test1() // TestClass test1
cls.test2() // TestProtocol test2

泛型

class Stack<E> {
    var elements = [E]()
    func push(_ element: E) {
        elements.append(element)
    }
    func pop() -> E { elements.removeLast() }
    func size() -> Int { elements.count }
}
// 扩展中依然可以使用原类型中的泛型类型
extension Stack {
    func top() -> E { elements.last! }
}
// 符合条件才扩展
extension Stack : Equatable where E : Equatable {
    static func == (left: Stack, right: Stack) -> Bool {
        left.elements == right.elements
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值