[解決済み] スレッドの作成はなぜ高いと言われるのですか?
質問
Javaチュートリアルでは、Threadの作成にはコストがかかると言われています。しかし、なぜ高価なのでしょうか?Java Threadが作成されるとき、一体何が起こっていて、その作成が高価になるのでしょうか?私はその発言を真実と受け止めていますが、JVMにおけるThreadの生成のメカニズムに興味があるだけです。
<ブロッククオートスレッドライフサイクルのオーバーヘッド。スレッドの生成と停止は無料ではありません。実際のオーバーヘッドはプラットフォームによって異なりますが、スレッドの生成には時間がかかり、リクエスト処理にレイテンシーが発生し、JVMとOSによる何らかの処理動作が必要になります。ほとんどのサーバーアプリケーションのように、リクエストが頻繁で軽量な場合、リクエストごとに新しいスレッドを作成すると、かなりのコンピューティングリソースを消費してしまいます。
から
Java並行処理の実際
Brian Goetz、Tim Peierls、Joshua Bloch、Joseph Bowbeer、David Holmes、Doug Lea 著
印刷 ISBN-10: 0-321-34960-1
解決するには?
<ブロッククオートなぜスレッドを作成するのか と言われています。 が高いのでしょうか?
なぜなら、>>は<高いからです。
Java のスレッド作成は、それなりに手間がかかるので、高価です。
- スレッドスタック用に大きなメモリブロックを確保し、初期化する必要があります。
- ホストOSにネイティブスレッドを作成/登録するためにシステムコールを行う必要があります。
- ディスクリプタを作成し、初期化し、JVM内部データ構造に追加する必要があります。
例えば、スレッドスタック、スタックから到達可能なすべてのオブジェクト、JVMスレッド記述子、OSネイティブスレッド記述子などです。
これらすべてのコストはプラットフォームによって異なりますが、私がこれまで出会ったどのJavaプラットフォームでも、決して安くはないものでした。
Googleで検索すると 旧ベンチマーク このレポートでは、Sun Java 1.4.1 と 2002 年代のデュアルプロセッサ Xeon で 2002 年代の Linux を実行した場合、スレッド作成レートが毎秒 4000 になることが報告されています。 もっと現代的なプラットフォームなら、もっといい数字が出るでしょうし、その方法論についてはコメントできませんが、少なくとも次のような目安にはなるでしょう。 高い スレッドの作成は、おそらく
Peter Lawrey氏のベンチマークによると、最近のスレッド作成は絶対的にかなり高速化しているようですが、JavaやOSの改良、あるいはプロセッサの高速化がどの程度影響しているかは不明です。 しかし、彼の数値は なお スレッドプールを使用した場合、毎回新しいスレッドを作成/開始する場合に比べて、150倍以上改善されることを示しています。 (そして彼は、これはすべて相対的なものであると指摘しています...)
上記はグリーンスレッドではなくネイティブスレッドを想定していますが、最近のJVMは性能上の理由からすべてネイティブスレッドを使用しています。 グリーンスレッドは作成コストが安いかもしれませんが、他の部分でその代償を払うことになります。
更新:OpenJDK Loomプロジェクト は、特に標準的なJavaスレッドの軽量な代替品を提供することを目的としています。 提案されているのは 仮想スレッド これは、ネイティブスレッドとグリーンスレッドのハイブリッドである。 簡単に言うと、仮想スレッドは、並列実行が必要なときにネイティブスレッドの下で使用するグリーンスレッドの実装のようなものです。
現在(2021年1月)、Project Loomの作業はまだプロトタイピングの段階であり、(AFAIK)Javaバージョンのリリースは目標とされていない。
Javaスレッドのスタックが実際にどのように割り当てられるのか、少し調べてみました。 Linux上のOpenJDK 6の場合、スレッドスタックは、以下の呼び出しによって割り当てられます。
pthread_create
がネイティブスレッドを生成します。 (JVMは
pthread_create
を使用すると、事前に割り当てられたスタックを使用することができます)。
そして、その中の
pthread_create
を呼び出すと、スタックが確保されます。
mmap
を次のようにします。
mmap(0, attr.__stacksize,
PROT_READ|PROT_WRITE|PROT_EXEC,
MAP_PRIVATE|MAP_ANONYMOUS, -1, 0)
によると
man mmap
は、その
MAP_ANONYMOUS
フラグを使用すると、メモリはゼロに初期化されます。
このように、新しいJavaスレッドスタックがゼロになることは(JVMの仕様上)必須ではなくても、実際には(少なくともLinux上のOpenJDK 6では)ゼロになっています。
関連
-
Eclipseプロンプトを実行する java仮想マシンを使用しない
-
スレッド "main" での例外 java.lang.ArrayIndexOutOfBoundsException: 1
-
[解決済み] この2回(1927年)を引き算すると、なぜおかしな結果になるのでしょうか?
-
[解決済み] なぜパスワードにはStringではなくchar[]が好まれるのですか?
-
[解決済み] serialVersionUIDとは何ですか、またなぜそれを使用する必要がありますか?
-
[解決済み] B "の印刷が "#"の印刷より劇的に遅いのはなぜですか?
-
[解決済み] Python 3で「1000000000000000 in range(1000000000000001)」はなぜ速いのですか?
-
[解決済み] Javaにおける "implements Runnable "と "extends Thread "の違いについて
-
[解決済み] プロセスとスレッドの違いは何ですか?
-
[解決済み】なぜJavaの+=, -=, *=, /=複合代入演算子はキャスティングを必要としないのですか?
最新
-
nginxです。[emerg] 0.0.0.0:80 への bind() に失敗しました (98: アドレスは既に使用中です)
-
htmlページでギリシャ文字を使うには
-
ピュアhtml+cssでの要素読み込み効果
-
純粋なhtml + cssで五輪を実現するサンプルコード
-
ナビゲーションバー・ドロップダウンメニューのHTML+CSSサンプルコード
-
タイピング効果を実現するピュアhtml+css
-
htmlの選択ボックスのプレースホルダー作成に関する質問
-
html css3 伸縮しない 画像表示効果
-
トップナビゲーションバーメニュー作成用HTML+CSS
-
html+css 実装 サイバーパンク風ボタン
おすすめ
-
Java エラー報告 スレッド "main" での例外 java.util.NoSuchElementException
-
springboot project MIMEタイプ text/htmlで転送された静的ファイルを読み込む。
-
スレッド "main "での例外をEclipseで解決 java.lang.Error: 未解決のコンパイル問題、コンパイラとパッケージの不整合
-
undefinedeclipse エラー。この行に複数のアノテーションが見つかりました: - 文字列を型解決に解決できない
-
jd-gui Java Exceptionが発生しました。
-
プロジェクトの依存関係を解決できなかった 解決
-
エラーの解決方法 jarfile XXX.jarにアクセスできません。
-
Methodのinvokeメソッド実装のJavaリフレクション
-
Javaエラーメッセージがenclosingクラスでない
-
あるコードに出会いましたが、何に使うのか理解できません。 List<String> list = new ArrayList<String>() { { a