Как ниже, я хочу остановить литерал чем-то вроде break
var numbers = [1,2,3,4]
numbers.forEach {
if $0 % 2 == 0 {
break
}
}
Как ниже, я хочу остановить литерал чем-то вроде break
var numbers = [1,2,3,4]
numbers.forEach {
if $0 % 2 == 0 {
break
}
}
forEach
не является циклом (это блок, переданный в цикл, но не сам цикл), или, точнее, forEach
не является частью Swift Flow Control. Вот почему вы не можете использовать break
или continue
.
Просто используйте цикл for-in.
Пример:
var numbers = [ 1,2,3,4]
func firstEvenNumber(inArray array: [Int]) -> Int? {
var firstMatch : Int?
for number in numbers {
if (number % 2) == 0 {
firstMatch = number
break
}
}
return firstMatch
}
firstEvenNumber(inArray: numbers) // 2
Вы можете использовать return
внутри замыкания forEach
, но он не прерывает цикл, он только возвращает блок в текущем проходе.
var numbers = [ 1,2,3,4]
func evenNumbers(inArray: [Int]) -> [Int] {
var matches : [Int] = []
numbers.forEach{
if $0 % 2 == 0 {
matches.append($0)
// early return on even numbers
// does not stop forEach
// comparable to a continue in a for in loop
return
}
// only reached on uneven numbers
}
return matches
}
evenNumbers(numbers) // [2,4]
Документ объяснен.
/// - Note: You cannot use the `break` or `continue` statement to exit the
/// current call of the `body` closure or skip subsequent calls.
/// - Note: Using the `return` statement in the `body` closure will only
/// exit from the current call to `body`, not any outer scope, and won't
/// skip subsequent calls.
Но вы бы хотели попробовать.
extension CollectionType {
/// Use as
///
/// let num = [1,2,3,4,5]
/// num.forEach{
/// if $1 > 3 { $2 = true }
/// print($0,$1)
/// }
func forEach(body: ((Self.Generator.Element, Int, inout Bool) -> Void)) {
var stop = false
let enumerate = self.enumerate()
for (index,value) in enumerate {
if stop { break }
body(value,index,&stop)
}
}
/// Use as
///
/// let num = [1,2,3,4,5]
/// num.forEach{
/// if $0 > 3 { $1 = true }
/// print($0)
/// }
func forEach(body: ((Self.Generator.Element, inout Bool) -> Void)) {
var stop = false
for value in self {
if stop { break }
body(value,&stop)
}
}
}
Поскольку вы хотите остановиться на первом совпадении, вы должны использовать first
а не forEach
:
numbers.first { (number) -> Bool in
return number % 2 == 0
}
Сделайте что-нибудь с результатом (первое четное число в списке):
if let firstEvenNumber = numbers.first(where: { $0 % 2 == 0 }) {
print("The first even number on the list is \(firstEvenNumber)")
}
Бонус: создайте список всех четных чисел, используя filter
:
let allEvenNumbers = numbers.filter { $0 % 2 == 0 }