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

Go言語の並行処理でよくあるデータ競合と回避策を徹底解説!初心者でも理解できるgoroutineとchannelの安全な使い方

Go言語の並行処理でよくあるデータ競合と回避策
Go言語の並行処理でよくあるデータ競合と回避策

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

生徒

「先生、Go言語の並行処理で“データ競合”っていう言葉を聞いたんですけど、何のことなんですか?」

先生

「良いところに気づきましたね。Go言語では、複数の処理を同時に動かす“goroutine(ゴルーチン)”を使いますが、同じデータを同時に扱うと“データ競合”という問題が起こるんです。」

生徒

「なるほど…。データが競争してしまうってことですか?どうやって防ぐんですか?」

先生

「その通りです。競合を防ぐには、Goの“channel(チャネル)”や“sync.Mutex”といった機能を使う方法があります。それぞれを丁寧に説明していきましょう!」

1. データ競合(Race Condition)とは?

1. データ競合(Race Condition)とは?
1. データ競合(Race Condition)とは?

Go言語では、goroutineを使うことで複数の処理を同時に実行できます。これは「並行処理(concurrency)」と呼ばれ、効率的なプログラムを作るために非常に便利な仕組みです。

しかし、複数のgoroutineが同じ変数やメモリに同時アクセスすると、予期しない結果になることがあります。これが「データ競合(Race Condition)」です。

たとえば、同じ変数を同時に書き換えると、どちらの値が最終的に保存されるかわからなくなります。つまり、処理の順番が保証されないのです。

2. データ競合の具体例

2. データ競合の具体例
2. データ競合の具体例

まず、データ競合が起こる典型的な例を見てみましょう。


package main

import (
    "fmt"
    "time"
)

func main() {
    counter := 0

    for i := 0; i < 5; i++ {
        go func() {
            counter++
        }()
    }

    time.Sleep(time.Second)
    fmt.Println("最終結果:", counter)
}

このプログラムでは、5つのgoroutineが同じ変数counterに同時にアクセスして値を増やしています。一見正しく動くように見えますが、実行するたびに結果が変わる可能性があります。これは、複数のgoroutineが同時にcounterを書き換えているためです。


最終結果: 3

上のように、期待している「5」ではなく「3」などになることがあります。これがデータ競合です。

3. Goでデータ競合を検出する方法

3. Goでデータ競合を検出する方法
3. Goでデータ競合を検出する方法

Goでは、race detector(レース検出ツール)を使って簡単にデータ競合を確認できます。実行時に次のようにコマンドを付けるだけです。


go run -race main.go

これを使うと、どの変数で競合が起きているかを詳しく教えてくれます。開発中は必ずこの機能でテストするのがおすすめです。

4. channelを使ったデータ競合の回避

4. channelを使ったデータ競合の回避
4. channelを使ったデータ競合の回避

Goでは、channel(チャネル)を使うことで、goroutine間の安全なデータの受け渡しができます。channelは“データの通り道”のようなもので、ひとつのgoroutineが送信し、もう一方が受信します。


package main

import (
    "fmt"
)

func main() {
    ch := make(chan int)
    counter := 0

    for i := 0; i < 5; i++ {
        go func() {
            ch <- 1
        }()
    }

    for i := 0; i < 5; i++ {
        counter += <-ch
    }

    fmt.Println("安全な結果:", counter)
}

このようにchannelを使うと、データは順番にやり取りされるため、同時アクセスによる競合が起きません。結果も常に正しい「5」が出力されます。


安全な結果: 5

5. sync.Mutexによるデータ保護

5. sync.Mutexによるデータ保護
5. sync.Mutexによるデータ保護

もう1つの代表的な回避策が、sync.Mutex(ミューテックス)です。これは、「この変数を今は私が使っています。他の人は待ってね!」という“鍵”のような役割をします。


package main

import (
    "fmt"
    "sync"
    "time"
)

func main() {
    var mu sync.Mutex
    counter := 0

    for i := 0; i < 5; i++ {
        go func() {
            mu.Lock()
            counter++
            mu.Unlock()
        }()
    }

    time.Sleep(time.Second)
    fmt.Println("Mutexで保護した結果:", counter)
}

このようにmu.Lock()でデータをロックし、mu.Unlock()で開放することで、同時書き込みを防ぎます。結果は必ず「5」になります。

6. channelとMutexの使い分け

6. channelとMutexの使い分け
6. channelとMutexの使い分け

「channel」と「Mutex」はどちらもデータ競合を防ぐための仕組みですが、目的が少し異なります。

  • channel:データの受け渡しを安全に行いたいとき
  • Mutex:同じデータを一時的にロックして処理したいとき

例えば、goroutine同士で「値のやり取り」をしたいならchannelを使い、「同じ変数を操作したい」ならMutexを使うと覚えるとわかりやすいです。

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

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

※ Amazon広告リンク

7. データ競合を防ぐためのポイントまとめ

7. データ競合を防ぐためのポイントまとめ
7. データ競合を防ぐためのポイントまとめ
  • 複数のgoroutineが同じ変数を扱うときは注意!
  • go run -raceで必ず競合をチェック
  • channelで安全にデータを受け渡し
  • Mutexで共有データをロックして保護
  • “どんな処理を同時に走らせたいか”を意識して設計する

Go言語の並行処理はとても強力ですが、安全に使うためには「データ競合」を理解し、適切に防ぐことが大切です。初心者のうちは、まずはchannelでgoroutineの連携を練習すると良いでしょう。

関連セミナーのご案内

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

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

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

具体的な開発内容と環境

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

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

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

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

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

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

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

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

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

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

セミナー画像

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

カテゴリの一覧へ
新着記事
New1
Swift
Swiftのnilとは?Optionalとの関係や初期化について初心者向けにやさしく解説!
New2
Go言語
Go言語のinit関数の役割と使い方!プログラム起動時の初期化処理
New3
Go言語
Go言語のマップの順序保証がない理由と扱い方の工夫をやさしく解説!初心者でもわかる基本知識
New4
Kotlin
Kotlinのforループの基本!範囲・配列・コレクションの繰り返し処理
人気記事
No.1
Java&Spring記事人気No1
Kotlin
Kotlinのsettings.gradleファイルを完全解説!初心者でもわかるプロジェクト設定の基本
No.2
Java&Spring記事人気No2
Go言語
Swiftの配列(Array)の使い方を完全ガイド!初心者でもわかるデータのまとめ方
No.3
Java&Spring記事人気No3
Go言語
Go言語のSQLインジェクション対策を完全解説!初心者でも安全なデータベース操作がわかる
No.4
Java&Spring記事人気No4
Swift
Swiftの高階関数map・filter・reduceを完全解説!初心者でもわかる配列操作の基本
No.5
Java&Spring記事人気No5
Go言語
Swiftの文字列操作を完全ガイド!初心者でもわかるStringの基本
No.6
Java&Spring記事人気No6
Go言語
Go言語のgo installコマンドの役割とインストール先の仕組みを徹底解説!
No.7
Java&Spring記事人気No7
Kotlin
KotlinのRoomで複雑なクエリを使いこなす!初心者でもわかる応用テクニック
No.8
Java&Spring記事人気No8
Swift
Swift Playgroundの使い方を完全解説!初心者に最適な学習環境の始め方