Swift 中闭包的简单使用

本文主要是介绍Swift中闭包的简单使用,将从“闭包的定义”、"闭包的创建、赋值、调用"、“闭包常见的几种使用场景”,"使用闭包可能引起的循环强引用" 四个方面入手,重点介绍闭包如何使用,没有高深的概念,只是专注于实际使用,属于入门级水平,后面还会有关于闭包更加详细和深入理解的文章。希望大家在阅读完本文后能够对闭包有一个整体的理解以及能够简单的使用它。

闭包的定义

在Swift开发文档中是这样介绍闭包的:闭包是可以在你的代码中被传递和引用的功能性独立模块。Swift 中的闭包和 C 以及 Objective-C 中的 block 很像,还有其他语言中的匿名函数也类似。闭包的作用主要是:够捕获和存储定义在其上下文中的任何常量和变量的引用, 能够为你处理所有关于捕获的内存管理的操作(概念性问题,可以不用纠结太多啦)。

闭包的表达式语法

闭包表达式语法有如下的一般形式:

 { (parameters/接收的参数) -> (return type/闭包返回值类型) in
  statements/保存在闭包中需要执行的代码
 }

闭包根据你的需求是有类型的,闭包的类型 一般形式如下:

(parameters/接收的参数) -> (return type/闭包返回值类型)

利用typealias为闭包类型定义别名

这里先介绍一下 typealias的使用 : typealias是Swift中用来为已经存在的类型重新定义名字的关键字(类似于OC语法中的 typedef),重新命名的新名字用来替代之前的类型,并且能够使代码变得更加清晰简单容易理解。typealias 的用法很简单,直接用 = 赋值就可以了:

typealias <type name> = <type expression>

这里我们可以用 typealias 来为看似较为复杂的闭包类型定义别名,这样以后我们就可以用别名直接去申明这样类型的闭包了,例子如下:

//为没有参数也没有返回值的闭包类型起一个别名
 typealias Nothing = () -> ()

 //如果闭包的没有返回值,那么我们还可以这样写,
 typealias Anything = () -> Void

 //为接受一个Int类型的参数不返回任何值的闭包类型 定义一个别名:PrintNumber
 typealias PrintNumber = (Int) -> ()

 //为接受两个Int类型的参数并且返回一个Int类型的值的闭包类型 定义一个别名:Add
 typealias Add = (Int, Int) -> (Int)

闭包是否接受参数、接受几个参数、返回什么类型的值完全取决于你的需求。

闭包的创建、赋值、调用

闭包表达式语法能够使用常量形式参数、变量形式参数和输入输出形式参数,但不能提供默认值。可变形式参数也能使用,但需要在形式参数列表的最后面使用。元组也可被用来作为形式参数和返回类型。在闭包的中会用到一个关键字in,in 可以看做是一个分割符,他把该闭包的类型和闭包的函数体分开,in前面是该闭包的类型,in后面是具体闭包调用时保存的需要执行的代码。表示该闭包的形式参数类型和返回类型定义已经完成,并且闭包的函数体即将开始执行。这里总结了一下可能用到的几种形式实现闭包的创建、赋值、调用的过程。例子如下:

方式一:利用typealias最完整的创建

//为(_ num1: Int, _ num2: Int) -> (Int) 类型的闭包定义别名:Add
 typealias Add = (_ num1: Int, _ num2: Int) -> (Int)
//创建一个 Add 类型的闭包常量:addCloser1
 let addCloser1: Add
//为已经创建好的常量 addCloser1 赋值
 addCloser1 = {
  (_ num1: Int, _ num2: Int) -> (Int) in
  return num1 + num2
 }
//调用闭包并接受返回值
 let result = addCloser1(20, 10)

形式二:闭包类型申明和变量的创建合并在一起

//创建一个 (_ num1: Int, _ num2: Int) -> (Int) 类型的闭包常量:addCloser1
 let addCloser1: (_ num1: Int, _ num2: Int) -> (Int)
