検索
ホームページJava&#&チュートリアルLinux と Java のゼロコピーについて学ぶ

Linux と Java のゼロコピーについて学ぶ

Linux 従来の IO

皆さんこんにちは。私は Linux ディスク上にあるデータです。ここで、ディスクからネットワーク カードに送信するには、次の手順を実行する必要があります:

読み取り操作

Linux と Java のゼロコピーについて学ぶ

上に示すように、オペレーティング システムのメモリはカーネル空間とユーザー空間に分割されます。まず、ユーザー空間のアプリケーションがデータ読み取り操作を開始します (JVM が read() システム コールを開始するなど)。このとき、オペレーティング システムは コンテキスト スイッチ を実行します。ユーザー空間からカーネル空間に切り替えます。

その後、カーネル空間がディスクに通知し、カーネルはディスクからカーネルバッファに私をコピーします。この処理は「DMA(ダイレクト メモリ アクセス)」と呼ばれるハードウェアによって行われるため、CPU の関与は必要ありません。

次に、カーネルは私をカーネル バッファからアプリケーション バッファにコピーします。これには CPU の参加が必要です。

最後にコンテキスト切り替えを実行し、ユーザー空間コンテキストに戻ります。

読み取り操作プロセス全体には、2 つのコンテキスト スイッチと 2 つのコピーが必要です。

関連学習の推奨事項: Java ビデオ チュートリアル

書き込み操作 It読み取り操作と似ていますが、方向が逆であり、やはり 2 つのコンテキスト スイッチと 2 つのデータ コピーが必要です。ディスクに書き込まれたり、ネットワーク カードに書き込まれたりする可能性があります。

Linux と Java のゼロコピーについて学ぶ

メモリ マッピング

上記のプロセスからわかるように、ディスクからネットワークに送信したい場合は、カード、合計 4 つのコンテキスト スイッチと 4 つのコピー操作。オペレーティング システムによってカーネル空間とユーザー空間の間でコピーされていましたが、実際にはこの期間中は何もせず、何も変更せず、ただコピーしているだけでした。そのため、この IO モデルはオペレーティング システムのリソースの無駄であり、何度もコピーして、肉体的にも精神的にも疲れ果てました。さらに、オペレーティング システムのリソースは非常に貴重です~

現在、主流のオペレーティング システムはすべて

仮想メモリを使用しています。簡単に言うと、 仮想アドレスは物理アドレス を置き換えるために使用されます。これにより、複数の仮想メモリが同じ物理アドレスのみを必要とすることが可能になり、仮想メモリ空​​間は物理メモリ空間よりもはるかに大きくなる可能性があります。

オペレーティング システムがユーザー空間のアプリケーション バッファーとカーネル空間のカーネル バッファーを同じ物理アドレスにマップできれば、多くのコピー プロセスが不要になるのではないでしょうか?以下に示すように:

Linux と Java のゼロコピーについて学ぶ

##Linux Zero Copyしたがって、この問題を解決するために、賢明な Linux 開発者はいくつかの新しいシステム コールを作成しました。これをするために作られました。主に 2 つの方法があります:

mmap write
  • sendfile

mmap write

mmap()

システム コールは、まず DMA コピーを使用してディスクからカーネル バッファに読み取り、次にメモリ マッピングを使用して ユーザー バッファとカーネル読み取りバッファのメモリ アドレスを同じメモリ アドレスにします。つまり、CPU がカーネル読み取りバッファからユーザー バッファにコピーする必要はありません。

write()

システム コールを使用する場合、CPU は、送信する必要があるカーネル バッファ (ユーザー バッファに相当) からカーネル バッファに直接書き込みます。たとえば、ネットワーク送信バッファ (ソケット バッファ) を作成し、それを DMA 経由でネットワーク カード ドライバ (またはディスク) に渡し、送信の準備をします。

mmap + write#mmap 書き込みメソッドでは、データの読み取りと書き込みに、合計 2 つのシステム コール、4 つのコンテキスト スイッチ、2 つの DMA コピー、および 1 つの CPU コピーが必要です。

