Как я мог создать функцию с обработчиком завершения в Swift?

117

Мне просто было любопытно, как я подойду к этому. Если бы у меня была функция, и я хотел бы, чтобы что-то произошло, когда она будет полностью выполнена, как бы я добавил это в функцию? Спасибо

traw1233
источник
2
На Youtube есть потрясающее видео: google.com/…
Bright Future

Ответы:

174

Допустим, у вас есть функция загрузки для загрузки файла из сети и вы хотите получать уведомление, когда задача загрузки завершена.

typealias CompletionHandler = (success:Bool) -> Void

func downloadFileFromURL(url: NSURL,completionHandler: CompletionHandler) {

    // download code.

    let flag = true // true if download succeed,false otherwise

    completionHandler(success: flag)
}

// How to use it.

downloadFileFromURL(NSURL(string: "url_str")!, { (success) -> Void in

    // When download completes,control flow goes here.
    if success {
        // download success
    } else {
        // download fail
    }
})

Надеюсь, поможет.

tounaobun
источник
2
Это сработало бы отлично, но больше из любопытства, мне было интересно, можете ли вы как-то написать обработчик завершения в свою функцию.
traw1233
1
Привет, Floks, я хочу вызвать этот CompletionHandler из другой функции, как этого добиться?
Химаншу джамнани
любой пример для цели c
Xcodian Solangi
Когда я вызываю его другим классом, он не сеет параметр успеха обработчика завершения.
Chandni
86

Мне было трудно понять ответы, поэтому я предполагаю, что у любого другого новичка, такого как я, может быть такая же проблема, как и у меня.

Мое решение делает то же самое, что и главный ответ, но, надеюсь, немного более ясное и легкое для понимания для новичков или людей, которые просто не понимают в целом.

Чтобы создать функцию с обработчиком завершения

func yourFunctionName(finished: () -> Void) {

     print("Doing something!")

     finished()

}

использовать функцию

     override func viewDidLoad() {

          yourFunctionName {

          //do something here after running your function
           print("Tada!!!!")
          }

    }

Ваш результат будет

Делать что-то

Тада !!!

Надеюсь это поможет!

Сирил
источник
80

Простой пример Swift 4.0:

func method(arg: Bool, completion: (Bool) -> ()) {
    print("First line of code executed")
    // do stuff here to determine what you want to "send back".
    // we are just sending the Boolean value that was sent in "back"
    completion(arg)
}

Как это использовать:

method(arg: true, completion: { (success) -> Void in
    print("Second line of code executed")
    if success { // this will be equal to whatever value is set in this method call
          print("true")
    } else {
         print("false")
    }
})
Бобби
источник
12

Для этой цели мы можем использовать замыкания . Попробуйте следующее

func loadHealthCareList(completionClosure: (indexes: NSMutableArray)-> ()) {
      //some code here
      completionClosure(indexes: list)
}

В какой-то момент мы можем вызвать эту функцию, как указано ниже.

healthIndexManager.loadHealthCareList { (indexes) -> () in
            print(indexes)
}

Пожалуйста, перейдите по следующей ссылке для получения дополнительной информации о закрытии .

https://developer.apple.com/library/ios/documentation/Swift/Conceptual/Swift_Programming_Language/Closures.html

arango_86
источник
5

Swift 5.0 +, простой и короткий

пример:

Стиль 1

    func methodName(completionBlock: () -> Void)  {

          print("block_Completion")
          completionBlock()
    }

Стиль 2

    func methodName(completionBlock: () -> ())  {

        print("block_Completion")
        completionBlock()
    }

Использование:

    override func viewDidLoad() {
        super.viewDidLoad()
        
        methodName {

            print("Doing something after Block_Completion!!")
        }
    }

Вывод

block_Completion

Что-то делать после Block_Completion !!

Лахдип Сингх
источник
0

Я немного запутался в обработчиках завершения, сделанных на заказ. В вашем примере:

Допустим, у вас есть функция загрузки для загрузки файла из сети и вы хотите получать уведомление, когда задача загрузки завершена.

typealias CompletionHandler = (success:Bool) -> Void

func downloadFileFromURL(url: NSURL,completionHandler: CompletionHandler) {

    // download code.

    let flag = true // true if download succeed,false otherwise

    completionHandler(success: flag)
}

Ваш по- // download codeпрежнему будет выполняться асинхронно. Почему код не будет идти прямо на let flag = trueи , completion Handler(success: flag)не дожидаясь вашей загрузки кода , который будет закончен?

Тяжелое дыхание
источник
В конце концов, что-то должно сидеть и ждать запуска кода, а не гигантская башня из асинхронных слонов, спущенных вниз. «Выполнить асинхронно» означает, что есть два потока. Один из них сидит и ждет, пока задание будет выполнено, другой продолжает и нет. Обработчик завершения вызывается или, по крайней мере, должен быть вызван по расписанию в конце потока, выполняющего работу.
Crowman 07
0

В дополнение к вышесказанному: Может использоваться скользящее закрытие .

downloadFileFromURL(NSURL(string: "url_str")!)  { (success) -> Void in

  // When download completes,control flow goes here.
  if success {
      // download success
  } else {
    // download fail
  }
}
Shrawan
источник