Swift之“闭包”的应用


相信了解swift的,都应该知道,swift舍弃了OC中的block,但是block的应用又比较灵活,作用也很强大。swift中当然不能缺少这样的设计模式,于是推出了闭包,其功能和作用与OC的block有着异曲同工之妙,但是在写法上,差距比较大。本文,主要是来一探“闭包”的使用方法。
一:闭包简介
  1,闭包也是一个函数,函数是什么?函数就是一段代码块。这个没有什么可纠结的。  2,闭包的使用方法,闭包主要有两种使用方法,1),闭包在方法中的应用。2),闭包在属性中的应用。
二:闭包的结构分析
  mathFunction:(text:String) -> Void
mathFunction是闭包的名字(名字后面必须跟冒号)()小括号里面的是参数text是参数名String是参数类型->指向意思,其后面表示的就是返回值类型了注:如果是要带多个参数的话,其格式为(参数:参数类型 , 参数:参数类型)中间用逗号隔开。
三:闭包在方法的应用
方式1:直接在方法中插入闭包
  /**闭包在方法中应用*/
  func dataBack(str:String , mathFunction:(text:String) -> Void){

      print("\(str)");
      mathFunction(text: "这是闭包在方法中的应用");
  }
这种定义方式,其带了两个参数,一个是Sting类型的,一个就是闭包,只不过闭包比较特殊,你可以理解为参数,也可以有其它理解。
    mathFunction(text: "这是闭包在方法中的应用");
这句代码是调用闭包,此代码中,是在方法里面调用了闭包,也就是在调用这个方法时,闭包也会被自动调用。如果你想在其它地方调用闭包,这时候就用到属性了,在这里就先不说了,在方式二中,我会讲到。
方式2:使用typealias定义闭包,放到方法和属性中。
 其相当于别名的作用了,先定义一个别名为newFuncy的闭包      typealias newFuncy = (text : NSString) -> Void;
定义一个myFuncy属性,其类型为newFuncy,其作用会在下面讲到var myFuncy:newFuncy?;
/**闭包在方法中第二种应用*/
func secondDataBack(str:String , mathFunction:newFuncy){

    print("\(str)");
    self.myFuncy = mathFunction;
}
这是使用typealias定义的闭包,在方式一中,我是在方法中直接调用闭包,在这里,就是我要讲的是,如果你想在其他地方,调用闭包的话,这时,你在方法中,将方法带有的闭包,赋值给你自己定义的属性闭包,在你想要调用闭包的地方,调用这个属性就可以了。相信这点大家都能理解吧!如果有不理解的,可以随时提问。
四:属性闭包的使用
如果上面的方式,大家觉得比较繁琐,想像block一样,在属性中简单灵活的运用,在这里,我告诉大家,是可以的。
首先定义一个闭包属性(当然你也可以采用typealias的方式来定义)
//闭包的属性的用法
var my:((text : NSString) -> Void)?;
我们都知道block定义成属性时,我们可以直接用他的set方法,这样,在block被调用时,其set方法内部的代码就会执行,但是闭包的,可以用set方法吗?swift中,set方法写起来也比较费劲,而且这样定义的属性,不能直接调用set方法吧?
笔者在探究过程中,是没有成功的写出它的set方法,如果有写出来的,希望你能在评论中加上,定当感激不尽。
来看一看,我自己探索出来的写法吧!
//利用闭包属性传递消息,注意,此代码不能写在与属性闭包定义的同一个类里,  会崩溃,具体崩溃原因,正在思考。
    viewC!.my = { (text) -> Void in
        self.str = "sss";
        print(text);
    }
这是笔者自己探究出来的一种写法,这样在调用闭包时,in后代码就会执行。里面的注意,相信大家也都看到了,有兴趣的可以去尝试下,笔者目前猜测,崩溃原因与“视图构造器”有关。如果有大神,能给出详细的解释,笔者跪谢啊。
最后一点,也是很多初学者经常容易忘得了,那就是在属性闭包调用时,要加一个判断,加什么判断呢?----->判断闭包是否存在。 为什么加判断呢?---->为了使逻辑更严谨,代码更安全。
  if((self.my) != nil){
      my!(text:"这是闭包在属性中的应用");
    }
判断方式,类似上面的代码。
五:闭包中类似block的引用计数增加的探究
首先定义两个属性,其实两个可以说成一个。
var str:NSString?;

var _str:NSString{

    set{
        str = newValue;
    }

    get{
        return str!;
    }
}
然后在属性闭包,和方法闭包中分别给这两个属性赋值。
viewC!.my = { (text) -> Void in

        self.str = "sss";
        self._str = "ssss"
    }

viewC?.dataBack("传给带闭包的方法的参数", mathFunction: { (text) -> Void in

        self.str = text;
        self._str = "ssss"
    });
通过探究发现,笔者的这两种闭包的写法,在其内部使用全局变量,是不会有黄色警告的,也就是说,没有对变量进行引用计数+1。说到这里,笔者突然想到一个问题,swift也是采用ARC来管理内存的吗?有空得去查下资料,这是因为笔者在写简书时,突然想到的,所以等笔者查完以后,在其它文章里写这些吧。

下面关于Swift的内容你可能也喜欢:

Ubuntu 14.04.4 下安装 Swift 2.2.1 

Ubuntu 15.10安装部署Swift开发环境   

Swift 的变化:从 2.2 到 3.0 会带来什么   

Swift 正式开源,同时开源 Swfit 核心库和包管理器  

Apple Swift学习教程   

使用 Swift 构建一个 iOS 的邮件应用  

Swift 2.0开源化   

Linux下搭建Swift语言开发学习环境  

Swift 的详细介绍:请点这里 

本文永久更新链接地址

相关内容