カテゴリ: Go言語 更新日: 2025/12/10

Go言語の構造体でのエラー処理の書き方と工夫例

Go言語の構造体でのエラー処理の書き方と工夫例
Go言語の構造体でのエラー処理の書き方と工夫例

先生と生徒の会話形式で理解しよう

生徒

「Go言語の構造体を使うときに、エラーが出たらどうやって処理すればいいんですか?」

先生

「Goでは関数が戻り値としてエラー(error型)を返す仕組みがあります。構造体のメソッドでも同じです。わかりやすく説明しますね。」

生徒

「errorって何ですか?どう使うんでしょう?」

先生

「error型は「何か問題が起こりましたよ」という合図です。では構造体でエラー処理をする例を見ていきましょう。」

1. 構造体メソッドでのエラー処理基本形

1. 構造体メソッドでのエラー処理基本形
1. 構造体メソッドでのエラー処理基本形

構造体とメソッドを使って、エラーを返す基本パターンをご紹介します。error型は組み込みの型で、問題がなければnil(「何もない」)が返ります。


type FileLoader struct {
    Path string
}

func (f *FileLoader) Load() (string, error) {
    if f.Path == "" {
        return \"\", fmt.Errorf(\"ファイルパスが空です\")
    }
    // 実際の読み込み処理...
    return \"ファイル内容\", nil
}

このようなコードではPathが空文字の場合、fmt.Errorfでエラーを作って返します。戻り値は文字列とerrorの二つです。

2. 呼び出し側でエラーを確認する

2. 呼び出し側でエラーを確認する
2. 呼び出し側でエラーを確認する

エラーを返すメソッドを呼び出すときは、必ず戻り値のerrorをチェックします。


