フックを使用して JVM を正常にシャットダウンするにはどうすればよいですか?

フックを使用して JVM を正常にシャットダウンするにはどうすればよいですか?

[[390011]]

序文

1. 基本的な概要

プログラムの起動は非常に簡単で、通常は起動時にいくつかのリソースの事前ロードが実行されます。ただし、シャットダウン時に、プリロードされたリソースが起動時に完全にクリーンアップされないことがあるため、フック関数を使用してこれを完了することができます。

2. JVMシャットダウンシナリオの分類

写真を見てみましょう。この写真は CSDN ブログの BarryWang から提供されたもので、ここで説明したいと思います。

上記からわかるように、JVM のシャットダウンは主に 3 つのカテゴリに分かれており、1 つ目は通常のシャットダウン、2 つ目は異常なシャットダウン、3 つ目は強制シャットダウンです。最初の 2 つの方法では、フック関数を使用して正常に閉じることができますが、強制的に閉じる場合はフック関数は機能しません。

これらの概念を念頭に置いて、デモンストレーション用のケースを直接使用し、分析してみましょう。

1. コードデモンストレーションフック関数

1. JVMが正常にシャットダウンする

コードを直接見てみましょう。

  1. パブリッククラステスト{
  2. パブリックボイドスタート(){
  3. Runtime.getRuntime().addShutdownHook(新しいスレッド(()->
  4. System.out.println ( "フック関数が実行されました。ここでリソースを閉じることができます" )
  5. ));
  6. }
  7. 公共 静的void main(String[] args)は例外をスローします{
  8. 新しいテスト().start();
  9. System.out.println ( "メインアプリケーションが実行中です" ) ;
  10. }
  11. }
  12. //コンソール出力
  13. //メインアプリケーションが実行中
  14. //フック関数が実行され、ここでリソースを閉じることができます

コンソール出力を見ると、メインアプリケーションの実行後にフック関数が呼び出され、その後 JVM が正式にシャットダウンされることがわかります。

2. 異常閉鎖

コードのデモを直接見てみましょう。ここでは、異常終了の 2 番目の OOM 状況を示します。まずヒープ サイズを 20M に設定し、コード内に 500M のオブジェクトを作成すると、OOM が発生します。パラメータは -Xmx20M です。

  1. パブリッククラステスト{
  2. パブリックボイドスタート(){
  3. Runtime.getRuntime().addShutdownHook(新しいスレッド(()->
  4. System.out.println ( "フック関数が実行されました。ここでリソースを閉じることができます" )
  5. ));
  6. }
  7. 公共 静的void main(String[] args)は例外をスローします{
  8. 新しいテスト().start();
  9. System.out.println ( "メインアプリケーションが実行中です" ) ;
  10. ランタイムを取得します。
  11. バイト[] b = 新しいバイト[500*1024*1024];
  12. }
  13. }
  14. //コンソール出力
  15. //メインアプリケーションが実行中
  16. //フック関数が実行され、ここでリソースを閉じることができます

コンソールから、異常終了した場合でもフック関数が呼び出されることがわかります。

3. 強制終了

ここでは、Runtime.getRuntime().halt() を使用して、強力なシャットダウンを示します。このメソッドと System.exit の違いは、System.exit はフック関数を実行しますが、Runtime.getRuntime().halt() は実行しないことです。

  1. パブリッククラステスト{
  2. パブリックボイドスタート(){
  3. Runtime.getRuntime().addShutdownHook(新しいスレッド(()->
  4. System.out.println ( "フック関数が実行されました。ここでリソースを閉じることができます" )
  5. ));
  6. }
  7. 公共 静的void main(String[] args)は例外をスローします{
  8. 新しいテスト().start();
  9. System.out.println ( "メインアプリケーションが実行中です" ) ;
  10. ランタイムを取得します。
  11. }
  12. }
  13. //コンソール出力
  14. //メインアプリケーションが実行中

上記のコードの出力から、Runtime.getRuntime().halt(1) を呼び出すと JVM が強制的にシャットダウンされ、フック関数は実行される前にシャットダウンされることがわかります。ただし、System.exit を使用すると、引き続き実行されます。したがって、JVM をシャットダウンするには通常、System.exit が使用されます。

4.フック機能を削除する

上記はフック関数の役割を示しています。簡単に取り外せる場合もあります。

  1. パブリッククラステスト{
  2. 公共 静的void main(String[] args)は例外をスローします{
  3. //新しいTest().start();
  4. スレッド willNotRun = new Thread(() ->
  5. システム。 out .println( "実行されません!" ));
  6. Runtime.getRuntime().addShutdownHook(willNotRun);
  7. System.out.println ( "メインアプリケーションが実行中です" ) ;
  8. Runtime.getRuntime().removeShutdownHook(willNotRun);
  9. }
  10. }
  11. //コンソール出力
  12. //メインアプリケーションが実行中

さて、フック関数の基本的な動作はここに書かれています。使い方は比較的簡単です。しかし、以前に Spring の起動プロセスを見たことがあるのですが、その起動プロセスをもう一度見てみると、フック関数も使用されていることがわかりました。

2. 典型的なアプリケーションシナリオ

1. Springを使用する