//为已经创建好的常量 addCloser1 赋值
 addCloser1 = {
   (_ num1: Int, _ num2: Int) -> (Int) in
   return num1 + num2
 }
 //调用闭包并接受返回值
 let result = addCloser1(20, 10)

形式三:省略闭包接收的形参、省略闭包体中返回值

//创建一个 (Int, Int) -> (Int) 类型的闭包常量:addCloser1
 let addCloser1: (Int, Int) -> (Int)
//为已经创建好的常量 addCloser1 赋值
 addCloser1 = {
   (num1, num2) in
   return num1 + num2
 }
//调用闭包并接受返回值
 let result = addCloser1(20, 10)

形式四:在形式三的基础上进一步精简

//创建一个 (Int, Int) -> (Int) 类型的闭包常量:addCloser1 并赋值
 let addCloser1: (Int, Int) -> (Int) = {
   (num1, num2) in
   return num1 + num2
  }
//调用闭包并接受返回值
 let result = addCloser1(20, 10)

形式五:如果闭包没有接收参数省略in

//创建一个 () -> (String) 类型的闭包常量:addCloser1 并赋值
  let addCloser1: () -> (String) = {
   return "这个闭包没有参数,但是有返回值"
   }
 //调用闭包并接受返回值
  let result = addCloser1()

形式六:简写的实际参数名

//创建一个 (String, String) -> (String) 类型的闭包常量:addCloser1 并赋值
  let addCloser1: (String, String) -> (String) = {
   return "闭包的返回值是:\($0),\($1)"
   }
 //调用闭包并接受返回值
  let result = addCloser1("Hello", "Swift!")

说明: 得益于Swift的类型推断机制,我们在使用闭包的时候可以省略很多东西,而且Swift自动对行内闭包提供简写实际参数名,你也可以通过 $0, $1, $2 等名字来引用闭包的实际参数值。如果你在闭包表达式中使用这些简写实际参数名,那么你可以在闭包的实际参数列表中忽略对其的定义,并且简写实际参数名的数字和类型将会从期望的函数类型中推断出来。in关键字也能被省略,$0 和 $1 分别是闭包的第一个和第二个 String类型的 实际参数(引自文档翻译)。

闭包常见的几种使用场景

基本掌握闭包的概念后,我们就可以利用闭包做事情了,下面介绍一下闭包在开发中的可能被用到的场景。
场景一:利用闭包传值
开发过程中常常会有这样的需求:一个页面的得到的数据需要传递给前一个页面使用。这时候使用闭包可以很简单的实现两个页面之间传值。


图片发自简书App

场景再现:

第一个界面中有一个用来显示文字的UILabel和一个点击进入到第二个界面的UIButton,第二个界面中有一个文本框UITextField和一个点击返回到上一个界面的UIButton,现在的需求是在第二个界面的UITextField中输入完文字后,点击返回按钮返回到第一个界面并且将输入的文字显示在第一个界面(当前页面)的UILabel中。

实现代码:

首先在第二个界面的控制器中定义一个( String) -> ()可选类型的闭包常量closer作为SecondViewController的属性。closer接收一个String类型的参数(就是输入的文字)并且没有返回值。然后在返回按钮的点击事件中传递参数执行闭包。

import UIKit
class SecondViewController: UIViewController {
 //输入文本框
 @IBOutlet weak var textField: UITextField!
 //为创建一个(String) -> () 的可选类型的闭包变量作为控制器的属性
 var closer: ((String) -> ())?
 //返回按钮的点击事件
 @IBAction func backButtonDidClick(_ sender: AnyObject) {
  //首先判断closer闭包是否已经被赋值,如果已经有值,直接调用该闭包,并将输入的文字传进去。
  if closer != nil {
   closer!(textField.text!)
  }
  navigationController?.popViewController(animated: true)
 }
}