loader := &FileLoader{Path: \"\"}
data, err := loader.Load()
if err != nil {
    fmt.Println(\"エラー発生:\", err)
    return
}
fmt.Println(data)

このようにエラーをチェックすることで、安全にプログラムを進められます。

3. カスタムエラー型でわかりやすく

3. カスタムエラー型でわかりやすく
3. カスタムエラー型でわかりやすく

もっと詳しく原因を伝えたい場合は、自分でエラーの型を作ることもできます。


type PathError struct {
    Path string
    Msg  string
}

func (e *PathError) Error() string {
    return fmt.Sprintf(\"%s: %s\", e.Path, e.Msg)
}

func (f *FileLoader) Load() (string, error) {
    if f.Path == \"\" {
        return \"\", &PathError{Path: f.Path, Msg: \"パスが指定されていません\"}
    }
    return \"データ\", nil
}

このカスタムエラーはError()メソッドを持つことでerror型として扱えます。エラーの種類を判別しやすくなります。

Go言語を基礎からスッキリ学びたい人や、 文法だけでなく「実用的な使い方」まで押さえたい人には、 定番の入門書がこちらです。

基礎からわかるGo言語をAmazonで見る

※ Amazon広告リンク

4. エラーの型判定で細かく分岐

4. エラーの型判定で細かく分岐
4. エラーの型判定で細かく分岐

呼び出し側でエラーの型を判定して、原因に応じた処理をすることができます。


data, err := loader.Load()
if err != nil {
    if pe, ok := err.(*PathError); ok {
        fmt.Println(\"パスエラーです:\", pe.Path)
    } else {
        fmt.Println(\"その他のエラー:\", err)
    }
    return
}

このように型アサーションを使うと、エラーによって処理を切り替えられます。

5. エラーに追加情報を重ねる(ラップ)

5. エラーに追加情報を重ねる(ラップ)
5. エラーに追加情報を重ねる(ラップ)

Go1.13以降ではfmt.Errorf「%w」というフォーマット指定子を使ってエラーをラップできます。


func (f *FileLoader) Load() (string, error) {
    if f.Path == \"\" {
        return \"\", fmt.Errorf(\"Load失敗: %w\", &PathError{Path: f.Path, Msg: \"空のパス\"})
    }
    return \"データ\", nil
}

こうすることで、元のエラーを保持しながら「Load失敗」という追加情報も伝えられます。

6. errors.Is/errors.Asで原因を探る

6. errors.Is/errors.Asで原因を探る
6. errors.Is/errors.Asで原因を探る

エラーがラップされていてもerrors.Iserrors.Asを使えば、中にある元のエラーを検出できます。


if errors.Is(err, &PathError{}) {
    fmt.Println(\"PathErrorが含まれています。原因:\", err)
}
var pe *PathError
if errors.As(err, &pe) {
    fmt.Println(\"具体的なパス:\", pe.Path)
}

このようにエラーの原因を詳細に確認できます。

7. ポイント整理

7. ポイント整理
7. ポイント整理
  • 構造体メソッドでerrorを返す習慣をつける
  • 呼び出し側で必ずerrorをチェックする
  • カスタムエラー型で原因を明確にする
  • ラップして情報を追記しつつ、元の原因も保持
  • errors.Iserrors.As で原因を判定

これらの工夫を組み合わせることで、Go言語の構造体を使ったコードの中でも、エラー処理が分かりやすく、メンテナンスしやすくなります。

まとめ

まとめ
まとめ

Go言語における構造体とエラー処理の仕組みを振り返ると、基本的なポイントは「関数やメソッドが error を戻り値として返し、その値を呼び出し側で丁寧に確認する」という、とてもシンプルでありながら強力な考え方にあります。構造体を用いたプログラムでは、さまざまな状況で異なるエラーが起こり得るため、どのようにエラーを表現し、どのように扱うかがコードの読みやすさや保守性に直結します。とくに、カスタムエラー型を用いた詳細なエラー情報の付与、ラップによるエラーの積み重ね、errors.Is と errors.As を活用した原因追跡などは、Go言語で堅牢なプログラムを書くうえで理解しておきたい要点です。

また、構造体メソッドにおけるエラー処理は、単にエラーを返すだけでなく、「どの構造体のどのフィールドが原因なのか」を分かりやすく伝える工夫が重要です。実際に PathError のようなカスタム型を導入することで、エラーの背景をひと目で把握でき、呼び出し側でも原因に応じた対処が可能になります。これは規模が大きくなるほど効果を発揮し、チーム開発の場面でも非常に役立つ考え方となります。

さらに、Go1.13 以降で導入されたエラーラップの仕組みを使えば、ひとつのエラーに複数の文脈を重ねることができ、ログ解析やデバッグ時にも強力なヒントとなります。エラーを単なる「失敗の通知」として扱うのではなく、「問題を正確に伝えるメッセージ」として整理して扱うことで、全体のコード品質が大きく向上します。

最後に、ここまで学んだ内容はすべて実際のプログラミングで頻繁に登場する要素ばかりです。構造体・メソッド・エラー処理は Go 言語の中核的な機能であり、これらを組み合わせて設計することで、安全で読みやすく、拡張性のあるプログラムを作成できるようになります。以下に、本記事の総まとめとして簡単なサンプルコードを再掲し、理解を深められるよう整理しておきます。

総まとめサンプル:構造体とエラー処理の基本パターン


type FileLoader struct {
    Path string
}

func (f *FileLoader) Load() (string, error) {
    if f.Path == "" {
        return "", fmt.Errorf("パスが空です")
    }
    return "ファイル内容", nil
}

func main() {
    loader := &FileLoader{Path: ""}
    data, err := loader.Load()
    if err != nil {
        fmt.Println("エラー発生:", err)
        return
    }
    fmt.Println("読み込み結果:", data)
}

このような基本形を理解したうえで、カスタムエラー型やラップ処理などを活用すると、より柔軟で扱いやすいエラー処理が可能になります。今後 Go 言語で構造体を扱うプログラムを書く際には、今回学んだポイントを意識しながら実装してみるとよいでしょう。

先生と生徒の振り返り会話

生徒

「先生、Go の構造体とエラー処理って、思っていたよりも奥が深いんですね!ただのエラーじゃなくて、原因を詳しく伝える工夫がたくさんできるのが驚きでした。」

先生

「そうですね。Go はシンプルな文法なのに、エラー処理の設計がとても強力なんです。特に、構造体と組み合わせるとデータの状態とエラーの関係を分かりやすく表現できます。」

生徒

「カスタムエラー型やラップを使うと、原因がすぐに分かるようになるのが便利だと思いました。大きいプログラムでも役に立ちそうです。」

先生

「その通りです。今回学んだテクニックを活用することで、エラーをただの問題ではなく“改善につながる情報”として扱えるようになりますよ。今後書くコードの質も大きく変わるはずです。」

生徒

「はい!構造体の設計もエラー処理の工夫も、しっかり使いこなせるように練習してみます!」

関連記事:
カテゴリの一覧へ
新着記事
New1
Go言語
Go言語の短絡評価(ショートサーキット)を使った条件式の工夫をやさしく解説!初心者でも理解できる基本知識
New2
Go言語
Go言語の構造体の初期化パターンとコンストラクタ的関数の書き方を徹底解説!初心者でもわかる基本と実用例
New3
Kotlin
Kotlinの例外処理とキャンセルの連携を完全ガイド!初心者でもわかるCoroutineExceptionHandlerの使い方
New4
Go言語
Go言語のクロージャとは?関数内関数の活用例と仕組み
人気記事
No.1
Java&Spring記事人気No1
Kotlin
KotlinのChannelでデータをやり取りする方法を完全ガイド!初心者にもわかる非同期通信の基本
No.2
Java&Spring記事人気No2
Go言語
Go言語でのDB接続情報を環境変数で管理する方法|初心者でも安全に設定
No.3
Java&Spring記事人気No3
Kotlin
Kotlinのビルド設定エラーと解決法まとめ!初心者向けGradleトラブル対処ガイド
No.4
Java&Spring記事人気No4
Kotlin
Android Studioのインストール手順と初期設定を初心者向けに完全解説!
No.5
Java&Spring記事人気No5
Kotlin
Gradleファイル(build.gradle.kts)の書き方と役割をやさしく解説!Kotlin初心者向け完全ガイド
No.6
Java&Spring記事人気No6
Kotlin
Kotlinでテキスト表示・編集!初心者でもわかるTextViewとEditTextの使い方
No.7
Java&Spring記事人気No7
Kotlin
Kotlin DSLとGroovy DSLの違いをやさしく解説!初心者でもわかるGradleスクリプトの使い分け
No.8
Java&Spring記事人気No8
Go言語
Go言語のgo.modファイル完全ガイド!初心者でもわかる仕組みと書き方

💻 作業効率アップに

ノートPCを縦置きしてデスクを広く。
省スペースで片づく定番スタンド

UGREEN 縦型スタンドをAmazonで見る

※ Amazon広告リンク