Swift学习:控制流(Control Flow)

  swift 提供了多种流程控制结构,包括可以多次执行任务的 while 循环,基于特定条件选择执行不同代码分支的 if、guard 和 switch 语句,还有控制流程跳转到其他代码位置的 break 和 continue 语句。

  swift 还提供了 for-in 循环,用来更简单的遍历数组(array),字典(dictionary),区间(range), 字符串(String)和其他序列类型。

  swift 的 switch 语句比 C 语言中更加强大。在 C 语言中,如果某个 case 不小心漏写了break,这个 case 就会贯穿至下一个 case,swift 无需写 break,所以不会发生这种贯穿的情况。case 还可以匹配很多不同的模式,包括间隔匹配(interval match),元组(tuple)和转换到特定类型。switch 语句的 case 中匹配的值可以绑定成临时常量或变量,在 case 体内使用,也可以用 where 来描述更复杂的匹配条件。

  For-In 循环

  你可以使用 for-in 循环来遍历一个集合中的所有元素,例如数字范围、数组中的元素或者字符串中的字符。

  下面的例子用来输出乘 5 乘法表前面的一部分内容:
  for index in 1...5 {

    print("\(index) times 5 is \(index * 5)")

  }

  例子中用来进行遍历的元素是使用闭区间操作符(...)表示的从 1 到 5 的数字区间。index 被赋值为闭区间中的第一个数字(1),然后循环中的语句被执行一次,在本例中,这个循环只包含一个语句,用来输出当前 index 值所对应的乘 5 乘法表的结果。该语句执行后,index 的值被更新为闭区间中的第二个数字(2),之后 print(_:separator:terminator:)函数会再执行一次。整个过程会进行到闭区间结尾为止。

  上面的例子中,index 是一个每次循环遍历开始时被自动赋值的常量。这种情况下,index 在使用前不需要声明,只需要将它包含在循环的声明中,就可以对其进行隐式声明,而无需使用 let 关键字声明。

  如果你不需要区间序列内每一项的值,你可以使用下环线(_)替代变量名来忽略这个值:

  let base = 3

  let power = 10

  var answer = 1

  for _ in 1...power {

    answer *= base

  }

  这个例子计算 base 这个数的 power 次幂,(本例中,是 3 的 10 次幂),从 1 (3 的 0 次幂)开始做 3 的乘法,进行 10 次,使用 1 到 10 的闭区间循环。这个计算并不需要知道每一次循环中计数器具体的值,只需要执行了正确的循环次数即可。下划线符号_ (替代循环中的变量) 能够忽略当前值,并且不提供循环遍历时对值的访问。

  使用 for-in 遍历一个数组所有元素:

  let names = ["Anna", "Alex", "Brian", "Jack"]

  for name in names {

    print("Hello, \(name)")

  }

  你也可以通过遍历一个字典来访问它的键值对。遍历字典时,字典的每项元素会以(key, value)元组的形式返回,你可以在 for-in 循环中使用显式的常量名称来解读(key, value)元组。下面的例子中,字典的键(key)解读为常量 animalName, 字典的值会被解读为常量 legCount:

  let numberOfLegs = ["spider": 8, "ant": 6, "cat": 4]

  for (animalName, legCount) in numberOfLegs {

    print("\(animalName)s have \(legCount) legs")

  }

  字典元素的遍历顺序和插入顺序可能不同,字典的内容在内部是无序的,所以遍历元素时不能保证顺序。

  While 循环

  while 循环会一直运行一段语句直到条件变成 false。这个循环适合使用在第一次迭代前,迭代次数未知的情况下。swift 提供两种while 循环形式:

  while 循环,每次在循环开始时计算条件是否符合

  repeat-while 循环,每次在循环结束时计算条件是否符合。

  while 

  while 循环从计算一个条件开始。如果条件为 true, 会重复运行一段语句,直到条件变为 false。

  下面是 while 循环的一般格式:

  while condition {

    statements

  }

  Repeat-while 

  while 循环的另外一种形式是 repeat-while,它和 while 的区别是在判断循环条件之前,先执行一次循环的代码块。然后重复循环直到条件为 false。

  注意:

  swift 语言的 repeat-while 循环和其他语言中的 do-while 循环是类似的。

  下面是 repeat-while 循环的一般格式:

  repeat {

    statements 

  } while condition

  条件语句

  根据特定的条件执行特定的代码通常是十分有用的。当错误发生时,你可能像运行额外的代码;或者,当值太大或太小时,向用户显示一条消息。要实现这些功能,你就需要使用条件语句。

  swift 提供两种类型的条件语句:if 语句和 switch 语句。通常,当条件较为简单且可能的情况很少时,使用 if 语句。而 switch 语句更适用于条件较复杂、有更多排列组合的时候。并且 switch 在需要用到模式匹配(pattern-matching)的情况下会更有用。

  If

  if 语句最简单的形式就是只包含一个条件,只有该条件为 true 时,才执行相关代码:

  当然 if 语句允许二选一执行,叫做 else 语句。也就是当条件为 false 时,执行 else 语句。

  可以把多个 if 语句连接在一起,来实现更多分支。

  Switch 语句会尝试把某个值与若干个模式(pattern)进行匹配。根据第一个匹配成功的模式,switch 语句会执行对应的代码。当有可能的情况较多时,通常用 switch 语句替换 if 语句。

  switch 语句最简单的形式就是把某个值与一个或若干个相同类型的值作比较:

  switch some value to consider {

    case value 1:

      respond to value 1

    case value2,

       value3:

      respond to value 2 or 3

    default:

      otherwise, do something else

  }

  switch 语句由多个 case 构成,每个由 case 关键字开始。为了匹配某些更特定的值,swift 提供了几种方法来进行更复杂的模式匹配。 

  与 if 语句类似,每一个 case 都是代码执行的一条分支。switch 语句会决定哪一条分支应该被执行,这个流程被称作根据给定的值切换(switching)。

  switch 语句必须是完备的。这就是说,每一个可能的值都必须至少一个 case 分支与之对应。在某些不可能涵所有值的情况下,你可以使用默认 (default)分支来涵盖其它所有没有对应的值。这个默认分支必须在 switch 语句的最后面。

  下面的例子使用 switch 语句来匹配一个名为 someCharacter 的小写字符:

  let someCharacter: Character = "z"

  switch someCharacter {

    case "a":

      print("The first letter of the alphabet")

    case "z":

      print("The last letter of the alphabet")

    default:

      print("Some other character")

    }

  在这个例子中,第一个 case 分支用于匹配第一个英文字母 a, 第二个 case 分支用于匹配最后一个字母 z。因为switch 语句必须有一个 case 分支用于覆盖所有可能的字符,而不仅仅是所有的英文字母,所以 switch 语句使用 default 分支来匹配除了 a 和 z 外的所有值,这个分支保证了switch 语句的完备性。

  不存在隐式的贯穿

  与 C 和 Objective-C 中的 switch 语句不同,在 swift 中,当匹配的 case 分支中的代码执行完毕后,程序会终止switch 语句,而不会继续执行下一个 case 分支。这也就是说,不需要在 case 分支中显式地使用 break 语句。这使得switch 语句更安全、更易用,也避免了因忘记写 break 语句而产生错误。

  注意:

  虽然在 swift 中 break 不是必须的,但你依然可以在 case 分支中的代码执行完毕前使用 break 跳出。

  每一个 case 分支都必须包含至少一条语句。像下面这样书写代码是无效的,因为第一个 case 分支是空的:

  let anotherCharacter: Character = "a"

  switch anotherCharacter {
    case "a": // 无效,这个分支下面没有语句

    case "A":

       print("The letter A")

    default:

       print("Not the letter A")

  }

  // 这段代码会报编译错误

  不像 C 语言里的 switch 语句,在 swift 中,switch 语句不会一起匹配 “a” 和 “A”。相反的,上面的代码会引发编译期错误:case “a”: 不包含任何可执行语句--这就避免了意外地从一个 case 分支贯穿到另外一个,使得代码更安全、也更直观。

  为了让单个 case 同时匹配 a 和 A,可以将这个两个值组合成一个复合匹配,并且用逗号分开:

  let anotherCharacter: Character = "a"

  switch anotherCharacter {

  case "a", "A":

    print("The Letter A")

  default:

    print("Not the letter A")

  }

  为了可读性,复合匹配可以写成多行形式。

  注意:

  如果想要显式贯穿 case 分支,请使用 fallthrough 语句。

  区间匹配

  case 分支的模式也可以是一个值的区间。下面的例子展示了如何使用 区间来输出任意数字对应的自然语言格式:

  let approximateCount = 62

  let countedThings = "moons orbiting Saturn"

  var naturalCount: String 

  switch approximateCount {

  case 0:

    naturalCount = "no"

  case 1..<5:

    naturalCount = "a few"

  case 5..<12:

    naturalCount = "several"

  case 12..<100:

    naturalCount = "dozens of"

  case 100..<1000:

    naturalCount = "hundreds of"

  default:

    naturalCount = "many"

  }

  在上例中,approximateCount 在一个 switch 声明中被评估。每一个 case 都与之进行比较。因为 approximateCount 落在了 12 到 100 的区间,所以 naturalCount 等于 “dozens of” 值,并且此后的执行跳出了 switch 语句。

  元组

  我们可以使用元组在同一个 switch 语句中测试多个值。元组中的元素可以是值,也可以是区间。另外,使用下划线(_)来匹配所有可能的值。

  下面的例子展示了如何使用一个 (Int, Int)类型的元组来分类下面的点(x, y):

  let somePoint = (1, 1)

  switch somePoint {

    case (0, 0):

      print("(0, 0) is at the origin")

    case (_, 0):

      print("(\(somePoint.0), 0) is on the x-axis")

    case (0, _):

      print("(0, \(somePoint.1)) is on the y-aixs")

    case (-2...2, -2...2):

      print("(\(somePoint.0), \(somePoint.1)) is inside the box")

    default:

      print("(\(somePoint.0), \(somePoint.1)) is outside of the box")

    }

  在上面的例子中,switch 语句会判断某个点是否是原点(0,0),是否在红色的 x 轴上,是否在橘黄色的 y 轴上,是否在一个以原点为中心的 4 X 4 的矩形里,或者在这个矩形外面。

  不像 C 语言,swift 允许多个case 匹配同一个值。实际上,在这个例子中,点(0,0)可以匹配所有四个 case。但是,如果存在多个匹配,那么只会执行第一个被匹配到的 case 分支。考虑点(0,0)会首先匹配 case(0, 0), 因此剩下的能够匹配的分支都会被忽视掉。

  值绑定(Value Bindings)

  case 分支允许将匹配的值绑定到一个临时的常量或变量,并且在 case 分支内使用--这种行为被称为值绑定(value binding),因为匹配的值在 case 分支体内,与临时的常量或变量绑定。

  下面的例子展示了如何在一个(Int, Int)类型的元组中使用值绑定来分类下图中的点(x, y):

  let anotherPoint = (2, 0)

  switch anotherPoint {

    case(let x, 0):

      print("on the x-axis with an x value of \(x)")

    case (0, let y):

      print("on the y-axis with a y value of \(y)")

    case let (x, y):

      print("somewhere else at (\(x), \(y))")

    }

    在上面的例子中,switch 语句会判断某个点是否在红色的 x 轴上,是否在橘黄色的 y 轴上,或者不在坐标轴上。这三个 case 都声明了常量 x 和 y 的占位符,用于临时获取元组 anotherPoint 的一个或两个值。第一个 case -- case(let x, 0) 将匹配一个纵坐标为 0 的点,并把这个点的横坐标赋给临时的常量 x。类似的,第二个 case -- case(0, let y) 将匹配一个横坐标为 0 的点,并把这个点的纵坐标赋给临时的常量 y。

  一旦声明了这些临时的常量,它们就可以在其对应的 case 分支里使用。在这个例子中它们用于打印给定点的类型。

  请注意,这个 switch 语句不包含默认分支。这是因为最后一个 case --case let(x, y) 声明了一个可以匹配余下所有值的元组。这使得 switch 语句已经完备了,因此不需要再书写默认分支。

  Where

  case 分支的模式可以使用where 语句来判断额外的条件。

  下面的例子把点(x, y)进行了分类:

  let yetAnotherPoint = (1, -1)

  switch yetAnotherPoint {

    case let (x, y) where x == y:

      print("(\(x), \(y)) is on the line x == y")

    case let (x, y) where x == -y:

      print("(\(x), \(y)) is on the line x == -y")

    case let (x, y) :

      print("(\(x), \(y)) is just some arbitrary point")

  }

  在上面的例子中,switch 语句会判断某个点是否在绿色的对角线 x == y 上,是否在紫色的对角线 x == -y 上,或者不在对角线上。

  这三个 case 都声明了常量 x 和 y 的占位符,用于临时获取元组 yetAnotherPoint 的两个值。这两个常量被用作 where 语句的一部分,从而创建一个动态的过滤器(filter)。当且仅当 where 语句的条件为 true 时,匹配到的 case 分支才会执行。

  就像是值绑定中的例子,由于最后一个 case 分支匹配了余下所有可能的值,switch 语句就已经完备了,因此不需要再书写默认分支。

  复合匹配

  当多个条件可以使用同一种方法来处理时,可以将这几种可能放在同一个 case 后面,并且用逗号隔开。当 case 后面的任意一种模式匹配的时候,这条分支就会被匹配。并且,如果匹配列表过长,还可以分行书写:

  let someCharacter: Character = "e"

  switch someCharacter {

    case "a", "e", "i", "o", "u":

      print("\(someCharacter) is a vowel")

    case "b", "c", "d", "f", "g", "h", "j", "k", "l", "m",

       "n", "p", "q", "r", "s", "t", "v", "w", "x", "y", "z":

      print("\(someCharacter) is a consonant")

    default:

      print("\(someCharacter) is not a vowel or a consonant")

    }

  这个 switch 语句中的第一个 case ,匹配了英语中的五个小写元音字母。相似的,第二个case 匹配了英语中所有的小写辅音字母。最终,default 分支匹配了其它所有字符。复合匹配同样可以包含值绑定。复合匹配里所有的匹配模式,都必须包含相同的值绑定。并且每一个绑定都必须获取到相同类型的值。这保证了,无论复合匹配中的哪个模式发生了匹配,分支体内的代码,都能获取到绑定的值,并且绑定的值都是一样的类型。

  let stillAnotherPoint = (9, 0)

  switch stillAnotherPoint {

    case (let distance, 0), (0, let distance):

      print("On an axis, \(distance) from the origin")

    default:

      print("Not on an axis")

  }

  上面的 case 语句有两个模式:(let distance, 0)匹配了在 x 轴上的值,(0, let distance)匹配了在 y 轴上的值。两个模式都绑定了 distance , 并且distance 在两种模式下,都是整型--这意味着分支体内的代码,只要 case 匹配,都可以获取到 distance 的值。

  控制转移语句

  控制转移语句改变你代码的执行顺序,通过它可以实现代码的跳转。swift 有五种控制转移语句:

  continue

  break

  fallthrough

  return

  throw

  会在下面讨论 continue、break 和 fallthrough 语句。return 语句将在在 函数章节分析,throw 语句会在错误抛出章节分析。

  continue

  continue 语句告诉一个循环体立刻停止本次循环,重新开始下次循环。就好像在说“本次循环我已经执行完了”,但是不会离开整个循环体。

  下面的例子把一个小写字符串中的元音字母和空格字符移除,生成了一个含义模糊的短句:

  let puzzleInput = "great minds think alike"

  var puzzleOutput = ""

  for character in puzzleInput.characters {

    switch character {

      case "a", "e", "i", "o",  "u", " ":

        continue

      default:

        puzzleOutput.append(character)

      }

    }

  }

  在上面的代码中,只要匹配到元音字母或者空格字符,就调用 continue 语句,使本次循环结束,重新开始下次循环。这种行为使 switch 匹配到元音字母和空格字符时不做处理,而不是让每一个匹配到的字符都被打印。

  Break

  break 语句会立刻结束整个控制流的执行。当你想要更早的结束一个 switch 代码块或者一个循环体时,你都可以使用 break 语句。

  循环语句中的break

  当在一个循环体中使用 break 时,会立刻中断该循环体的执行,然后跳转到表示循环体结束的大括号(})后的第一行代码。不会再有本次循环的代码被执行,也不会再有下次循环的产生。

  switch 语句中的 break

  当在一个 switch 代码块中使用 break 时,会立即中断该 switch 代码块的执行,并且跳转到表示 switch 代码块结束的大括号(})后的第一行代码。

  这种特性可以被用来匹配或者忽略一个或多个分支。因为 swift 的 switch 需要包含所有的分支而且不允许有为空的分支,有时为了使你的意图更明显,需要特意匹配或者忽略某个分支。那么当你想忽略某个分支时,可以在该分支内写上 break 语句。当那个分支被匹配到时,分支内的 break 语句立即结束 switch 代码块。

  注意:

  当一个 switch 分支仅仅包含注释时,会被报编译时错误。注释不是代码语句而且也不能让 switch 分支达到被忽略的效果。你应该使用 break 来忽略某个分支。

  下面的例子通过 switch 来判断一个 Character 值是否代表下面四种语言之一。为了简洁,多个值被包含在了同一个分支情况中。

  let numberSymbol: Character = "三" 

  var possibleIntegerValue: Int?

  switch numberSymbol {

    case "1", "?", "一", "?":

      possibleIntegerValue = 1

    case "2", "?", "二", "?":
      possibleIntegerValue = 2

    case "3", "?", "三", "?":

      possibleIntegerValue = 3

    case "4", "?", "四", "?":

      possibleIntegerValue = 4

    default:

      break

  }