这里有一个注意点:我们在为SecondViewController定义变量闭包属性的时候需要将类型申明为可选类型,闭包可选类型应该是((String) -> ())?而不是(String) -> ()?的,后者指的是闭包的返回值是可选类型。

回到第一个界面的控制器中,我们需要拖线拿到UILabel的控件,然后重写prepare(for segue: UIStoryboardSegue, sender:Any?) { }方法,在这个跳转方法中拿到跳转的目标控制器SecondVC并为他的闭包属性赋值,当然如果你的跳转按钮的点击事件是自己处理的,直接在按钮的点击事件中这样做就OK了。

import UIKit
class FirstViewController: UIViewController {
 //显示文字的label
 @IBOutlet weak var label: UILabel!
 //重写这个方法
 override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
  //拿到跳转的目标控制器
  let secondVC = segue.destination as! SecondViewController
  //为目标控制器的闭包属性赋值
  secondVC.closer = {
   //将闭包的参数(输入的文本内容)显示在label上
   self.label.text = $0
  }
 }
}

经过上面的处理,我们就可以实现两个页面之间的传值了(是不是很简单呢),当然在具体的开发中很可能不是传递文本内容这么简单,当需要传递更复杂的值时,我们可以将传递的值包装成一个模型,直接用闭包传递模型就好了。

场景二:闭包作为函数的参数

在OC语法中block可以作为函数的参数进行传递,在Swift中同样可以用闭包作为函数的参数,还记得上面利用typealias关键字定义别名吗,定义完的别名就是一个闭包类型,可以用它申明一个闭包常量或变量当做参数进行传递。一个最简单的闭包作为函数参数例子如下:

//为接受一个Int类型的参数并且返回一个Int类型的值的闭包类型定义一个别名:Number
 typealias Number = (num1: Int) -> (Int)
 //定义一个接收Number类型的参数没有返回值的方法
 func Text(num: Number) {
  //code
 }

闭包在作为函数的参数进行传递的时候根据函数接收参数的情况有很多种不同的写法。这里我们主要介绍一下尾随闭包的概念。

首先看一下一般形式的闭包作为函数的参数传递:

//拼接两个字符串和一个整数
 func combine(handle:(String, String) -> (Void), num: Int) {
   handle("hello", "world \(num)")
  }
//方法调用
 combine(handle: { (text, text1) -> (Void) in
   print("\(text) \(text1)")
   }, num: 2016)

可以看到上面的combine方法在主动调用的时候依旧是按照func(形参: 实参)这样的格式。当我们把闭包作为函数的最后一个参数的时候就引出了尾随闭包的概念。

一,尾随闭包

尾随闭包是指当需要将一个很长的闭包表达式作为函数最后一个实际参数传递给函数时,一个书写在函数形式参数的括号外面(后面)的闭包表达式:

 func combine1(num:Int, handle:(String, String)->(Void)) {
   handle("hello", "world \(num)")
 }
 combine1(num: 2016) { (text, text1) -> (Void) in
   print("\(text) \(text1)")
 }

进一步:如果闭包表达式被用作函数唯一的实际参数并且你把闭包表达式用作尾随闭包,那么调用这个函数的时候函数名字的()都可以省略:

 func combine2(handle:(String, String)->(Void)) {
   handle("hello", "world")
 }
 combine2 { (text, text1) -> (Void) in
   print("\(text) \(text1)")
 }

二,逃逸闭包

如果一个闭包被作为一个参数传递给一个函数,并且在函数return之后才被唤起执行,那么我们称这个闭包的参数是“逃出”这个函数体外,这个闭包就是逃逸闭包。此时可以在形式参数前写 @escaping来明确闭包是允许逃逸的。
闭包可以逃逸的一种方法是被储存在定义于函数外的变量里。比如说,很多函数接收闭包实际参数来作为启动异步任务的回调。函数在启动任务后返回,但是闭包要直到任务完成——闭包需要逃逸,以便于稍后调用。用我们最常用的网络请求举例来说:

