Defer

Відкладення|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("creating")
	f, err := os.Create(p)
	if err != nil {
		panic(err)
	}
	return f
}

func writeFile(f *os.File) {
	fmt.Println("writing")
	fmt.Fprintln(f, "data")
}

func closeFile(f *os.File) {
	fmt.Println("closing")
	err := f.Close()
	// Важливо виконувати перевірку на помилки, підчас завершення роботи
	// з файлом, навіть у відкладеній функції.
	if err != nil {
		fmt.Fprintf(os.Stderr, "error: %v\n", err)
		os.Exit(1)
	}
}
# Запуск програми доводить - файл було
# закрито після того, як до нього було записано дані.
$ go run defer.go
creating
writing
closing

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