谈谈Swift的extension

作者: Sheepy | 来源:发表于2015-08-27 00:25 被阅读11262次

3.times { puts 'hello world' }
这是一条Ruby语句,它会打印“hello world”三次,意图清晰,语法简洁。

如果你是C或者Java等语言的使用者,而从未接触过Ruby,此时是不是有种打开了新世界的大门的感觉_。反正当初我刚接触Ruby那会儿时常会感叹,啊原来代码还可以这么写。。。Ruby的灵活与优雅当然远不仅于此,这里就不展开了。今天的重点是,Swift也可以这么写。

首先我们要明确,Ruby之所以可以这么写是因为它是一个纯面向对象的语言,在Ruby的世界中,一切皆对象,所有的事物都有一个共同的祖先——Object3的类型是Fixnum,当然也是一个对象,所以它有自己的方法。times就是一个方法,可以看出应该是接收一个代码块(本例中为{puts 'hello world'}),然后调用它self(本例中self即为3)次。

而在Swift中,Int是一个struct(结构体),并不是个class。但是Swift中的struct与别的语言中的struct不同的是,struct内部也可以定义方法。所以条件满足了,现在就差个times方法了。

嗯,很遗憾Swift的Int并没有times方法。不过没关系,这个时候extension就派上用场了。没有我们就自己写一个嘛,私人化定制,满足一切突如其来的需求。先上代码好了:
定义:

extension Int {
    func times(closure: () -> ()) {
       for _ in 1...self {
            closure()
        }
    }
}

调用: 3.times { println("hello world") }

是不是跟开头Ruby语句有异曲同工之妙?extension不仅可以扩展方法,还可以扩展计算属性。比如《The Swift Programming Language》中的一个例子:

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
​println​(​"One inch is ​\(​oneInch​)​ meters"​)
​// prints "One inch is 0.0254 meters"
​let​ ​threeFeet​ = ​3​.​ft
​println​(​"Three feet is ​\(​threeFeet​)​ meters"​)
​// prints "Three feet is 0.914399970739201 meters”

extension在实际的开发中可以有很多应用,使用得当能显著提高代码的可读性,平常我们自己写的大多数辅助类其实都可以转化为extension。譬如我最近在自己的项目中给被点击的UIView(比如button)写了一个小动画,点击了之后组件会上浮一下,并且同时会有一下阴影效果。
定义:

extension UIView {
    func animateWhenClicked() {
        self.backgroundColor = UIColor(white: 0.9, alpha: 0.5)
        self.layer.transform = CATransform3DMakeScale(1, 1, 0)
        UIView.animateWithDuration(0.9, animations: {
            self.backgroundColor = UIColor.whiteColor()
            self.layer.transform = CATransform3DMakeTranslation(1, 1, 1)
        })
    }
}

使用(仅点击动画部分代码):

button.addTarget(viewCtrl, action: "clicked:", forControlEvents: .TouchUpInside)
func clicked(sender: UIButton) {
    sender.superview!.animateWhenClicked()
}
iOS Simulator Screen Shot 2015年8月26日 下午11.39.45.png

大概就是右上角的效果,呃,静态图看不出来,各位要是有兴趣可以拿我的扩展方法然后自己初始化个Button去跑跑看(直接sender.animateWhenClicked()就好,不用superview)。因为我在项目中的Button是一个自定义的贪婪Button,只要点击Button的父视图就会响应点击事件,所以是Button的superview调用了animateWhenClicked,这也是我把这个动画扩展到UIView而不是UIButton的原因。

我个人很喜欢extension,平常也经常使用。extension也可用于后期项目维护,可以在不修改旧代码的前提下扩展旧代码的功能。而且对于NSObject的子类或者有dynamic标记的方法,利用扩展还可以在运行时对某些方法的实现进行替换,类似于OC的Swizzle,今天就不展开了,下次有空继续。

相关文章

网友评论

  • 草帽三Jmg:楼主有没有试过 在 storyboard 里将 button 点击方法 拽进 extension​ 扩展里? 我xcode8 发现拽不进去 extension​ Democell {
    @IBAction func btnClick (_ sender:AnyObject) {} // 此方法关联不上 不知道为什么

    }
    一一总是要变二:@Sheepy 楼主高冷啊
    Sheepy:@被消灭加菲 没试过
  • pFruHMXB:看完函数类型再来看您举的例子明朗很多,感谢

    extension Int {
    func times(colosur: () -> Void) {
    for _ in 1...self {
    colosur()
    }
    }
    }

    接受一个函数类型变量 为
    var colosur: () -> Void = print()

    这样就好理解多啦
    b2efe7751b24:多谢了.我一开始也很懵
  • pFruHMXB:好厉害 不过在项目中调用不起来,只在playground中运行起来了.
  • 5150250fc088:子类的类扩展中重写父类的方法不执行,请问这是什么原因。
    5150250fc088:@Sheepy 好的谢谢
    Sheepy:@为啥没带小卢卡斯 你可以看看卓同学这篇文章 http://www.jianshu.com/p/34e04e173ce5
    Sheepy:@为啥没带小卢卡斯 扩展不能重写父类方法的
  • 那一片阳光:可以可以,简洁明了!
  • 07a0b9addc9e:新手不是太懂,问一下,
    self.layer.transform = CATransform3DMakeScale(1, 1, 0)
    self.layer.transform = CATransform3DMakeTranslation(1, 1, 1)这两句干了什么事情?网上搜了一堆还是没整明白
    Sheepy:@疾风哥哥 就是让这个 view 垂直屏幕向外弹了一下
  • x1911:眼界大开,非常喜欢博主的原创,感谢
    Sheepy:@x1911 多谢支持~
  • codeGlider:markdown的代码区块怎么改颜色?求指导~
    codeGlider:@Sheepy 牛逼,跟github一样了啊
    Sheepy:@codeGlider 我昨天写的时候还没颜色,不过我找到根源了:http://www.jianshu.com/p/a62ccbdbb5a8

本文标题:谈谈Swift的extension

本文链接:https://www.haomeiwen.com/subject/fwyhcttx.html