if let integerValue = possibleIntegerValue {

   print("The integer value of \(numberSymbol) is \(integerValue).")

 } else {

   print("An integer value could not be found for \(numberSymbol).")

 }

  这个例子检查 numberSymbol 是否是拉丁、阿拉伯、中文 或 泰语中的 1 到 4 之一。如果被匹配到,该 switch 分支语句给 Int?类型变量 possibleIntegerValue 设置一个整数值。

  当 switch 代码块执行完后,接下来的代码通过使用可选绑定来判断 possibleIntegerValue 是否曾经被设置过值。因为可选类型的缘故,possibleIntegerValue 有一个隐式的初始值 nil,所以仅仅当 possibleIntegerValue 曾被 switch 代码块的前四个分支中的某个设置过一个值时,可选绑定才会被判定为成功。

  在上面的例子中,想要把 Character 所有的可能性都枚举出来是不现实的,所以使用 default 分支来包含所有上面没有匹配到字符的情况。由于这个 default 分支不需要执行任何动作,所以它只写了一条 break 语句。一旦落入到 default 分支中后,break 语句就完成了该分支的所有代码操作,代码继续向下,开始执行 if let 语句。

  贯穿

  swift 中的 switch 不会从上一个 case 分支落入到下一个 case 分支中。相反,只要第一个匹配到的 case 分支完成了它需要执行的语句,整个 switch 代码块完成了它的执行。相比之下,C 语言要求你显式的插入 break 语句到每个 case 分支的末尾来阻止自动落入到下一个 case 分支中。swift 的这种避免默认落入到下一个分支中的特性意味着它的 switch 功能要比 C 语言的更加清晰和可预测,可以避免无意识地执行多个 case 分支从而引发的错误。

  如果你确实需要 C 风格的贯穿特性,你可以在每个需要该特性的 case 分支中使用 fallthrough 关键字。下面的例子使用 fallthrough 来创建一个数字的描述语句。

  let integerToDescribe = 5

  var description = "The number \(integerToDescribe) is"

  switch integerToDescribe {

    case 2, 3, 5, 7, 11, 13, 17, 19:

      description += " a prime number, and also"

      fallthrough

    default:

      description += " an integer."

  }

  print(description)

  这个例子定义了一个 string 类型的变量 description 并且给它设置了一个初始值。函数使用 switch 逻辑来判断 integerToDescribe 变量的值。当 integerToDescribe 的值属于列表中的质数之一时,该函数在 description 后添加一段文字,来表明这个数字是一个质数。然后它使用 fallthrough 关键字来 “贯穿” 到 default 分支中。default 分支在 description 的最后添加一段额外的文字,至此 switch 代码块执行完了。

  如果 integerToDescribe 的值不属于列表中的任何质数,那么它不会匹配到第一个 switch 分支。而这里没有其他特别的分支情况,所以 integerToDescribe 匹配到 default 分支中。

  当 switch 代码块执行完后,使用 print(_:separator:terminator:) 函数打印该数字的描述。在这个例子中,数字 5 被准确的识别为一个质数。

  注意:

  fallthrough 关键字不会检查它下一个将落入执行的 case 中的匹配条件。fallthrough 简单地使代码继续连接到下一个 case 中的代码,这和 C 语言标准中的 switch 语句特性是一样的。

  带标签的语句

  在 swift 中,你可以在循环体和条件语句中嵌套循环体和条件语句来创造复杂的控制流结构。并且,循环体和条件语句都可以使用 break 语句来提前结束整个代码块。因此,显式地指明 break 语句想要终止的是哪个循环体或者条件语句,会很有用。类似地,如果你有许多嵌套的循环体,显式指明 continue 语句想要影响哪一个循环体也会非常有用。

  为了实现这个目的,你可以用标签(statement label)来标记一个循环体或者条件语句,对于一个条件语句,你可以使用 break 加标签的方式,来结束这个被标记的语句。 对于一个循环语句,你可以使用 break 或者 continue 加标签,来结束或者继续这条被标记语句的执行。

  声明一个带标签的语句是通过在该语句的的关键词的同一行前面放置一个标签,作为这个语句的前导关键字(introducor keyword), 并且该标签后面跟随一个冒号。下面是一个针对 while 循环体的标签语法,同样的规则适用于所有的循环体和条件语句。

  label name: while condition { statements }

  注意:

  如果上述的 break 语句没有使用 gameLoop 标签,那么它将会中断 switch 语句而不是 while 循环。使用 gameLoop 标签清晰的表明了 break 想要中断的是哪个代码块。同时请注意,当调用 continue gameLoop 去跳转到下一次循环迭代时,这里使用 gameLoop 标签并不是严格必须的。因为这个游戏中,只有一个循环体,所以 continue 语句会影响到哪个循环体是没有歧义的。然而,continue 语句使用 gameLoop 标签也是没有危害的。这样做符合标签的使用规则,同时参照旁边的 break gameLoop, 能够使游戏的逻辑更加清晰和易于理解。

  提前退出

  像 if 语句一样,guard 的执行取决于一个表达式的布尔值。我们可以使用 guard 语句来要求条件必须为真时,以执行 guard 语句后的代码。不同于 if 语句,一个 guard 语句总是有一个 else 从句,如果条件不为真则执行 else 从句中的代码。

  func greet(person: [String: String]) {

    guard let name = person["name"] else {

      return

    }

    print("Hello \(name)")

    guard let location = person["location"] else {

      print("I hope the weather is nice near you.")

      return

    }

    print("I hope the weather is nice in \(location).")

  }

  greet(["name": "John"])

  // 输出 “Hello John!”

  // 输出 "I hope the weather is nice near you."

  greet(["name": "Jane", "location": "Cupertino"])

  // 输出 “Hello, Jane!”

  // 输出 "I hope the weather is nice in Cupertino."

  如果 guard 语句的条件被满足,则继续执行 guard 语句大括号后的代码。将变量或者常量的可选绑定作为 guard 语句的条件,都可以保护 guard 语句后面的代码。

  如果条件不被满足,在 else 分支上的代码就会被执行。这个分支必须转移控制以退出 guard 语句出现的代码段。它可以用控制语句如 return,break,continue 或者 throw 做这件事,或者调用一个不返回的方法或函数,例如 fatalError()。

  相比于可以实现同样功能的 if 语句,按需使用 guard 语句会提升我们代码的可读性。它可以使你的代码连贯的被执行而不需要将它包在 else 块中,它可以使你在紧邻条件判断的地方,处理违规的情况。

  检测 API 的可用性

  swift 内置支持检查 API 可用性,这可以确保我们不会在当前部署机器上,不小心地使用了不可用的 API。

  编译器使用 SDK 中的可用信息来验证我们的代码中使用的所有 API 在项目指定的部署目标上是否可用。如果我们尝试使用一个不可用的 API,Swift 会在编译时出错。

  我们在 if 或 guard 语句中使用 可用性(availability condition)去有条件的执行一段代码,来在运行时判断调用的 API 是否可用。编译器使用从可用性条件语句中获取的信息去验证,在这个代码块中调用的 API 是否可用。

  if #available(iOS 10, macOS 10.12, *) {

    // 在 iOS 使用 iOS 10 的 API,在 macOS 使用 macOS 10.12 的 API

  } else {

    // 使用先前版本的 iOS 和 macOS 的 API

  }

  以上可用性条件指定,在 iOS 中, if 语句的代码块仅仅在 iOS 10 及更高的系统下运行; 在 macOS 中,仅在 macOS 10.12 及更高才会运行。最后一个参数, * ,是必须的,用于指定在所有其它平台中,如果版本号高于你的设备指定的最低版本, if 语句的代码块将会运行。

  在它的一般形式中,可用性条件使用了一个平台名字和版本列表。平台名字可以是 iOS ,macOS, watchOS 和 tvOS (访问声明属性来获取完整列表)。除了指定像 iOS 8 的主版本号,我么可以指定像 iOS 8.3 以及 macOS 10.10.3 的子版本号。

  if #available(platform name version, ..., *)  {

    APIs 可用,语句将执行

  } else {

    APIs 不可用, 语句将不执行。

  }

END  

posted @ 2017-11-15 07:25  鳄鱼不怕牙医不怕  阅读(373)  评论(0编辑  收藏  举报