カテゴリ: 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型として扱えます。エラーの種類を判別しやすくなります。

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

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

※ Amazon広告リンク

まとめ

まとめ
まとめ

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

生徒

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

先生

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

生徒

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

関連セミナーのご案内

【超入門】ゼロから始めるGo言語プログラミング:最速で「動くアプリ」を作るマンツーマン指導

「プログラミングの仕組み」が根本からわかる。Go言語でバックエンド開発の第一歩を。

本講座を受講することで、単なる文法の暗記ではなく、「プログラムがコンピュータの中でどう動いているか」という本質的な理解につながります。シンプルながら強力なGo言語(Golang)を通じて、現代のバックエンドエンジニアに求められる基礎体力を最短距離で身につけます。

具体的な開発内容と環境

【つくるもの】
ターミナル(黒い画面)上で動作する「対話型計算プログラム」や、データを整理して表示する「ミニ・ツール」をゼロから作成します。自分の書いたコードが形になる感動を体験してください。

【開発環境】
プロの現場でシェアNo.1のVisual Studio Code (VS Code)を使用します。インストールから日本語化、Go言語用の拡張機能設定まで、現場基準の環境を一緒に構築します。

この60分で得られる3つの理解

1. 環境構築の完全な理解

「なぜ動くのか」という設定の仕組みを理解し、今後の独学で詰まらない土台を作ります。

2. Go言語の基本構造(変数・型)

データの種類やメモリの概念など、他言語にも通じるプログラミングの本質を学びます。

3. 読みやすいコードの書き方

ただ動くだけでなく、誰が見ても分かりやすい「綺麗なコード」を書くための考え方を伝授します。

※本講座は、将来的にバックエンドエンジニアクラウドインフラに興味がある未経験者のためのエントリー講座です。マンツーマン形式により、あなたの理解度に合わせて進行します。

セミナー画像

初めてのGo言語を一緒に学びましょう!

関連記事:
カテゴリの一覧へ
新着記事
New1
Go言語
Go言語のwhile的なforループの使い方!条件式ループの基本を解説
New2
Go言語
Go言語プログラムの実行方法まとめ!VSCode・ターミナルでの実行手順を解説
New3
Swift
Swift意味とは?プログラミング言語・金融・鳥の違いを徹底解説
New4
Swift
Swift 戻り値の扱い方と複数戻り値の返し方|初心者でも分かる関数の基本
人気記事
No.1
Java&Spring記事人気No1
Go言語
Go言語でリダイレクト処理を行う方法(http.Redirect)を初心者向けに解説
No.2
Java&Spring記事人気No2
Swift
Swift開発環境の構築方法を徹底解説!Xcode・Windows・Linux対応
No.3
Java&Spring記事人気No3
Kotlin
Android Studioのインストール手順と初期設定を初心者向けに完全解説!
No.4
Java&Spring記事人気No4
Go言語
Go言語のgo.modファイル完全ガイド!初心者でもわかる仕組みと書き方
No.5
Java&Spring記事人気No5
Kotlin
Gradleファイル(build.gradle.kts)の書き方と役割をやさしく解説!Kotlin初心者向け完全ガイド
No.6
Java&Spring記事人気No6
Go言語
Go言語で条件式を1行で書くコツ!三項演算子の代替と短縮記法
No.7
Java&Spring記事人気No7
Swift
Swift Playgroundの使い方を完全解説!初心者に最適な学習環境の始め方
No.8
Java&Spring記事人気No8
Kotlin
Kotlinの演算子一覧と使い方!算術・比較・論理演算子の基本を解説