专栏首页学海无涯20.Swift学习之扩展

20.Swift学习之扩展

扩展

  • 为现有的类、结构体、枚举类型、或协议添加了新功能。扩展和 Objective-C 中的分类类似。
  • 扩展可以:
    • 添加计算实例属性和计算类型属性;
    • 定义实例方法和类型方法;
    • 提供新初始化器;
    • 使现有的类型遵循某协议

语法

extension SomeType {
    // new functionality to add to SomeType goes here
}

扩展可以使已有的类型遵循一个或多个协议。在这种情况下,协议名的书写方式与类或结构体完全一样:

extension SomeType: SomeProtocol, AnotherProtocol {
    // implementation of protocol requirements goes here
}

扩展计算属性

  • 扩展可以向已有的类型添加计算实例属性和计算类型属性。
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
print("One inch is \(oneInch) meters")
let threeFeet = 3.ft
print("Three feet is \(threeFeet) meters")

扩展构造函数

  • 扩展可向已有的类型添加新的初始化器
extension Rect {
    init(center: Point, size: Size) {
        let originX = center.x - (size.width / 2)
        let originY = center.y - (size.height / 2)
        self.init(origin: Point(x: originX, y: originY), size: size)
    }
}

let centerRect = Rect(center: Point(x: 4.0, y: 4.0),
                      size: Size(width: 3.0, height: 3.0))

扩展方法

  • 扩展可以为已有的类型添加新的实例方法和类型方法。
extension Int {
    func repetitions(task: () -> Void) {
        for _ in 0..<self {
            task()
        }
    }
}


3.repetitions {
    print("Hello!")
}

扩展异变实例方法

  • 增加了扩展的实例方法仍可以修改(或异变)实例本身
extension Int {
    mutating func square() {
        self = self * self
    }
}
var someInt = 3
someInt.square()

扩展协议

  • 协议可以通过扩展来提供方法和属性的实现以遵循类型。
//RandomNumberGenerator 是一个协议
extension RandomNumberGenerator {
    func randomBool() -> Bool {
        return random() > 0.5
    }
}

面向协议编程

针对某个需要实现的功能,可以使用协议定义出接口,然后利用协议扩展提供默认的实现。需要这个功能,只需要声明遵守了这个协议即可,遵守某个协议的对象调用协议声明的方法时,如果类本身没有提供实现,协议扩展提供的默认实现会被调用。

  • 案例一
protocol Eat {
    func eat()
}

class Person: Eat {
    func eat() {
        print("吃饭了")
    }
}

var p = Person()
p.eat()
  • 改进
extension Eatable {
    func eat() {
        print("吃饭了")
    }
}
class Person: Eatable {
}
var p = Person()
p.eat()
  • 再次修改
class Person: Eatable {
    
    func eat() {
        print("人要吃饭了")
    }
}

var p = Person()
p.eat()
  • 案例二
protocol Coder {
    var haveFun:Bool {get set}
    var ownMoney:Bool {get set}
}


protocol Swifter {
    var codingLevel:Int {get set}
}


struct CoderA : Coder {
    
    var name:String
    var haveFun: Bool
    var ownMoney: Bool
}


struct CoderB : Coder, Swifter {
    
    var name:String
    var haveFun: Bool = true
    var ownMoney: Bool = true
    var codingLevel: Int = 3
}

struct CoderC : Coder, Swifter{
    
    var name:String
    var haveFun: Bool = true
    var ownMoney: Bool = true
    var codingLevel: Int = 5
}

可以发现CoderB与CoderC有冗余

  • 改进
protocol Coder {
    var haveFun:Bool {get set}
    var ownMoney:Bool {get set}
}


protocol Swifter {
    var codingLevel:Int {get set}
}

//where限定条件
extension Coder where Self:Swifter {
    
    var haveFun:Bool { return true}
    var ownMoney:Bool { return true}
}


struct CoderA : Coder {
    
    var name:String
    var haveFun: Bool
    var ownMoney: Bool
}


struct CoderB : Coder, Swifter {
    
    var name:String
    var haveFun: Bool
    var ownMoney: Bool
    var codingLevel: Int = 3
}

struct CoderC : Coder, Swifter{
    
    var name:String
    var haveFun: Bool
    var ownMoney: Bool
    var codingLevel: Int = 5
}

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • Swift学习之5.1和5.2新特性

    Swift 5.1 内置于 Xcode 11,新增了很多新特性,比较重要的有以下几个。

    YungFan
  • SwiftUI-数据流

    SwiftUI中的界面是严格数据驱动的:运行时界面的修改,只能通过修改数据来间接完成,而不是直接对界面进行修改操作。

    YungFan
  • Swift 5.1 新特性

    Swift 5.1 内置于 Xcode 11,新增了很多新特性,比较重要的有以下几个。

    YungFan
  • html页面显示服务器时间

    lblTimer = $("#lbltimer"); d = new Date('<%=DateTime.Now.ToString("yyyy...

    冰封一夏
  • 使用await和async关键字开发nodejs应用批量取出简书网站的文章标题和超链接

    async用来表示函数是异步的,定义的函数会返回一个promise对象,可以使用then方法添加回调函数。

    Jerry Wang
  • 简写js的积累

    写js的时候,有很多小技巧可以让我们的代码更整洁,只是我们都不注意积累,先上几个自己平时用的,以后慢慢积累。

    wade
  • 批量导出某个简书用户的所有文章列表和文章超链接

    虽然简书提供了批量下载文章的功能,但是下载到本地的文章都是markdown格式的,不包含文章的链接,这不满足我的需求。

    Jerry Wang
  • DIY一个Sketch插件,生成猫猫狗狗的全家福

    最近朋友圈都在玩的全家福: ? 看了下是使用 cocos2D 引擎制作的, http://www.cocos.com/creator 主要是图片合成,利用前端...

    mixlab
  • 批量导出某个简书用户的所有文章列表和文章超链接

    虽然简书提供了批量下载文章的功能,但是下载到本地的文章都是markdown格式的,不包含文章的链接,这不满足我的需求。

    Jerry Wang
  • Java 10 var关键字详解和示例教程【面试+工作】

    在本文中,我将通过示例介绍新的Java SE 10特性——“var”类型。你将学习如何在代码中正确使用它,以及在什么情况下不能使用它。

    Java帮帮

扫码关注云+社区

领取腾讯云代金券