Spring がコンテキストを閉じるときに、フック関数を使用して残りのリソースを閉じることができます。この方法は、ApplicationContext を使用してフック関数を登録することです。

  1. ApplicationContext.registerShutdownHook();
  2. //上記のコードは分析して見ることができます
  3. パブリックvoid registerShutdownHook() {
  4. this.shutdownHook がnull場合
  5. this.shutdownHook = 新しいスレッド() {
  6. @オーバーライド
  7. パブリックボイド実行(){
  8. //スプリングは正常に閉じます
  9. クローズ();
  10. }
  11. };
  12. //残りのリソースを閉じるためのフック関数を呼び出す
  13. Runtime.getRuntime().addShutdownHook(this.shutdownHook);
  14. }
  15. }

ソース コードからわかるように、Spring は実際にシャットダウンするために Java のフック関数を呼び出します。

2. その他の用途

多くのブログで、Spark と Hadoop のシャットダウンも目にしました。ソースコードを読んでいないので、ここでは結論を述べます。その他の使用シナリオでは、基本的に Java フック関数が呼び出されて実行されます。

結論は

JVM をシャットダウンするときに、フック関数をカプセル化してスレッドを正常にシャットダウンすることができます。ただし、使用時には以下の点に注意する必要があります。

1. フック関数は本質的にはスレッドである

複数のフックが同時に実行され、JVM はそれらの実行順序を保証しません。したがって、一連の操作を 1 つのフックで実行するのが最適です。

2. フック内に新しいフックを作成できなくなりました

シャットダウン フックでは、フックの登録や削除などの操作を実行することはできません。そうしないと、JVM によって IllegalStateException がスローされます。 System.exit() も使用できません。前述のように、System.exit() はフック関数の実行をトリガーしますが、Runtime.halt() はシャットダウンを強制できるため、Runtime.halt() でも実行できます。

3. フックには時間のかかる操作を入れないのがベスト

フック関数は主に残りのリソースを閉じるために使用されるため、時間のかかる操作は実行しないでください。

はい、とりあえずここでやめておきます。

この記事はWeChat公式アカウント「Yugong Yaoyishan」から転載したものです。下のQRコードからフォローできます。この記事を転載する場合は、Yugongyaoyishanの公式アカウントまでご連絡ください。

<<:  新しいテクノロジートレンド: SaaS セキュリティ構成管理 (SSPM)

>>:  4 Windows 仮想デスクトップ管理の制限

推薦する

武漢クラウドは世界に輝く真珠を育てるデジタル基盤を構築

11月16日から17日にかけて、スマートシティ分野の「オスカー」として知られるグローバルスマートシテ...

地域人材ネットワークの運営に関する考察

おそらく、ローカル人材ネットワークの運営はすでに飽和状態にあるが、競争があるからこそ、利益ポイントが...

#五一# edgenat: 全品 30% オフ、最低 35 元、韓国 VPS\香港 VPS\米国 VPS、すべてのハイエンド回線/無制限トラフィック、香港\韓国専用サーバー

edgenatは、5月10日まで毎年恒例のメーデーゴールデンウィークイベントを開始しました。具体的な...

hostslayer - $6/2G メモリ/40g SSD/1Gbps/無制限トラフィック/ニューヨーク choopa

HostSlim BVは2月1日、正式にhostslayerを買収したことを発表しました。Hosts...

テンセントクラウドの6つのエッジアベイラビリティゾーンは同日に開始され、新しいインフラストラクチャのレイアウトが引き続き加速している。

テンセントクラウドは12月17日、武漢、杭州、長沙、福州、済南、石家荘の6つの省都に位置するエッジア...

実用情報 | 360 Gamesが「2016年中国モバイルゲーム業界動向レポート」を発表 模倣による反撃の時代は終わりに近づいている

モバイルゲーム市場は今や大手企業の「遊び場」となっており、中小のチームが反撃の機会を得るのはますます...

音楽ウェブサイトの SEO 戦略: キーワード

音楽を聴くのが好きな友達は、いくつかの音楽ウェブサイトが好きです。中国で有名な音楽ウェブサイトには、...

VirtNetwork - $7/12 コア/KVM/2g メモリ/110g ハードディスク/5T トラフィック

VPS業者のvirtnetwork.comはこれまでHostCatブログに登場したことがありません。...

王童:パーソナルブランドを構築する過程で、具体的な理論を作り、コンテンツをどのようにアウトプットしていくのでしょうか?

ショートビデオ、セルフメディア、インフルエンサーのためのワンストップサービスショッピングモール内の店...

大ヒットした3Dタイタニックからの主要なトラフィック選択戦略の簡単な分析

仕事が忙しくて、最近ヒットした映画「タイタニック3D」の視覚体験を楽しむために映画館に行く時間があり...

現在の SEO、路地に入ってしまったのでしょうか?

現在の SEO の道は少し迷っているようで、人々はとらえどころがないと感じています。たとえば、検索エ...

ウェブサイトユーザーの粘着性

ユーザーの粘着性はよく話題に上がるトピックですが、この最も一般的なトピックは、長年の経験を持つ SE...

MIT の定番コース「分散システム」のビデオ版がオンラインになりました。ネットユーザー:ついに非秘密バージョンが登場

この記事はAI新メディアQuantum Bit(公開アカウントID:QbitAI)より許可を得て転載...

キングサーバー: VPSのみ2.5ドル、専用サーバー45ドル、ロシア/米国/オランダ

ロシアのホスティング会社、King Servers BV をご紹介します。同社の主な業務は、仮想ホス...

aoyoyun (Maxthon Host) ロサンゼルス CU2-China Unicom VIP 回線 VPS、ハイエンド AS9929+AS4809 回線の簡単なレビュー

11年間運営してきた老舗のVPSベンダーであるAoyoyunは、3月にロサンゼルスのデータセンターで...