Linux と Java のゼロコピーについて学ぶ

Linux2.4 カーネルは、sendfile を最適化し、収集操作を提供します。この操作により、上の図の最後の CPU コピーを削除できます。原則として、データをコピーするのではありません。代わりに、 、前のカーネル バッファ (図の場合は読み取りバッファなど) 内のデータのメモリ アドレスとオフセット レコードがターゲットのカーネル バッファ (図の場合はソケット バッファなど) に送信されるため、コピー段階では、このポインタを使用してデータを直接コピーできます。

Linux と Java のゼロコピーについて学ぶ

Java NIO はゼロ コピーを使用します

Linux のゼロ コピーは、実際にオペレーティング システム リソースの一部を節約できます。したがって、Java の NIO は、ゼロ コピーをサポートするためにいくつかのクラスを提供します。

  • DirectByteBuffer
  • FileChannel

前の「Java NIO - バッファ」では、次のようになります。この記事では DirectByteBuffer について簡単に紹介しています。 ByteBuffer には主に 2 つの実装があり、1 つは DirectByteBuffer で、もう 1 つは HeapByteBuffer です。

このうち、DirectByteBuffer はヒープ外に直接メモリを割り当て、最下位層が JNI を介してオペレーティング システムの NIO システム コールを直接呼び出すため、比較的パフォーマンスが高くなります。 HeapByteBuffer はヒープ内メモリであり、データをもう一度コピーする必要があるため、パフォーマンスは比較的低くなります。

FileChannel は Java NIO が提供するファイルコピー用のクラスで、ファイルをディスクやネットワークなどにコピーできます。

mapこのメソッドは、実際にはオペレーティング システムのメモリ マッピング メソッドを使用して、カーネル バッファのメモリとユーザー バッファのメモリをアドレスにマップします。

transferToメソッドは、現在のチャネル コンテンツを別のチャネルに直接転送します。つまり、このメソッドには、カーネル バッファからユーザー バッファへの読み取りおよび書き込みの問題がありません。最下層はsendfileシステムコールです。 transferFrom方法は同じです。

サンプル コード:

File file = new File("test.txt");RandomAccessFile raf = new RandomAccessFile(file, "rw");FileChannel fileChannel = raf.getChannel();SocketChannel socketChannel = SocketChannel.open(new InetSocketAddress("", 8080));// 直接使用了transferTo()进行通道间的数据传输fileChannel.transferTo(0, fileChannel.size(), socketChannel);

著者: パブリック アカウント _xy の技術サークル

リンク: www.imooc.com/article/289550

出典: MOOC。 com

上記のコンテンツは MOOC.com からのものです

ゼロ コピーの再理解

  1. ゼロ コピーは、MOOC.com からのものです。運用 システムの観点から。カーネル バッファ間でデータが重複しないためです (カーネル バッファのみにデータのコピーが 1 つあります)。

  2. ゼロ コピーは、データ コピーの量を減らすだけでなく、コンテキストの切り替えの減少、CPU キャッシュの擬似共有の減少、CPU チェックサムの計算の不要など、他のパフォーマンス上の利点ももたらします。

mmap と sendFile の違い

  1. mmap は少量のデータの読み取りと書き込みに適しており、sendFile は大きなファイルの転送に適しています。

  2. mmap には 4 つのコンテキスト スイッチと 3 つのデータ コピーが必要です。sendFile には 3 つのコンテキスト スイッチと少なくとも 2 つのデータ コピーが必要です。

  3. sendFile は DMA を使用して CPU のコピーを減らすことができますが、mmap はできません (カーネルからソケット バッファにコピーする必要があります)。

以上がLinux と Java のゼロコピーについて学ぶの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明
この記事はlearnkuで複製されています。侵害がある場合は、admin@php.cn までご連絡ください。
高度なJavaプロジェクト管理、自動化の構築、依存関係の解像度にMavenまたはGradleを使用するにはどうすればよいですか?高度なJavaプロジェクト管理、自動化の構築、依存関係の解像度にMavenまたはGradleを使用するにはどうすればよいですか?Mar 17, 2025 pm 05:46 PM

