Swift 扩展
现有的类,结构或枚举类型的功能可以在扩展的帮助下加入。类型的功能可以使用扩展加入,但重写的功能不能使用扩展。
Swift 扩展功能:
-
添加计算属性并计算类型属性
-
定义实例和类型的方法
-
提供了新的初始化
-
定义下标
-
定义和使用新的嵌套类型
-
使现有类型符合协议
扩展用关键字 extension 声明
语法
extension SomeType { // new functionality can be added here }
现有类型也可以用扩展加入使它作为一个协议标准和其语法类似于类或结构。
extension SomeType: SomeProtocol, AnotherProtocol { // protocol requirements is described here }
计算属性
计算“实例”和“type”属性也可以扩展在扩展的帮助下实现。
extension Int { var add: Int {return self + 100 } var sub: Int { return self - 10 } var mul: Int { return self * 10 } var div: Int { return self / 5 } } let addition = 3.add println("Addition is \(addition)") let subtraction = 120.sub println("Subtraction is \(subtraction)") let multiplication = 39.mul println("Multiplication is \(multiplication)") let division = 55.div println("Division is \(division)") let mix = 30.add + 34.sub println("Mixed Type is \(mix)")
当我们使用 playground 运行上面的程序,得到以下结果。
Addition is 103 Subtraction is 110 Multiplication is 390 Division is 11 Mixed Type is 154
初始化器
Swift 能够灵活地通过扩展新的初始化添加到现有的类型。用户可以添加自己的自定义类型来扩展已定义的类型,额外的初始化选项也是可以的。 扩展仅支持 init(). 而 deinit() 不被扩展支持。
struct sum { var num1 = 100, num2 = 200 } struct diff { var no1 = 200, no2 = 100 } struct mult { var a = sum() var b = diff() } let calc = mult() println ("Inside mult block \(calc.a.num1, calc.a.num2)") println("Inside mult block \(calc.b.no1, calc.b.no2)") let memcalc = mult(a: sum(num1: 300, num2: 500),b: diff(no1: 300, no2: 100)) println("Inside mult block \(memcalc.a.num1, memcalc.a.num2)") println("Inside mult block \(memcalc.b.no1, memcalc.b.no2)") extension mult { init(x: sum, y: diff) { let X = x.num1 + x.num2 let Y = y.no1 + y.no2 } } let a = sum(num1: 100, num2: 200) println("Inside Sum Block:\( a.num1, a.num2)") let b = diff(no1: 200, no2: 100) println("Inside Diff Block: \(b.no1, b.no2)")
当我们使用 playground 运行上面的程序,得到以下结果。
Inside mult block (100, 200) Inside mult block (200, 100) Inside mult block (300, 500) Inside mult block (300, 100) Inside Sum Block:(100, 200) Inside Diff Block: (200, 100)
方法
新实例方法和类型的方法,可以在扩展的帮助下进一步加入到子类。
extension Int { func topics(summation: () -> ()) { for _ in 0..<self { summation() } } } 4.topics({ println("Inside Extensions Block") }) 3.topics({ println("Inside Type Casting Block") })
当我们使用 playground 运行上面的程序,得到以下结果。
Inside Extensions Block Inside Extensions Block Inside Extensions Block Inside Extensions Block Inside Type Casting Block Inside Type Casting Block Inside Type Casting Block
topics() 函数使用参数是 (summation: () -> ()) 的类型表示该函数不带任何参数 而且它不会返回任何值。调用该函数多次, 块被初始化,并调用 topic()方法初始化。
不同实例方法变形
实例方法时也可以作为扩展声明的变形。
修改自身的结构和计数的方法或它的属性必须标注实例方法变形,就像是从一个原始的实现变形的方法。
extension Double { mutating func square() { let pi = 3.1415 self = pi * self * self } } var Trial1 = 3.3 Trial1.square() println("Area of circle is: \(Trial1)") var Trial2 = 5.8 Trial2.square() println("Area of circle is: \(Trial2)") var Trial3 = 120.3 Trial3.square() println("Area of circle is: \(Trial3)")
当我们使用 playground 运行上面的程序,得到以下结果。
Area of circle is: 34.210935 Area of circle is: 105.68006 Area of circle is: 45464.070735
下标
添加新标已声明实例也可以扩展。
extension Int { subscript(var multtable: Int) -> Int { var no1 = 1 while multtable > 0 { no1 *= 10 --multtable } return (self / no1) % 10 } } println(12[0]) println(7869[1]) println(786543[2])
当我们使用 playground 运行上面的程序,得到以下结果。
2 6 5
嵌套类型
嵌套类型为类,结构和枚举实例,也可在扩展的帮助下进行扩展。
extension Int { enum calc { case add case sub case mult case div case anything } var print: calc { switch self { case 0: return .add case 1: return .sub case 2: return .mult case 3: return .div default: return .anything } } } func result(numb: [Int]) { for i in numb { switch i.print { case .add: println(" 10 ") case .sub: println(" 20 ") case .mult: println(" 30 ") case .div: println(" 40 ") default: println(" 50 ") } } } result([0, 1, 2, 3, 4, 7])
当我们使用 playground 运行上面的程序,得到以下结果。
10 20 30 40 50 50