func request(methodType:RequestMethodType, urlString: String, parameters: [String : AnyObject], completed: @escaping (AnyObject?, NSError?) -> ()) {
  // 1.封装成功的回调
  let successCallBack = { (task : URLSessionDataTask?, result : Any?) -> Void in
   completed(result as AnyObject?, nil)
  }
  // 2.封装失败的回调
  let failureCallBack = { (task : URLSessionDataTask?, error : Error?) -> Void in
   completed(nil, error as NSError?)
  }
  //判断是哪种请求方式
  if methodType == .get {
   get(urlString, parameters: parameters, success: successCallBack, failure: failureCallBack)
  } else {
   post(urlString, parameters: parameters, success: successCallBack, failure: failureCallBack)
  }
 }

这里的completed闭包被作为一个参数传递给request函数,并且在函数调用get或post后才会被调用。

使用闭包可能引起的循环强引用
Swift中不当的使用闭包可能会引起循环强引用,之所以称之为“强”引用,是因为它会将实例保持住,只要强引用还在,实例是不允许被销毁的。循环强引用会一直阻止类实例的释放,这就在你的应用程序中造成了内存泄漏。
举个例子:

import UIKit
class ThirdViewController: UIViewController {
 var callBack: ((String) -> ())?
 override func viewDidLoad() {
  super.viewDidLoad()
  printString { (text) in
   print(text)
   //闭包中捕获了self
   self.view.backgroundColor = UIColor.red
  }
 }
 func printString(callBack:@escaping (String) -> ()) {
  callBack("这个闭包返回一段文字")
  //控制器强引用于着callBack
  self.callBack = callBack
 }
 deinit {
  print("ThirdViewController---释放了")
 }
}

当你在定义printString这个方法时执行self.callBack = callBack代码实际上是self对callBack闭包进行了强引用,到这里其实并没有产生循环引用,但是当你在调用printString方法的闭包里面又访问了self.view.backgroundColor属性,此时强引用就发生了,即self引用了callBack,而callBack内部又引用着self,谁都不愿意松手,我们就说这两者之间产生了循环强引用。

使用闭包何时会出现循环强引用 :

当你把一个闭包分配给类实例属性的时候,并且这个闭包中又捕获了这个实例。捕获可能发生于这个闭包函数体中访问了实例的某个属性,比如 self.someProperty ,或者这个闭包调用了一个实例的方法,例如 self.someMethod() 。这两种情况都导致了闭包捕获了self ,从而产生了循环强引用。

闭包循环引用的本质是:

闭包中循环强引用的产生,是因为闭包和类相似(还有一种两个类实例之间的循环强引用),都是引用类型。当你把闭包赋值给了一个属性,你实际上是把一个引用赋值给了这个闭包。两个强引用让彼此一直有效。

如何解决闭包的循环强引用:

方式一:类似于OC中使用__weak解决block的循环引用,Swift中支持使用weak关键字将类实例声明为弱引用类型(注意,弱引用类型总是可选类型),打破类实例对闭包的强引用,当对象销毁之后会自动置为nil,对nil进行任何操作不会有反应。

import UIKit
class ThirdViewController: UIViewController {
 var callBack: ((String) -> ())?
 override func viewDidLoad() {
  super.viewDidLoad()
  //将self申明为弱引用类型,打破循环引用
  weak var weakSelf = self
  printString { (text) in
   print(text)
   //闭包中铺捕获了self
   weakSelf?.view.backgroundColor = UIColor.red
  }
 }
 func printString(callBack:@escaping (String) -> ()) {
  callBack("这个闭包返回一段文字")
  //控制器强引用于着callBack
  self.callBack = callBack
 }
 deinit {
  print("ThirdViewController---释放了")
 }
}

