ホームページ バックエンド開発 Golang Go 言語 Websocket アプリケーションの同時実行セキュリティ問題を解決する

Go 言語 Websocket アプリケーションの同時実行セキュリティ問題を解決する

Dec 14, 2023 pm 01:47 PM
言語を移動 websocket 同時実行の安全性

Go 言語 Websocket アプリケーションの同時実行セキュリティ問題を解決する

WebSocket は、強力なリアルタイム パフォーマンスを備えた双方向通信を実現できる最新のネットワーク通信プロトコルです。 Go 言語は本質的に同時実行性をサポートしているため、Websocket アプリケーションで非常に優れたパフォーマンスを発揮します。ただし、同時実行にはいくつかの問題も伴い、Websocket アプリケーションでは、これは主に同時実行のセキュリティに反映されます。この記事では、Go Websocket アプリケーションの同時実行セキュリティの問題を解決する方法を説明し、デモンストレーションします。

  1. 問題の背景

Websocket アプリケーションでは、クライアントはいつでもサーバーにメッセージを送信でき、サーバーもいつでもクライアントにメッセージを送信できます。 。したがって、Websocket メッセージを処理するときは、同時実行の問題を考慮する必要があります。 Go 言語では、Goroutine を使用して WebSocket メッセージを同時に処理できます。

ただし、同時実行により、競合状態やデッドロックなど、同時実行のセキュリティ上の問題が発生することがあります。競合状態によりデータの不整合が発生したり、デッドロックによりプログラムがフリーズしたりすることがあります。したがって、Websocket アプリケーションでは、これらの同時実行セキュリティの問題を解決する必要があります。

  1. 解決策

2.1 ミューテックス ロック

ミューテックス ロックは、Go 言語で最も一般的な同時実行制御メカニズムの 1 つです。共有リソースを保護し、複数のゴルーチンが同時に共有リソースにアクセスすることを防ぎ、データの正確性と一貫性を保証します。

Websocket アプリケーションでは、ミューテックス ロックを通じて共有リソースの同時実行セキュリティを確保できます。たとえば、次のコードは、ミューテックス ロックを使用して、共有マップに同時に書き込む複数のゴルーチンの安全性を確保する方法を示しています。

type safeMap struct {
    m map[string]int
    sync.Mutex
}

func (sm *safeMap) Inc(key string) {
    sm.Lock()
    sm.m[key]++
    sm.Unlock()
}
ログイン後にコピー

この例では、構造体に sync.Mutex を埋め込みます。共有リソースを保護するための、safeMap タイプのミューテックス。この構造では、複数のゴルーチンによって共有されるリソースを表すマップ型変数 m を定義します。次に、マップ内のデータに対して自動インクリメント操作を実行するために、safeMap のメソッド Inc を定義しました。メソッド Inc では、最初にロックし、次にインクリメント操作を実行し、最後にロックを解除します。

2.2 ロックフリーの同時実行性

同時実行のセキュリティ問題を解決するもう 1 つの方法は、ロックフリーの同時実行性を使用することです。ロックフリーの同時実行では、非ブロッキング アルゴリズムを使用することで、ミューテックス ロックによって引き起こされるパフォーマンスの損失を回避します。システムの同時実行性とスループットを向上させることができ、高性能、低遅延、高スループットのシステムでよく使用されます。

Go 言語では、sync/atomic パッケージのアトミック操作関数を使用して、ロックフリーの同時実行性を実現できます。たとえば、次のコードは、アトミック操作を使用して共有変数に対する同時操作を実装する方法を示しています。

type Counter struct {
    count int32
}

func (c *Counter) Inc() {
    atomic.AddInt32(&c.count, 1)
}

func (c *Counter) Dec() {
    atomic.AddInt32(&c.count, -1)
}

func (c *Counter) Get() int32 {
    return atomic.LoadInt32(&c.count)
}
ログイン後にコピー

