Swift3.0 函数闭包与OC Block

刚接触Swift,如有不对的点,欢迎指正。转载请说明出处

  • Swift中定义一个基本函数
//定义一个函数,接收一个字符串,返回一个String类型的值
func test(name:String) -> String {

        return ("输出了\(name)")
    }
//通用形式
 func name(parameters) -> return type {
        function body
    }
  • Swift 中基本的闭包函数与OC中Block的相似点

  • 带参闭包

    //OC中Block是这样
    void (^test) (NSString *)=^(NSString *name){
    
            NSLog(@"%@",name);
    
        };
    
        test(@"测试");
    //对应到swift的带参闭包
    
    let test={(_ name:String)->() in
    
      // in 后面就是回调之后处理的函数 ,相当于是Block之后的{ }
                print(name)
    
            }
    
            test("测试")
    
    //通用形式
    {
    (parameters) -> returnType in
       code
    }

这里下划线 _ 的作用用来忽略外部參数名,具体可以参考这篇文章http://www.cnblogs.com/bhlsheji/p/4746072.html

  • OC中带Block参数的函数定义
-(void)loadData:(  void(^)(void) )completetion
{
    completetion();
}
  • Swift中带闭包参数的函数定义
func loadData(completetion: ()->()) -> () {
       completetion()
    }
//样式 func: ()->() 
  • 举个比较常用的例子,网络数据获取
//OC中Block传值
-(void)loadData:(  void(^)(NSArray *) )completetion
{

//这里暂时先忽略掉线程,简单处理,重点在swift闭包
    NSLog(@"耗时操作");

    sleep(2);//模拟网络请求

    NSArray *[email protected][@"1",@"2"];

    NSLog(@"主线程回调");

    completetion(arr); //返回获得的数据

}

调用:
 [self loadData:^(NSArray *callBack){

        NSLog(@"%@,%@",callBack[0],callBack[1]);

    }];

输出:
2017-03-22 18:48:45.273 tessst[3642:187462] 耗时操作
2017-03-22 18:48:47.345 tessst[3642:187462] 主线程更新
2017-03-22 18:48:47.346 tessst[3642:187462] 1,2

//swift中闭包传值
       func loadDate(completion: @escaping (_ result : [String])->()) -> () {

//这里有一个很重要的参数 @escaping,逃逸闭包
//简单来说就是 闭包在这个函数结束前内被调用,就是非逃逸闭包,调用的地方超过了这函数的范围,叫逃逸闭包
//一般网络请求都是请求后一段时间这个闭包才执行,所以都是逃逸闭包。
// 在Swift3.0中所有的闭包都默认为非逃逸闭包,所以需要用@escaping来修饰
        DispatchQueue.global().async {

            print("耗时操作\(Thread.current)")
            Thread.sleep(forTimeInterval: 2)
            let json=["1","2"]

            DispatchQueue.main.async {
                print("主线程更新\(Thread.current)")

                completion(json)
//函数在执行完后俩秒,主线程才回调数据,超过了函数的范围,这里就是属于逃逸闭包,如果不用@escaping,编译器是编译不过的
            }
        }
    }

主函数调用loadDate:
 loadDate { (callBack) in
        print("\(callBack)")
        }

输出值
耗时操作<NSThread: 0x608000069140>{number = 1, name = main}
主线程更新<NSThread: 0x608000069140>{number = 1, name = main}
1,2
  • 在Block中经常会有循环引用的情况,闭包中也一样,常用的解决方式有俩种
//第一种
weak var weakwelf=self
//套用oc的方式(__weak typedef(weakself)=self).
//这里要注意,不能用  let  ,因为self可能会释放指向nil,相当于是一个可变值

//调可选项发送消息的时候 用 ? 解包 不用 !

pringt("weakself ?.view") 不用" weakself!.view"
//因为强制解包一旦weakself=nil时会崩溃

//第二种,在调用时候
//[weak self]标识在{}中所有的self都是弱引用
 loadDate { [weak self] in
            print(self?.view)
        }

续更...

时间: 2024-12-05 18:23:35

Swift3.0 函数闭包与OC Block的相关文章

python学习-42 装饰器 --- 函数闭包1

函数闭包举例: def father(name): print('hello world') def son(): print('儿子说:我的爸爸是%s' % name) def grandfson(): print('孙子说:我的爷爷是%s' % name) grandfson() son() father('小明') 运行结果: hello world 儿子说:我的爸爸是小明 孙子说:我的爷爷是小明 Process finished with exit code 0 函数的包: 就是嵌套里的