この記事では、Javaプロジェクト管理、自動化の構築、依存関係の解像度にMavenとGradleを使用して、アプローチと最適化戦略を比較して説明します。

適切なバージョン化と依存関係管理を備えたカスタムJavaライブラリ(JARファイル)を作成および使用するにはどうすればよいですか?適切なバージョン化と依存関係管理を備えたカスタムJavaライブラリ(JARファイル)を作成および使用するにはどうすればよいですか?Mar 17, 2025 pm 05:45 PM

この記事では、MavenやGradleなどのツールを使用して、適切なバージョン化と依存関係管理を使用して、カスタムJavaライブラリ(JARファイル)の作成と使用について説明します。

カフェインやグアバキャッシュなどのライブラリを使用して、Javaアプリケーションにマルチレベルキャッシュを実装するにはどうすればよいですか?カフェインやグアバキャッシュなどのライブラリを使用して、Javaアプリケーションにマルチレベルキャッシュを実装するにはどうすればよいですか?Mar 17, 2025 pm 05:44 PM

この記事では、カフェインとグアバキャッシュを使用してJavaでマルチレベルキャッシュを実装してアプリケーションのパフォーマンスを向上させています。セットアップ、統合、パフォーマンスの利点をカバーし、構成と立ち退きポリシー管理Best Pra

キャッシュや怠zyなロードなどの高度な機能を備えたオブジェクトリレーショナルマッピングにJPA(Java Persistence API)を使用するにはどうすればよいですか?キャッシュや怠zyなロードなどの高度な機能を備えたオブジェクトリレーショナルマッピングにJPA(Java Persistence API)を使用するにはどうすればよいですか?Mar 17, 2025 pm 05:43 PM

この記事では、キャッシュや怠zyなロードなどの高度な機能を備えたオブジェクトリレーショナルマッピングにJPAを使用することについて説明します。潜在的な落とし穴を強調しながら、パフォーマンスを最適化するためのセットアップ、エンティティマッピング、およびベストプラクティスをカバーしています。[159文字]

Javaのクラスロードメカニズムは、さまざまなクラスローダーやその委任モデルを含むどのように機能しますか?Javaのクラスロードメカニズムは、さまざまなクラスローダーやその委任モデルを含むどのように機能しますか?Mar 17, 2025 pm 05:35 PM

Javaのクラスロードには、ブートストラップ、拡張機能、およびアプリケーションクラスローダーを備えた階層システムを使用して、クラスの読み込み、リンク、および初期化が含まれます。親の委任モデルは、コアクラスが最初にロードされ、カスタムクラスのLOAに影響を与えることを保証します

See all articles

ホット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ヘンタイを無料で生成します。

ホットツール

AtomエディタMac版ダウンロード

AtomエディタMac版ダウンロード

最も人気のあるオープンソースエディター

SAP NetWeaver Server Adapter for Eclipse

SAP NetWeaver Server Adapter for Eclipse

Eclipse を SAP NetWeaver アプリケーション サーバーと統合します。

PhpStorm Mac バージョン

PhpStorm Mac バージョン

最新(2018.2.1)のプロフェッショナル向けPHP統合開発ツール

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

mPDF

mPDF

mPDF は、UTF-8 でエンコードされた HTML から PDF ファイルを生成できる PHP ライブラリです。オリジナルの作者である Ian Back は、Web サイトから「オンザフライ」で PDF ファイルを出力し、さまざまな言語を処理するために mPDF を作成しました。 HTML2FPDF などのオリジナルのスクリプトよりも遅く、Unicode フォントを使用すると生成されるファイルが大きくなりますが、CSS スタイルなどをサポートし、多くの機能強化が施されています。 RTL (アラビア語とヘブライ語) や CJK (中国語、日本語、韓国語) を含むほぼすべての言語をサポートします。ネストされたブロックレベル要素 (P、DIV など) をサポートします。