В Go принято сообщать об ошибках через явное, отдельное
возвращаемое значение. Это контрастирует с исключениями,
используемыми в таких языках, как Java и Ruby, и
перегруженным одиночным значением результата/ошибки,
иногда используемым в подходе C. Go, позволяет легко
увидеть, какие функции возвращают ошибки, и обрабатывать
их, используя те же языковые конструкции, которые
используются для любых других задач без ошибок.
|
|
|
|
|
import (
"errors"
"fmt"
)
|
По соглашению, ошибки - это последнее возвращаемое
значение с типом error в стандартной реализации.
|
func f1(arg int) (int, error) {
if arg == 42 {
|
errors.New создает стандартную ошибку с
указаннным сообщением
|
return -1, errors.New("can't work with 42")
|
|
|
Значение nil в поле ошибки, говорит о том, что
ошибок нет.
|
|
Можно использовать пользовательские типы в качестве
ошибок, применяя к ним метод Error() . Вот вариант
в примере выше, который использует пользовательский
тип для явного представления ошибки аргумента.
|
type argError struct {
arg int
prob string
}
|
|
func (e *argError) Error() string {
return fmt.Sprintf("%d - %s", e.arg, e.prob)
}
|
|
func f2(arg int) (int, error) {
if arg == 42 {
|
В этом случае мы используем синтаксис &argError
для создания новой структуры, предоставляя
значения для двух полей arg и prob .
|
return -1, &argError{arg, "can't work with it"}
}
return arg + 3, nil
}
|
|
|
Два цикла ниже тестируют каждую из наших функций,
возвращающих ошибки. Обратите внимание, что
использование встроенной проверки ошибок в строке
if является обычный явлением в Go.
|
for _, i := range []int{7, 42} {
if r, e := f1(i); e != nil {
fmt.Println("f1 failed:", e)
} else {
fmt.Println("f1 worked:", r)
}
}
for _, i := range []int{7, 42} {
if r, e := f2(i); e != nil {
fmt.Println("f2 failed:", e)
} else {
fmt.Println("f2 worked:", r)
}
}
|
Если вы хотите использовать данные в пользовательской
ошибке, вам нужно получить ошибку как экземпляр
пользовательского типа через утверждение типа.
|
_, e := f2(42)
if ae, ok := e.(*argError); ok {
fmt.Println(ae.arg)
fmt.Println(ae.prob)
}
}
|