方式二:作为第一种方式的简化操作,我们可以在闭包的第一个大括号后面紧接着插入这段代码[weak self],后面的代码直接使用self?也能解决循环引用的问题。

import UIKit
class ThirdViewController: UIViewController {
  var callBack: ((String) -> ())?
  override func viewDidLoad() {
    super.viewDidLoad()
    printString {[weak self] (text) in
      print(text)
      self?.view.backgroundColor = UIColor.red
    }
  }
  func printString(callBack:@escaping (String) -> ()) {
    callBack("这个闭包返回一段文字")
    //控制器强引用于着callBack
    self.callBack = callBack
  }
  deinit {
    print("ThirdViewController---释放了")
  }
}

方式三:在闭包和捕获的实例总是互相引用并且总是同时释放时,可以将闭包内的捕获定义为无主引用unowned。

import UIKit
class ThirdViewController: UIViewController {
  var callBack: ((String) -> ())?
  override func viewDidLoad() {
    super.viewDidLoad()
    printString {[unowned self] (text) in
      print(text)
      self?.view.backgroundColor = UIColor.red
    }
  }
  func printString(callBack:@escaping (String) -> ()) {
    callBack("这个闭包返回一段文字")
    //控制器强引用于着callBack
    self.callBack = callBack
  }
  deinit {
    print("ThirdViewController---释放了")
  }
}

注意:unowned是Swift中另外一种解决循环引用的申明无主引用类型的关键字,类似于OC中的__unsafe_unretained;大家都知道__weak和__unsafe_unretained的相同点是可以将该关键字修饰的对象变成弱引用解决可能存在的循环引用。不同点在于前者修饰的对象如果发现被销毁,那么指向该对象的指针会立即指向nil,而__unsafe_unretained修饰的对象如果发现被销毁,指向该对象的指针依然指向原来的内存地址,如果此时继续访问该对象很容易产生坏内存访问/野指针/僵尸对象访问。
同样的道理Swift中也是一样的。和弱引用类似,无主引用不会牢牢保持住引用的实例。但是不像弱引用,总之,无主引用假定是永远有值的。因此,无主引用总是被定义为非可选类型。你可以在声明属性或者变量时,在前面加上关键字unowned 表示这是一个无主引用。由于无主引用是非可选类型,你不需要在使用它的时候将它展开。无主引用总是可以直接访问。不过 ARC 无法在实例被释放后将无主引用设为 nil ,因为非可选类型的变量不允许被赋值为 nil 。如果此时继续访问已经被释放实例很容易产生坏内存访问/野指针/僵尸对象访问。

所以Swift建议我们如果被捕获的引用永远不为 nil ,应该用unowned而不是weak,相反,如果你不确定闭包中捕获的引用是不是存在为nil的可能,你应该使用weak。

以上的代码是根据最新的Swift3.0语法编写的,经本人在Xcode8.0、iOS10.0环境下编译通过。有任何疑问欢迎在评论区留言,感觉大家的阅读。

(0)

