目录
扩展
扩展的语法 (extension关键字。在其他语言里用来表示继承
计算型属性(extension只能添加计算属性get方法,不能添加存储属性set方法,或观察者willSet、didSet,存储属性里
构造器(扩展构造器必须调用指定构造器。只是改变存储属性?
方法 (扩展示例方法和类方法)
可变实例方法(mutating关键字用来表示,可修改self或属性)
下标
嵌套类型
总结:
扩展
扩展可以给一个现有的类,结构体,枚举,还有协议添加新的功能。它还拥有不需要访问被扩展类型源代码就能完成扩展的能力(即逆向建模)。扩展和 Objective-C 的分类很相似。(与 Objective-C 分类不同的是,Swift 扩展是没有名字的。)
Swift 中的扩展可以:
- 添加计算型实例属性和计算型类属性
- 定义实例方法和类方法
- 提供新的构造器
- 定义下标
- 定义和使用新的嵌套类型
- 使已经存在的类型遵循(conform)一个协议
扩展可以给一个类型添加新的功能,但是不能重写已经存在的功能。(不被调用?只调用到老的?
扩展的语法 (extension 关键字。在其他语言里用来表示继承
extension SomeType {
// 在这里给 SomeType 添加新的功能
}
扩展可以添加一个或多个协议。协议名称的写法和类或者结构体一样:
extension SomeType: SomeProtocol, AnotherProtocol {
// 协议所需要的实现写在这里
}
计算型属性(extension只能添加计算属性get方法,不能添加存储属性set方法,或观察者willSet、didSet,存储属性里
扩展可以给现有类型添加计算型实例属性和计算型类属性。(Instance的get方法和类型的get方法
给Double添加扩展。添加了5个计算属性(get 方法.都是只读
extension Double {
var km: Double { return self * 1_000.0 }
var m: Double { return self }
var cm: Double { return self / 100.0 }
var mm: Double { return self / 1_000.0 }
var ft: Double { return self / 3.28084 }
}
let oneInch = 25.4.mm //.mm 即调用了Double类型的计算属性mm
print("One inch is \(oneInch) meters")
// 打印“One inch is 0.0254 meters”
let threeFeet = 3.ft
print("Three feet is \(threeFeet) meters")
// 打印“Three feet is 0.914399970739201 meters”
let aMarathon = 42.km + 195.m
print("A marathon is \(aMarathon) meters long")
// 打印“A marathon is 42195.0 meters long”
扩展可以添加新的计算属性,但是它们不能添加存储属性,或向现有的属性添加属性观察者(只能添加get 方法,不能添加set 或观察extension 里的属性
构造器(扩展构造器必须调用指定构造器。只是改变存储属性?
扩展可以给现有的类型添加新的构造器。它使你可以把自定义类型作为参数来供其他类型的构造器使用,或者在类型的原始实现上添加额外的构造选项。
扩展可以给一个类添加新的便利构造器(convenience关键字修饰,必须调用指定构造器),但是它们不能给类添加新的指定构造器(Designated)或者析构器(deinit)。指定构造器和析构器必须始终由类的原始实现提供
扩展给一个值类型添加构造器只是用于给所有的存储属性提供默认值,并且没有定义任何自定义构造器,那么你可以在该值类型扩展的构造器中使用默认构造器和成员构造器。
使用扩展给另一个模块中定义的结构体添加构造器,那么新的构造器直到定义模块中使用一个构造器之前,不能访问?self 。
struct Size {
var width = 0.0, height = 0.0
}
struct Point {
var x = 0.0, y = 0.0
}
struct Rect {
var origin = Point()
var size = Size()
}
扩展extension Rect.
extension Rect {
init(center: Point, size: Size) {
let originX = center.x - (size.width / 2)
let originY = center.y - (size.height / 2)
扩展构造器,调用了struct 的成员构造器
self.init(origin: Point(x: originX, y: originY), size: size)
}
}
方法 (扩展示例方法和类方法)
扩展可以给现有类型添加新的实例方法和类方法。
extension Int {
func repetitions(task: () -> Void) { 接收一个 () -> Void 类型的参数
for _ in 0..<self {
task()
}
}
}
使用:
3.repetitions {
print("Hello!")
}
// Hello!
// Hello!
// Hello!
可变实例方法(mutating关键字用来表示,可修改self或属性)
通过扩展添加的实例方法同样也可以修改(或?mutating(改变))实例本身。
extension Int {
mutating func square() { 原始值求平方:
self = self * self
}
}
var someInt = 3
someInt.square()
someInt 现在是 9
下标
扩展可以给现有的类型添加新的下标
从数字右侧开始,返回小数点后的第 n 位:
extension Int {
subscript(digitIndex: Int) -> Int {
var decimalBase = 1
for _ in 0..<digitIndex {
decimalBase *= 10
}
return (self / decimalBase) % 10
}
}
746381295[0] // 返回 5
746381295[1] // 返回 9
嵌套类型
扩展可以给现有的类,结构体,还有枚举添加新的嵌套类型:
extension Int {
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
}
}
}
任意 Int 的值都可以使用这个嵌套类型:
func printIntegerKinds(_ numbers: [Int]) {
for number in numbers {
switch number.kind {
case .negative:
print("- ", terminator: "-123")
case .zero:
print("0 ", terminator: "000")
case .positive:
print("+ ", terminator: "123")
}
}
print("")
}
printIntegerKinds([3, 19, -27, 0, -6, 0, 7])
number.kind 已经被认为是 Int.Kind 类型。所以,在 switch 语句中所有的 Int.Kind case 分支可以被缩写,就像使用 .negative 替代 Int.Kind.negative
总结:
1、扩展关键字extension.用来给表示。
2、扩展可添加类型 & 实例(只能是计算属性get,不能添加存储属性和观察者
3、扩展可以添加构造器(只能是便利构造器cenvenience,不能与类型相同, 不能添加指定构造器
4、扩展添加方法(实例方法和类型方法)
5、扩展添加可变方法。mutaing关键字来修饰,可修改返回类型值和和属性值
6、扩展给现有类添加下标。(subscript)
7、添加嵌套类型
|