相信了解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"
});
解决内存泄露
声明若引用属性
weak var weakSelf = self;
viewC?.dataBack("传给带闭包的方法的参数", mathFunction: { (text) -> Void in
weakSelf str = text;
weakSelf._str = "ssss"
});
如果在做的诸位开发者,有不同的理解和见解,欢迎大家和我一起讨论。技术群512847147
网友评论