相关推荐

  • Swift 3.0基础学习之闭包

    前言 闭包是功能性自包含模块,可以在代码中被传递和使用. Swift 中的闭包与 C 和 Objective-C中的 blocks 以及其他一些编程语言中的 lambdas 比较相似.下面这篇文章就来详细介绍了关于Swift 3.0中的闭包,感兴趣的一起来看看吧. 开始 闭包的书写格式如下: { (parameters) -> return type in statements } 如 reversedNames = names.sorted(by: { (s1: String, s2: Str

  • 详解Swift中的函数及函数闭包使用

    一.引言 函数是有特定功能的代码段,函数会有一个特定的名称调用时来使用.Swift提供了十分灵活的方式来创建与调用函数.事实上在Swift,每个函数都是一种类型,这种类型由参数和返回值来决定.Swift和Objective-C的一大区别就在于Swift中的函数可以进行嵌套. 而Swift中的闭包是有一定功能的代码块,这十分类似于Objective-C中的block语法.Swift中的闭包语法风格十分简洁,其作用和函数的作用相似. 二.函数的创建与调用 函数通过函数名,参数和返回值来定义,参数和返

  • 深入理解Swift语言中的闭包机制

    在 Swift 中的闭包类似于结构块,并可以在任何地方调用,它就像 C 和 Objective C 语言内置的函数. 函数内部定义的常数和变量引用可被捕获并存储在闭包.函数被视为封闭的特殊情况,它有 3 种形式. 在 Swift 语言闭合表达式,如下优化,重量轻语法风格,其中包括: 推导参数并从上下文菜单返回值的类型 从单封表达的隐性返回 简略参数名称 尾部闭包语法 语法 下面是一个通用的语法定义用于闭包,它接受参数并返回数据的类型: 复制代码 代码如下: {(parameters) -> re

  • Swift教程之闭包详解

    闭包(Closures)是独立的函数代码块,能在代码中传递及使用.Swift中的闭包与C和Objective-C中的代码块及其它编程语言中的匿名函数相似. 闭包可以在上下文的范围内捕获.存储任何被定义的常量和变量引用.因这些常量和变量的封闭性,而命名为"闭包(Closures)".Swift能够对所有你所能捕获到的引用进行内存管理. NOTE 假如你对"捕获(capturing)"不熟悉,请不要担心,具体可以参考Capturing Values(捕获值). 全局函数

  • Swift中闭包实战案例详解

    前言 无论苹果的官方文档还是由官方文档衍生出来的一些文章和书籍都比较重视基础语法知识的讲解,对于实战中的应用提及的都很少,所以当我们想使用"闭包"解决一些问题的时候,会忽然出现看着一堆理论知识却不知从何下手的尴尬感,这就是理论和时实战的区别了. 本文不赘述Swift闭包的的基本语法了,百度或者Google下有很多资料.如题所示本文着重讲述Swift闭包的一些实战案例,有需要的小伙伴可以参考下,经验丰富的大神也请指教. 关于如何理解闭包 学习闭包的第一个难点就是理解闭包,可能很多人用了很

  • IOS swift3.0 下闭包语法整理

    IOS swift3.0 下闭包语法整理 一.闭包的概念 有oc基础的都知道,闭包其实是oc里面的block,语法格式不一样,但作用是一样的.主要是用于callBack(异步回调)或者两个类之间的通信.它的本质一个函数,一个可执行的代码块,只是这个函数是没有名字的,也就是匿名函数.你也可以把他看作如 int.float一样,是一种数据类型,一种可以作为参数传递的数据类型. 二.基本语法 1.闭包的声明 //定义一个求和闭包 //闭包类型:(Int,Int)->(Int) let add:(Int

  • Swift 中闭包的简单使用

    本文主要是介绍Swift中闭包的简单使用,将从"闭包的定义"."闭包的创建.赋值.调用"."闭包常见的几种使用场景","使用闭包可能引起的循环强引用" 四个方面入手,重点介绍闭包如何使用,没有高深的概念,只是专注于实际使用,属于入门级水平,后面还会有关于闭包更加详细和深入理解的文章.希望大家在阅读完本文后能够对闭包有一个整体的理解以及能够简单的使用它. 闭包的定义 在Swift开发文档中是这样介绍闭包的:闭包是可以在你的代码中

  • Swift中优雅处理闭包导致的循环引用详解

    前言 Objective-C 作为一门资历很老的语言,添加了 Block 这个特性后深受广大 iOS 开发者的喜爱.在 Swift 中,对应的概念叫做 Closure,即闭包.虽然更换了名字,但是概念和用法还是相似的,就算是副作用也一样,有可能导致循环引用. 下面我们用一个例子看一下,首先我们需要第一个控制器(FirstViewController),它所做的就是简单的推出第二个控制器(SecondViewController). class FirstViewController: UIVie

  • Swift中转义闭包示例详解

    目录 前言 转义与非转义闭包 逃离方法 将转义关闭付诸行动 注意强参考周期 内存泄漏背后的原因 消除强引用循环 概括 前言 Swift 是一种非常强大的编程语言,是为 Apple 生态系统开发应用程序的首选:iOS.macOS.watchOS 和 tvOS.作为使用 Swift 编写代码的开发人员,我们经常使用闭包:语言的一个重要而重要的章节. 闭包不是初学者开始的主题.然而,这是每个人都必须尽快了解的东西.有很多方面需要了解并了解它们的工作原理.在所有这些中,有一个特定的:转义闭包和@esca

  • Java中闭包简单代码示例

    一.闭包的定义. 有很多不同的人都对闭包过进行了定义,这里收集了一些. # 是引用了自由变量的函数.这个函数通常被定义在另一个外部函数中,并且引用了外部函数中的变量. -- <<wikipedia>> # 是一个可调用的对象,它记录了一些信息,这些信息来自于创建它的作用域.-- <<Java编程思想>> # 是一个匿名的代码块,可以接受参数,并返回一个返回值,也可以引用和使用在它周围的,可见域中定义的变量.-- Groovy ['ɡru:vi] # 是一个表

  • Objective-C中的block与Swift中的尾随闭包使用教程

    前言 在项目开发中经常会去查iOS闭包怎么写,因为它的语法太古怪,两种语言写法不一,经常搞混,干脆记录下常用的写法算了 闭包定义 闭包是指可以包含自由(未绑定到特定对象)变量的代码块:这些变量不是在这个代码块内或者任何全局上下文中定义的,而是在定义代码块的环境中定义(局部变量)."闭包" 一词来源于以下两者的结合:要执行的代码块(由于自由变量被包含在代码块中,这些自由变量以及它们引用的对象没有被释放)和为自由变量提供绑定的计算环境(作用域). OC中的block与Swift中的尾随闭包

  • Swift中的常量和变量简单概述

    1.在Swift中规定:在定义一个标识符时必须明确说明该标识符是一个常量还是变量 2.使用let来定义常量,定义之后不可以修改 3.使用var来定义变量,定义之后可以修改 4.定义常量和变量 常量:let 标识符的名称 : 类型 = 赋值 变量:var 标识符的名称 : 类型 = 赋值 5.类型的首字母要大写,常量与变量名不能包含数学符号,箭头 6.常量的使用注意 6.1 在开发中,apple建议优先使用常量,只有发现需要修改时再改成变量,因为常量更加安全,不会被任意的修改 6.2 常量的本质.

  • 在 Swift 中测试 UIAlertController的方法

    最近我读了一篇在 Objective-C 中使用 control swizzling 测试 UIAlertController 的 文章 .这样的文章总是促使我寻找一种不使用 control swizzling 也可以测试同样东西的方法.虽然,我知道 swizzling 是开发者的一个非常有力的工具,但我个人是尽可能去避免去使用它的.事实上,在最近的六年时间里,我只在一个应用上用了 swizzling.所以我相信我们现在可以不使用 swizzling 来实现测试. 那么问题来了,如何在 Swif

  • Swift中的指针操作详解

    前言 Objective-C和C语言经常需要使用到指针.Swift中的数据类型由于良好的设计,使其可以和基于指针的C语言API无缝混用.但是语法上有很大的差别. 默认情况下,Swift 是内存安全的,这意味着它禁止我们直接操作内存,并且确保所有的变量在使用前都已经被正确地初始化了.但是,Swift 也提供了我们使用指针直接操作内存的方法,直接操作内存是很危险的行为,很容易就出现错误,因此官方将直接操作内存称为 "unsafe 特性". 一旦我们开始直接操作内存,一切就得靠我们自己了,因

随机推荐