详解 swift3.0 和 OC 的混编

前言: 我们在一些情况下,仅仅使用swift 是无法完成一个项目的,在swift项目中必要用到 OC 实现一些功能,比如,项目要使用一些第三方的框架,但这个第三方的框架却是用 OC 实现的,或者你的项目必须调用原有项目的部分功能,而原来的项目也是用 OC 实现的.在这样的一些背景下,你就必须要实现这两种语言的混编.反过来也一样,比如开发者打算用swift实现一些新的功能要用到三方框架,而人家是使用swift写的,而你的项目确实OC的,这时候你也得实现一个混编.下面我们分两种情况分别说明一些. 一

4 .Swift函数|闭包

在编程中,我们常把能完成某一特定功能的一组代码,并且带有名字标记类型叫做函数,在C语言中,我们知道函数名就是一个指针,它指向了函数体内代码区的第一行代码的地址,在swift中也具有同样的功效. 在Swift中函数的表现形式如下: 1. func 函数名(参数名1:参数类型,参数名2,参数类型)->返回值类型 {函数的实现部分} func sayHello()->Void { print("hello world") } //有参数,有返回值, func sayHello2(

Swift语法基础入门三(函数, 闭包)

Swift语法基础入门三(函数, 闭包) 函数: 函数是用来完成特定任务的独立的代码块.你给一个函数起一个合适的名字,用来标识函数做什么,并且当函数需要执行的时候,这个名字会被用于“调用”函数 格式: func 函数名称(参数名:参数类型, 参数名:参数类型...) -> 函数返回值 { 函数实现部分 } 没有参数没有返回值 可以写为 ->Void 可以写为 ->() 可以省略 Void.它其实是一个空的元组(tuple),没有任何元素,可以写成() func say() -> V

Swift2.3 --&gt; Swift3.0 的变化

Swift3.0语法变化 首先和大家分享一下学习新语法的技巧: 用Xcode8打开自己的Swift2.3的项目,选择Edit->Convert->To Current Swift Syntax- 让Xcode帮我们把Swift2.3的代码转换为Swift3.0. 手动调出Xcode自动转换Swift2.3 到 Swift3.0 弹出语言版本选择界面,选择Covert to Swift3,Next:  进入选择模块界面: 选择模块界面 建议只选择自己创建的模块,第三方框架的模块最好不要使用Xco

突然兴起复习一下Swift3.0

/// 参考Swift3.0.1文档 /// 摘录来自: Apple Inc. "The Swift Programming Language (Swift 3.0.1)". iBooks. import UIKit class ViewController: UIViewController { override func viewDidLoad() { super.viewDidLoad() aSwiftTour() } func aSwiftTour(){ //Swift中的输出

UICollectionView在Swift3.0中的用法

UICollectionView在Swift3.0中的用法 UICollectionView的初始化跟OC中是相似的,创建 GameView 集成自 UICollectionView .注意不同于UITableView的用法,他需要用 UICollectionViewFlowLayout 来指定一些需要设置的属性,或者可以通过遵守 UICollectionViewDelegateFlowLayout 这个代理来实现.下面我用设置属性的方式来实现的,比较方便. //布局 layout.scroll

iOS指向函数的指针和block

  一:block基础知识 block基础知识 基本概念:block是用来保存一段代码的:^:是block得标志  好比*:是指针的标志 特点:1:保存一段代码: 2:可以有参数和返回值: 3:可以作为函数的参数传递: 与代码块的区别,代码块里的代码会自动执行,block中代码要手动调用: 二:普通数据类型.指向函数的指针.block的定义的类比 1:基本数据类型: 例如:int a = 10; 格式:数据类型  变量名 = 值: 2:指向函数的指针:可以仿照上边基本数据类型的定义 例如:voi

python之函数式编程与函数闭包

防伪码:忘情公子著 Python函数式编程: 函数式编程: 也称作泛函编程,是一种编程范型,说白了就是实现可以把函数当参数传递给另一个函数: 它将电脑运算视为数学上的函数计算,并且避免状态以及可变数据: 函数式编程语言最重要的基础是lambda演算,而且lambda演算的函数可以接受函数当作输入和输出 Python支持有限的函数式编程功能: filter(func,seq): 调用一个布尔函数func来迭代遍历每个seq中的元素:返回一个使func返回值为true的元素的序列 In [1]: d