Go за Прикладом: Відкладення (Defer)

Конструкція defer або відкладення використовують щоб впевнитись, що виклик функції буде виконано пізніше, по завершенню роботи програми, найчастіше - для звільнення залучених ресурсів. defer вживають там, де ensure та finally були б використані в інших мовах.

package main
import (
    "fmt"
    "os"
)

Припустимо, нам потрібно створити файл, записати до нього якусь інформацію, і коли ми закінчимо роботу з ним - закрити його. Ми відкладемо закриття файлу, за допомогою інструкції defer.

func main() {

Після створення об’єкту File ( результат роботи createFile ), ми відкладаємо його закриття за допомогою closeFile. Виклик closeFile буде виконано по закінченні роботи функції main, але вже після того, як writeFile завершить свою роботу.

    f := createFile("/tmp/defer.txt")
    defer closeFile(f)
    writeFile(f)
}
func createFile(p string) *os.File {
    fmt.Println("створюємо")
    f, err := os.Create(p)
    if err != nil {
        panic(err)
    }
    return f
}
func writeFile(f *os.File) {
    fmt.Println("записуємо")
    fmt.Fprintln(f, "дані")
}

Важливо виконувати перевірку на помилки, підчас завершення роботи з файлом, навіть у відкладеній функції.

func closeFile(f *os.File) {
    fmt.Println("закриваємо")
    err := f.Close()
    if err != nil {
        fmt.Fprintf(os.Stderr, "error: %v\n", err)
        os.Exit(1)
    }
}

Запуск програми доводить - файл було закрито після того, як до нього було записано дані.

$ go run defer.go
створюємо
записуємо
закриваємо

Тепер нам відомі такі поняття як паніка (panic) та відкладення (defer), і ми готові ознайомитись з концептом відновлення (recover).

Наступний приклад: Відновлення (Recover).