MySQL でストリーミングクエリを使用してデータ OOM を回避する

MySQL でストリーミングクエリを使用してデータ OOM を回避する

1. はじめに

プログラムがMySQLデータベースにアクセスするときに、クエリされるデータの量が特に大きい場合、データベース ドライバーは読み込まれたすべてのデータをメモリにロードし、メモリ オーバーフロー (OOM) が発生する可能性があります。

実際、 MySQLデータベースはストリーミング クエリを提供しており、これにより、修飾されたデータをバッチでメモリにロードできるため、OOM を効果的に回避できます。この記事では、主にストリーミング クエリの使用方法を紹介し、パフォーマンス テストで通常のクエリと比較します。

2. JDBCはストリーミングクエリを実装する

ストリーミング クエリは、JDBC のPreparedStatement/StatementsetFetchSizeメソッドをInteger.MIN_VALUEに設定するか、 Statement.enableStreamingResults()メソッドを使用することで実装できます。ResultSet.next ResultSet.next()メソッドが実行されると、データベース接続を通じて 1 つずつ返されるため、大量のクライアント メモリを占有することはありません。

パブリック int execute(String sql, boolean isStreamQuery) は SQLException をスローします {
 接続 conn = null;
 PreparedStatement stmt = null;
 結果セット rs = null;
 整数カウント = 0;
 試す {
  //データベース接続を取得します。conn = getConnection();
  if (isStreamQuery) {
   //ストリーミング クエリ パラメータを設定します。stmt = conn.prepareStatement(sql, ResultSet.TYPE_FORWARD_ONLY, ResultSet.CONCUR_READ_ONLY);
   stmt.setFetchSize(Integer.MIN_VALUE);
  } それ以外 {
   //通常のクエリ stmt = conn.prepareStatement(sql);
  }

  //クエリを実行して結果を取得します。rs = stmt.executeQuery();
  // 結果を走査します while (rs.next ()) {
   System.out.println(rs.getString(1));
   カウント++;
  }
 } キャッチ (SQLException e) {
  e.printStackTrace();
 ついに
  閉じる(stmt、rs、conn);
 }
 カウントを返します。
}

「PS」: 上記の例では、パラメーターisStreamQuery使用して、次のテスト比較で「ストリーミング クエリ」「通常のクエリ」を切り替えます。

3. パフォーマンステスト

テスト用にテストテーブルmy_testが作成されます。データの総量は27wです。テストには次の 4 つのテストケースが使用されます。

  • 大容量データ一般クエリ(270,000件)
  • 大容量データストリーミングクエリ(270,000レコード)
  • 小容量データ一般クエリ(10件)
  • 小容量データストリーミングクエリ(10項目)

3.1. 大容量データの一般クエリをテストする

@テスト
パブリック void testCommonBigData() は SQLException をスローします {
 文字列 sql = "select * from my_test";
 テスト実行(sql, false);
}

3.1.1. クエリ時間

27wのデータ量は38秒かかります

3.1.2. メモリ使用量

約1Gのメモリを使用

3.2. 大容量データストリーミングクエリのテスト

@テスト
パブリック void testStreamBigData() は SQLException をスローします {
 文字列 sql = "select * from my_test";
 テストを実行します(sql, true);
}

3.2.1. クエリ時間

27wのデータ量は37秒かかります

3.2.2. メモリ使用量

バッチで取得されるため、メモリは30〜270mの間で変動します。

3.3. 少量データに対する通常のクエリのテスト

@テスト
パブリック void testCommonSmallData() は SQLException をスローします {
 文字列 sql = "select * from my_test limit 100000, 10";
 テスト実行(sql, false);
}

3.3.1. クエリ時間

10個のデータには1秒かかります

3.4. 少量のデータでストリーミングクエリをテストする

@テスト
パブリック void testStreamSmallData() は SQLException をスローします {
 文字列 sql = "select * from my_test limit 100000, 10";
 テストを実行します(sql, true);
}

3.4.1. クエリ時間

10個のデータには1秒かかります

IV. 結論

MySQL ストリーミング クエリはメモリ使用量に明らかな最適化効果をもたらしますが、クエリ速度にはほとんど影響しません。主に、大量のデータをクエリするときにメモリ使用量が高くなるシナリオを解決するために使用されます。

「デモアドレス」: https://github.com/zlt2000/mysql-stream-query

これで、MySQL でストリーミング クエリを使用してデータ OOM を回避する方法に関するこの記事は終了です。MySQL ストリーミング クエリに関するより関連性の高いコンテンツについては、123WORDPRESS.COM で以前の記事を検索するか、次の関連記事を引き続き参照してください。今後とも 123WORDPRESS.COM をよろしくお願いいたします。

以下もご興味があるかもしれません:
  • MySQL OOM (メモリオーバーフロー) の解決策
  • MySQL スレーブが oom-killer をトリガーする問題の解決方法
  • MySQL OOM シリーズ 3: MySQL が殺されるという不運から逃れる
  • MySQL OOM システム 2 OOM キラー
  • MySQL OOM シリーズ 1 Linux メモリ割り当て

<<:  Docker環境を構築する簡単な方法

>>:  ログインボックスのメールプロンプトを実装するネイティブJS

推薦する

MySQL データベースで機密データの暗号化と復号化を実装する方法

目次1. 準備2. MySQL暗号化関数方式2.1 MySQL 暗号化2.2 MYSQL 復号化3....

Linux コマンドラインでパケットをキャプチャするために tcpdump を使用するいくつかの機能

tcpdump は、ネットワークの問題のトラブルシューティングに効果的に役立つ、柔軟で強力なパケット...

CSSはリモコンのボタンを模倣する

注: このデモはミニプログラム環境でテストされており、他の h5 および pc Web ページにも適...

MySQL で特定の日、月、または年のデータをクエリするためのコードの詳細な説明

今日 テーブル名から * を選択します。ここで、to_days(時間フィールド名) = to_day...

負のz-indexを持つ要素がクリックできない問題の解決策

最近、ポップアップ広告に取り組んでいました。デフォルト ページには z-index が設定されていな...

Pythonは出力をcsv操作に書き込む

以下のように表示されます。 def test_write(self): フィールド=[] field...

Nexusプライベートサーバー構築原理とチュートリアル分析

1つ。 Nexus プライベート サーバーを構築する理由は何ですか?社内の開発メンバーは全員外部ネッ...

JS配列メソッドsome、every、findの使用に関する詳細

目次1. いくつか2. すべての3. 見つける1. いくつかsome()メソッドは、指定された関数の...

Vue.jsはシンプルな折りたたみパネルを実装します

この記事では、Vue.jsの具体的なコードを共有して、シンプルな折りたたみパネルを実装する例を紹介し...

MySQL DMLステートメントの使用に関する詳細な説明

序文:前回の記事では、注意深い学生であれば発見できたかもしれない DDL ステートメントの使用法を中...

よくある CSS エラーと解決策

コードをコピーコードは次のとおりです。 IE6 と FF の違い: background:orang...

VMware 仮想マシンの 3 つのネットワーク方式と原則 (概要)

1. ブリッジ: デフォルトでは VMnet0 が使用されます1. 原則:ブリッジは、それぞれ 2...

Vueはキャンバスの手書き入力を使用して中国語を認識します

効果画像: 序文:最近、屋外の大画面プロジェクトに取り組んでいました。システムの入力方法は使いにくか...

Dockerでudpポート番号を指定する問題を解決する

Docker はコンテナを起動するときにアクセス ポートを指定します。複数の -p オプションを使用...