首页 > 代码库 > Swift之“闭包”的应用

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来管理内存的吗?有空得去查下资料,这是因为笔者在写简书时,突然想到的,所以等笔者查完以后,在其它文章里写这些吧。

如果在做的诸位开发者,有不同的理解和见解,欢迎大家和我一起讨论。技术群512847147

Swift之“闭包”的应用