在OC中习惯用block来传值,而swift中,block被重新定义了一下,叫闭包;
使用的技巧:谁定义谁传值;
案例使用A、B控制器:
1~4步在B中执行,最后在A中执行;
- B控制器:
1-定义
格式: typealias 闭包名称 = (参数名称: 参数类型) -> 返回值类型
typealias block = (str: String) -> void
2- 声明
var callBack = block?()
3- 赋值
需要定义一个方法,参数是和block类型一致得闭包,并赋值给block
```
func callBackFunction ( block: (str: String) -> Void ) {
callBackBlock = block
}
```
4- 传值
func buttonClick () { //需要传值的方法
if callBackBlock != nil {
callBackBlock!( "传这个值给A") //注意,这里是使用属性传值,不是方法
}
}
5 - A控制器中
B.callBackFunction { (str) in
print("这里使用传过来的值")
}
swift中的闭包和oc中的block的定义和用法对比
0.1 2017.06.01 15:41* 字数 521 阅读 5981评论 0喜欢 11
一.闭包的介绍
二.block的用法回顾
<1>. block写法总结:
block的写法:
类型:
返回值类型(^block的名称)(block的参数)
值:
^(参数列表) {
// 执行的代码
}
//例子
int (^sumOfNumbers)(int a,int b) = ^(int a,int b) {
return a + b;
};
<2>. block实现两个界面之间的传值
①在后面控制器的 .h文件 中声明block
// 一会要传的值为Nsstring类型
typedef void (^newBlock)(Nsstring *);
@interface NewViewController : UIViewController
// 声明block属性
@property (nonatomic,copy) newBlock block;
②在后面控制器的 .m文件 中设置block
- (void)viewWilldisappear:(BOOL)animated
{
[super viewWilldisappear:YES];
if (self.block != nil) {
self.block(@"呵呵");
}
}
③在前面控制器的 .m文件 中接收传来的值
NewViewController *newVC = [[NewViewController alloc] init];
// 接收block传来的值
__weak ViewController *weakSelf = self;
newVC.block = ^(Nsstring *str){
NSLog(@"%@,%@",weakSelf,str);
};
<3>. block作为参数进行延时回调
@interface HttpTool : NSObject
-(void)loadRequest:(void (^)())callBackBlock;
@end
@implementation HttpTool
-(void)loadRequest:(void (^)())callBackBlock
{
dispatch_async(dispatch_get_global_queue(0,0),^{
NSLog(@"异步延时请求操作在这里,加载网络数据:%@",[NSThread currentThread]);
dispatch_async(dispatch_get_main_queue(),^{
callBackBlock();
});
});
}
@end
-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
[self.httpTool loadRequest:^{
NSLog(@"主线程中,将数据回调.%@",[NSThread currentThread]);
}];
}
三.闭包的用法
<1>. 闭包写法总结:
类型:(形参列表)->(返回值)
技巧:初学者定义闭包类型,直接写()->().再填充参数和返回值
值:
{
(形参) -> 返回值类型 in
// 执行代码
}
let b = { (parm : Int) -> (Int) in
print(parm)
}
//调用
b(100)
<2>.闭包的简写
httpTool.loadRequest({
print("回到主线程",NSThread.currentThread());
})
httpTool.loadRequest() {
print("回到主线程",NSThread.currentThread());
}
// 开发中建议该写法
httpTool.loadRequest {
print("回到主线程",NSThread.currentThread());
}
<3>.使用闭包代替block,闭包作为参数进行延时回调
class HttpTool: NSObject {
func loadRequest(callBack : ()->()){
dispatch_async(dispatch_get_global_queue(0,0)) { () -> Void in
print("加载数据",[NSThread.currentThread()])
dispatch_async(dispatch_get_main_queue(),{ () -> Void in
callBack()
})
}
}
}
override func touchesBegan(touches: Set<UITouch>,withEvent event: UIEvent?) {
// 网络请求
httpTool.loadRequest ({ () -> () in
print("回到主线程",NSThread.currentThread());
})
}
<3>.实例二,闭包的回调传值
//[weak self]:解决循环引用导致的内存泄露
override func touchesBegan(_ touches: Set<UITouch>,with event: UIEvent?) {
delayMethod {[weak self] (re) ->() in
print("$$$$$$$$$$$$$$$$$:\(re)%%%%%%%%%%%\(String(describing: self))")
}
delayMethod(comletion: {[weak self] (re)->() in
print("********:\(re)*************\(String(describing: self))")
})
}
//@escaping:逃逸闭包。它的定义非常简单而且易于理解。如果一个闭包被作为一个参数传递给一个函数,并且在函数return之后才被唤起执行,那么这个闭包是逃逸闭包。
func delayMethod(comletion: @escaping (_ results: String,_ resultss:String) -> ()) ->(){
//开启一个全局异步子线程
dispatchQueue.global().async {
Thread.sleep(forTimeInterval: 2.0)
//回调到主线程
dispatchQueue.main.async(execute: {
print("主线程更新 UI \(Thread.current)")
comletion("qwertyui","asdf")
})
}
}
<4>.闭包进行两个界面的传值
class NewViewController: UIViewController {
//声明闭包
typealias lewisCloser = (_ paramOne : String? ) -> ()
//定义个变量 类型就是上面声明的闭包
var customeCloser: lewisCloser?
override func touchesBegan(_ touches: Set<UITouch>,with event: UIEvent?) {
if(customeCloser != nil) {
customeCloser!("要发给第一个界面的值")
}
self.dismiss(animated: true,completion: nil)
}
override func viewDidLoad() {
super.viewDidLoad()
// Do any additional setup after loading the view.
}
}
<2>.在第一个界面实现闭包,取得要穿的值
let vc = NewViewController()
//实现闭包
vc.customeCloser = {(cusValue) -> () in
//cusValue就是传过来的值
print("^^^^^^^^^^^^^^^^^^^^^:\(cusValue!)")
}
self.present(vc,animated: true,completion: nil)
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。