この例では、アトミック パッケージの AddInt32 関数と LoadInt32 関数を使用してカウンターを実装します。 int32 型の count 変数を含む構造体 Counter を定義します。 Counter 構造体は、Inc、Dec、Get という 3 つのメソッドも実装します。 Inc メソッドと Dec メソッドでは、アトミック操作 AddInt32 を使用して、共有変数のカウントをインクリメントおよびデクリメントします。 Get メソッドでは、アトミック操作 LoadInt32 を使用して共有変数 count の値を取得します。

  1. サンプル コード

次に、同時実行の安全性を確保するためにミューテックス ロックを使用する Websocket アプリケーションのサンプル コードを示します。

package main

import (
    "fmt"
    "net/http"
    "sync"

    "github.com/gorilla/websocket"
)

var upgrader = websocket.Upgrader{
    ReadBufferSize:  1024,
    WriteBufferSize: 1024,
}

type Connection struct {
    ws *websocket.Conn
    mu sync.Mutex
}

func main() {
    http.HandleFunc("/", handler)
    http.ListenAndServe(":8080", nil)
}

func handler(w http.ResponseWriter, r *http.Request) {
    c, err := upgrader.Upgrade(w, r, nil)
    if err != nil {
        fmt.Println(err)
        return
    }

    conn := &Connection{ws: c}

    go conn.WriteLoop()
    conn.ReadLoop()
}

