Errors

Errors #

// Go での慣習として、エラーを独立した返り値として明示的に返す。
// これは例外を使う Java や Ruby、結果とエラーを一つの値で返す C などの言語とは対照的である。
// Go のやり方だとどの関数がエラーを返すかがわかりやすく、またエラーのために特別な言語機能を増やす必要がない。

package main

import (
	"errors"
	"fmt"
)

// 慣習的に、最後の返り値を組み込みインターフェースである `error` 型の値にすることで、エラーの有無を返す。
func f1(arg int) (int, error) {
	if arg == 42 {

		// `errors.New` は基本的な `error` 型の値を作る関数である。引数はエラーメッセージを表す。
		return -1, errors.New("can't work with 42")

	}

	// error の値 `nil` はエラーが無かったことを表す。
	return arg + 3, nil
}

// `Error()` メソッドを実装すれば、自分で作った型を `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
}

func main() {

	// 以下の2つのループでは、上で実装したエラーを返す関数を試している。
	// `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)
	}
}
$ go run errors.go
f1 worked: 10
f1 failed: can't work with 42
f2 worked: 10
f2 failed: 42 - can't work with it
42
can't work with it

# エラー処理についてもっと知りたい人は、この[素晴らしいブログポスト](http://blog.golang.org/2011/07/error-handling-and-go.html)を読んでほしい。