Swift学习:扩展

本篇将详细总结介绍Swift扩展的用法;
扩展就是为一个已有的类、结构体、枚举类型或者协议类型添加新功能。这包括在没有权限获取原始源代码的情况下扩展类型的能力(即逆向建模 )

主要内容:
1.理解扩展
2.扩展的基本使用
3.嵌套类型
4.扩展系统类库

一、理解扩展

1.1扩展特点:

1.扩展和 Objective-C 中的分类类似。(与 Objective-C 不同的是,Swift 的扩展没有名字。)
3.扩展可以为一个类型添加新的功能,但是不能重写已有的功能

1.2扩展功能:

1、添加计算型实例属性和计算型类型属性。
2、定义实例方法和类型方法
3、提供新便利构造器和便利析构器
4、定义下标
5、定义和使用新的嵌套类型
6、使一个已有类型符合某个协议”

1.3使用注意:

1.不可以添加存储属性,也不可以为已有属性添加属性观察器
2.扩展中不能为类添加新的指定构造器,因为指定构造器和析构器必须由原始的类来实现
3.扩展可以为一个类型添加新的功能,但是不能重写已有的功能。
4.通过扩展为一个已有类型添加新功能,那么新功能对该类型的所有已有实例都是可用的,即使它们是在这个扩展定义之前创建的。

二、扩展的基本使用

定义如下的Point 、Size、Rectangle,并对其进行扩展。

struct  Point{
    var x = 0.0
    var y = 0.0
}

struct Size{
    var width = 0.0
    var height = 0.0
}

class Rectangle {
    var origin:Point = Point()
    var size = Size()
    init (origin:Point, size: Size){
        self.origin = origin
        self.size = size
    }
}

2.1.扩展属性

扩展可以添加新的计算型属性,但是不可以添加存储型属性,也不可以为已有属性添加属性观察器

extension Rectangle{
    //注意:扩展不能扩展存储型属性
    //var center:Point = Point() //报错
    //只能扩展计算型属性
    var center: Point {
        get {
            let center_x = origin.x + size.width/2
            let center_y = origin.y + size.height/2
            return Point(x: center_x, y: center_y)
        }
        set{
            origin.x = newValue.x - size.width/2
            origin.y = newValue.y - size.height/2
        }
    }
}

let rect1 = Rectangle(origin: Point(x:0,y:0), size: Size(width: 100, height: 100))
print(rect1.center)              //Point(x: 50.0, y: 50.0)
rect1.center = Point(x: 0, y: 0)
print(rect1.center)              //Point(x: 0.0, y: 0.0)

2.2.扩展方法

扩展可以为已有类型添加新的实例方法和类型方法。

extension Rectangle{
    //注意:这里直接修改了属性,如果是结构体Struct,不能直接这样修改
    //func之前需要使用 mutating
    func translate(x:Double , y:Double){
        self.origin.x += x
        self.origin.y += y
    }
}

let rect2  = Rectangle(origin: Point(x:0,y:0), size: Size(width: 100, height: 100))
rect2.translate(x: 100, y: 100)
print(rect2.center)     //Point(x: 150.0, y: 150.0)

2.3.扩展构造器

扩展能为类添加新的便利构造器,但是它们不能为类添加新的指定构造器或析构器。指定构造器和析构器必须总是由原始的类实现提供.

extension Rectangle{
    //注意:在扩展中添加了构造函数,必须是便利构造函数,其中调用指定构造函数
    convenience init(center:Point, size:Size) {
        let origin_x = center.x - size.width/2
        let origin_y = center.y - size.height/2
        //便利构造函数必须调用指定构造函数
        self.init(origin:Point(x: origin_x, y: origin_y),size:size)
    }
}


let rect3 = Rectangle(center: Point(x:200, y:200), size: Size(width: 100, height: 100))

注意1:如果你使用扩展提供了一个新的构造器,你依旧有责任确保构造过程能够让实例完全初始化。

注意2:如果你使用扩展为一个值类型添加构造器,同时该值类型的原始实现中未定义任何定制的构造器且所有存储属性提供了默认值,那么我们就可以在扩展中的构造器里调用默认构造器和逐一成员构造器。

2.4.扩展下标

扩展可以为已有类型添加新下标。下面的例子为Swift内建类型Int添加了一个整型下标。该下标 [n] 返回十进制数字从右向左数的第n个数字:

extension Int {
    subscript(digitIndex: Int) -> Int {
        var decimalBase = 1
        for _ in 0..<digitIndex {
            decimalBase *= 10
        }
        return (self / decimalBase) % 10
    }
}

 //测试代码:
print(746381295[3]) //1
print(746381295[6]) //6

三、嵌套类型

扩展可以为已有的类、结构体和枚举添加新的嵌套类型。在下面的示例中,Rectangle中就嵌套一个枚举类型,用以获取矩形的各个定点坐标。

extension Rectangle{
    enum Vertex:Int {
        case TopLeft
        case TopRight
        case BottomLeft
        case BottomRight
    }
    
    //扩展出一个方法来获取各个顶点坐标
    func pointAtVertex(v:Vertex) -> Point{
        switch v {
        case .TopLeft:
            return origin
        case .TopRight:
            return Point(x: origin.x + size.width, y: origin.y)
        case .BottomLeft:
            return Point(x: origin.x, y: origin.y + size.height)
        case .BottomRight:
            return Point(x: origin.x + size.width, y: origin.y + size.height)
        }
    }
    
    //扩展下标
    subscript (index:Int) ->Point{
        assert(index >= 0 && index<4,"out of range")
        //修改枚举的原始值是Int类型,而这里🈶使用rowValue方法构建了枚举型
        return pointAtVertex(v: Vertex(rawValue: index)!)  //已经使用了断言,这里使用强制解包
    }
}

//测试代码:
let rect4  = Rectangle(origin: Point(x:0,y:0), size: Size(width: 100, height: 100))
let point1 = rect4.pointAtVertex(v: .TopRight)
print(point1)    //Point(x: 100.0, y: 0.0)
let point2 = rect4.pointAtVertex(v: Rectangle.Vertex.BottomRight)
print(point2)    //Point(x: 100.0, y: 0.0)
let point3 = rect4[2]
print(point3)    //Po

四、扩展系统类库

4.1.扩展Double

扩展Double,为其添加计算型属性,提供与距离单位协作的基本支持。

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 distance1 = 25.4.mm
print("distance1 is \(distance1) meters")    //distance1 is 0.0254 meters
let distance2 = 42.mm + 1.km
print("disatance2 is \( distance2) meters")  //disatance2 is 1000.042 meters

4.2.扩展Int类

扩展Int,为其添加平方、立方、范围判断等方法。

extension Int{
    //平方运算
    var square:Int{
        return self * self
    }
    
    //立方运算
    var cube:Int{
        return self * self * self
    }
    
    //判断整型是否在某个范围内
    func inRange(closedLeft:Int, opendRight:Int) -> Bool{
        return self >= closedLeft && self < opendRight
    }
    
    //通过传入一个闭包参数,简单的实现循环操作
   func  repetitions(task: () -> Void){
        for _ in 0...10{
        //执行闭包
            task()
        }
    }
}

 //测试代码:
var tempNum = 10;
tempNum.square()        //100
tempNum.inRange(closedLeft: 10, opendRight: 100)   //false
tempNum.repetitions {
    print(tempNum)
}
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,951评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,606评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,601评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,478评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,565评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,587评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,590评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,337评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,785评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,096评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,273评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,935评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,578评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,199评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,440评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,163评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,133评论 2 352

推荐阅读更多精彩内容