func (conn *Connection) ReadLoop() {
    defer conn.ws.Close()
    for {
        _, message, err := conn.ws.ReadMessage()
        if err != nil {
            fmt.Println(err)
            break
        }

        fmt.Printf("Received message: %s
", message)
    }
}

func (conn *Connection) WriteLoop() {
    defer conn.ws.Close()
    for {
        conn.mu.Lock()
        err := conn.ws.WriteMessage(websocket.TextMessage, []byte("Hello, world!"))
        conn.mu.Unlock()
        if err != nil {
            fmt.Println(err)
            break
        }
    }
}
ログイン後にコピー

この例では、私たちは単純な Websocket アプリケーションを実装しました。これには、クライアント メッセージを読み取る ReadLoop と、クライアントにメッセージを送信する WriteLoop が含まれています。このアプリケーションでは、各クライアントの接続を Connection 構造にカプセル化し、sync.Mutex タイプの mutex mu を埋め込みます。 WriteLoop でこのミューテックス ロックを使用して、共有リソース conn.ws の同時実行の安全性を確保します。ミューテックス ロックを使用すると、複数のゴルーチンが同じ Websocket 接続に同時にデータを書き込む問題を回避できます。

以下は、アトミック操作を使用してロックフリーの同時実行性を実現する Websocket アプリケーションのサンプル コードです:

package main

import (
    "fmt"
    "net/http"
    "sync/atomic"

    "github.com/gorilla/websocket"
)

var upgrader = websocket.Upgrader{
    ReadBufferSize:  1024,
    WriteBufferSize: 1024,
}

type Connection struct {
    ws    *websocket.Conn
    count int32
}

func main() {
    http.HandleFunc("/", handler)
    http.ListenAndServe(":8080", nil)
}

func handler(w http.ResponseWriter, r *http.Request) {
    c, err := upgrader.Upgrade(w, r, nil)
    if err != nil {
        fmt.Println(err)
        return
    }

    conn := &Connection{ws: c}

    go conn.WriteLoop()
    conn.ReadLoop()
}

func (conn *Connection) ReadLoop() {
    defer conn.ws.Close()
    for {
        _, message, err := conn.ws.ReadMessage()
        if err != nil {
            fmt.Println(err)
            break
        }

        fmt.Printf("Received message: %s
", message)
    }
}

func (conn *Connection) WriteLoop() {
    defer conn.ws.Close()
    for {
        n := atomic.AddInt32(&conn.count, 1)
        if n > 10 {
            break
        }

        err := conn.ws.WriteMessage(websocket.TextMessage, []byte("Hello, world!"))
        if err != nil {
            fmt.Println(err)
            break
        }
    }
}
ログイン後にコピー

この例では、アトミック パッケージの AddInt32 関数と LoadInt32 関数を使用して、カウンターを実装します。 int32 型の count 変数を含む構造体 Connection を定義します。 Connection 構造体は、ReadLoop と WriteLoop という 2 つのメソッドも実装します。 WriteLoop メソッドでは、アトミック操作 AddInt32 を使用して、共有変数の数をインクリメントします。次に、カウンター値が 10 を超えているかどうかを判断し、超えている場合はループを終了します。この例では、ミューテックスを使用する代わりに、アトミック操作を使用してロックフリーの同時実行性を実現します。

    #結論
この記事では、Go 言語の Websocket アプリケーションにおける同時実行セキュリティの問題を解決する方法を紹介します。ミューテックス ロックとロックフリーの同時実行という 2 つの解決策を提供します。ミューテックス ロックであっても、ロックなしの同時実行であっても、同時実行のセキュリティは保証されます。どの方法を選択するかは、特定のアプリケーションのシナリオと要件によって異なります。これらのテクノロジの使用方法を特定のサンプル コードを通じて示し、読者がこれらのテクノロジをよりよく理解して適用できるようにしたいと考えています。

以上がGo 言語 Websocket アプリケーションの同時実行セキュリティ問題を解決するの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

Go's Crawler Collyのキュースレッドの問題は何ですか? Go's Crawler Collyのキュースレッドの問題は何ですか? Apr 02, 2025 pm 02:09 PM

Go Crawler Collyのキュースレッドの問題は、Go言語でColly Crawler Libraryを使用する問題を調査します。 �...

GOの浮動小数点番号操作に使用されるライブラリは何ですか? GOの浮動小数点番号操作に使用されるライブラリは何ですか? Apr 02, 2025 pm 02:06 PM

GO言語の浮動小数点数操作に使用されるライブラリは、精度を確保する方法を紹介します...

GoおよびViperライブラリを使用するときにポインターを渡す必要があるのはなぜですか? GoおよびViperライブラリを使用するときにポインターを渡す必要があるのはなぜですか? Apr 02, 2025 pm 04:00 PM

ポインター構文とviperライブラリの使用における問題への取り組みGO言語でプログラミングするとき、特にポインターの構文と使用を理解することが重要です...

GOのどのライブラリが大企業によって開発されていますか、それとも有名なオープンソースプロジェクトによって提供されていますか? GOのどのライブラリが大企業によって開発されていますか、それとも有名なオープンソースプロジェクトによって提供されていますか? Apr 02, 2025 pm 04:12 PM

大企業または有名なオープンソースプロジェクトによって開発されたGOのどのライブラリが開発されていますか? GOでプログラミングするとき、開発者はしばしばいくつかの一般的なニーズに遭遇します...

Golandのカスタム構造ラベルが有効になっていないという問題を解決する方法は? Golandのカスタム構造ラベルが有効になっていないという問題を解決する方法は? Apr 02, 2025 pm 12:51 PM

GoLandを使用する場合のGolandのカスタム構造タグの問題に関して、Go Language Developmentに使用する場合、いくつかの構成の問題に遭遇することがよくあります。それらの1つは...

Goでは、Printlnとstring()関数を備えた文字列を印刷すると、なぜ異なる効果があるのですか? Goでは、Printlnとstring()関数を備えた文字列を印刷すると、なぜ異なる効果があるのですか? Apr 02, 2025 pm 02:03 PM

Go言語での文字列印刷の違い:printlnとstring()関数を使用する効果の違いはGOにあります...

言語スライスに行く:シングルエレメントスライスインデックス1インターセプトの場合、なぜエラーを報告しないのですか? 言語スライスに行く:シングルエレメントスライスインデックス1インターセプトの場合、なぜエラーを報告しないのですか? Apr 02, 2025 pm 02:24 PM

Go Language Slice Index:エラーなしでインデックス1からシングルエレメントスライスインターセプトがインターセプトされるのはなぜですか? GO言語では、スライスは底部を参照できる柔軟なデータ構造です...

GolangのLinux Iptablesリンクリストに操作を実装する方法は? GolangのLinux Iptablesリンクリストに操作を実装する方法は? Apr 02, 2025 am 10:18 AM

Golangを使用してLinuxを実装しています...

See all articles