UDP 接続オブジェクトの原理分析と使用例

UDP 接続オブジェクトの原理分析と使用例

以前、UDP を使い始めるために簡単な UDP サーバーとクライアントの例を作成しましたが、実際に使用してみると問題が発生しました。

前回使用したときも、接続オブジェクト DatagramSocket を static として記述し、クラスの初期化時に使用していました。しかし、システム内の多くの場所で使用されています。このクラスのオブジェクトを作成し続ける必要がありますか?

これを実行することは可能ですが、メモリ オーバーフローという結果を招く可能性があります。

UDP はステートレスです。DatagramSocket は、特定のアドレスのポートを指すために、毎回作成する必要はなく、一度作成するだけで済みます。

UDP はステートレスなので、DatagramSocket オブジェクトを作成すると、ネットワークを指すオブジェクトのみが作成されます。これは、特定の方向を向いた大きなスピーカーを設置しても、その方向で聞いている人がいるかどうかはわからないのと同じです。

サーバーが稼働していなくても、接続オブジェクトを作成してこのアドレスにデータを送信しても問題はありません。拡声器で特定の方向に向かって叫んでも、誰も聞いてくれなければ意味がありません。ただし、必要なときに応答が受信されない場合は、タイムアウト後にエラーが報告されます。

パッケージ udp; 
 
java.net.* をインポートします。 
 
/** 
 * @Description UDP クライアント プログラム。サーバーにデータを送信し、サーバーの応答情報を受信するために使用されます。* @author cuisuqiang 
 * @バージョン 1.0 
 * @since <a href="mailto:[email protected]" rel="external nofollow" >[email protected]</a> 
 */ 
パブリッククラス UdpClientSocket { 
  /** 
   * 接続オブジェクト */ 
  プライベート静的 DatagramSocket ds = null; 
  /** 
   * アドレスオブジェクト */ 
  プライベート静的SocketAddressアドレス = null; 
   
  /** 
   * クライアントのパケット送信方法と応答情報の受信方法をテストします*/ 
  パブリック静的void main(String[] args)は例外をスローします{ 
    初期化(); 
    while(true){ 
      UdpClientSocket.send(address,"こんにちは、親愛なる!".getBytes()); 
      UdpClientSocket を受信します。 
      試す { 
        スレッドをスリープ状態にします(3 * 1000); 
      } キャッチ (例外 e) { 
        e.printStackTrace(); 
      } 
    } 
  } 
   
  /** 
   * 接続とアドレスを初期化します */ 
  パブリック静的void init(){ 
    試す { 
      ds = new DatagramSocket(8899); // クライアントとしてローカルポートにバインド ds.setSoTimeout(2 * 1000); 
      アドレス = 新しい InetSocketAddress("127.0.0.1",3344); 
    } キャッチ (例外 e) { 
      e.printStackTrace(); 
    } 
  } 
   
  /** 
   * 指定されたサーバーにデータ情報を送信します */ 
  パブリック静的void送信(SocketAddressアドレス、byte[]バイト){ 
    試す { 
      DatagramPacket dp = 新しい DatagramPacket(バイト、バイト長、アドレス); 
      ds.send(dp); 
    } キャッチ (例外 e) { 
      e.printStackTrace(); 
    } 
  } 
 
  /** 
   * 指定されたサーバーから送り返されたデータを受信します*/ 
  パブリック静的void受信(){ 
    試す { 
      byte[] buffer = 新しいbyte[1024]; 
      DatagramPacket dp = 新しい DatagramPacket(バッファ、バッファ長); 
      ds.receive(dp);    
      byte[]データ = new byte[dp.getLength()]; 
      System.arraycopy(dp.getData(), 0, データ, 0, dp.getLength());  
      System.out.println("サーバー応答データ: " + new String(data)); 
    } キャッチ (例外 e) { 
      e.printStackTrace(); 
    } 
  } 
}

コードを実行した結果は次のとおりです。

java.net.SocketTimeoutException: 受信がタイムアウトしました
java.net.PlainDatagramSocketImpl.receive0(ネイティブメソッド)
java.net.PlainDatagramSocketImpl.receive(PlainDatagramSocketImpl.java:136) で
java.net.DatagramSocket.receive(DatagramSocket.java:712) で
udp.UdpClientSocket.receive(UdpClientSocket.java:69) で
udp.UdpClientSocket.main(UdpClientSocket.java:28) で

操作はタイムアウトしましたが、エラーはオブジェクトの作成とデータの送信によって発生したのではなく、データ受信時のタイムアウトによって発生しました。

このプログラムは実行し続けますので、サーバーを作成しましょう。

パッケージ udp;

java.net.DatagramPacket をインポートします。
java.net.DatagramSocket をインポートします。
java.net.InetSocketAddress をインポートします。
java.net.SocketAddress をインポートします。

/**
 * @UDP サービスクラスの説明 * @author cuisuqiang
 * @バージョン 1.0
 * @since [email protected]
 */
パブリッククラス UdpServerSocket {
	
	プライベート静的 DatagramSocket ds = null;
	プライベート静的SocketAddressアドレス = null;
	
	/**
	 * テスト方法 */
	パブリック静的void main(String[] args)は例外をスローします{
		初期化();
		System.out.println("---->サービスがリッスンを開始します!<----");
		(真)の間{
			UdpServerSocket.receive();
			UdpServerSocket.response(address,"こんにちは、食事はしましたか?");
		}		
	}
	
	パブリック静的void init(){
		試す {
			ds = 新しいデータグラムソケット(3344);
			ds.setSoTimeout(0);
			アドレス = 新しい InetSocketAddress("127.0.0.1",8899);
		} キャッチ (例外 e) {
			e.printStackTrace();
		}
	}

	/**
	 * データパケットを受信します。このメソッドはスレッドをブロックします */
	パブリック静的void受信() {
		試す {
			byte[] buffer = 新しいbyte[1024];
			DatagramPacket パケット = 新しい DatagramPacket(バッファ、バッファ長);
			ds.receive(パケット);
			文字列情報 = 新しい文字列(packet.getData(), 0, packet.getLength());
			System.out.println("情報を受信: " + info);
		} キャッチ (例外 e) {
			e.printStackTrace();
		}
	}

	/**
	 * 応答パケットを要求元に送信する */
	パブリック静的void応答(SocketAddressアドレス、文字列情報){
		試す {
			DatagramPacket dp = 新しい DatagramPacket(info.getBytes(), info.getBytes().length, address);
			dp.setData(info.getBytes());
			ds.send(dp);
		} キャッチ (例外 e) {
			e.printStackTrace();
		}		
	}
}

実行後、クライアントは正常にデータを送受信できるようになります。

実際に使用する場合は、システム スタートアップ項目を設定して、init 接続オブジェクトとアドレスを初期化し、使用時に例外をキャプチャします。

接続オブジェクトが毎回作成され、頻繁に使用される場合、通常、システムは数分以内にクラッシュします。

以上がこの記事の全内容です。皆様の勉強のお役に立てれば幸いです。また、123WORDPRESS.COM を応援していただければ幸いです。

以下もご興味があるかもしれません:
  • UDP シンプル サーバー クライアント コード例
  • Python UDPプログラミングの詳細な説明
  • TCP/UDP プロトコルを使用した C# サンプル コード
  • UDP チャット プログラムに基づく Java ネットワークの例の分析
  • Java シミュレーション UDP 通信サンプル コード
  • PythonはUDPプログラム通信プロセス図を実装します
  • PythonはUDPプロトコルによるファイル転送を実装する
  • Java UDP 通信クライアントとサーバーの例の分析

<<:  JavaScript 配列重複排除ソリューション

>>:  2 つの MySQL ユーザー削除ステートメント (delete user と drop user) の違い

推薦する

SQL文のANDとORの実行順序で発生する問題

質問昨日、データベースSQLを書いているときに問題が発生しました。問題の根本は、SQL ステートメン...

HTML ページでギリシャ文字を使用する方法

ギリシャ文字は、特に数学や物理学などの科学技術分野で非常によく使用される記号列であり、特定の意味を持...

Centos 7 mysql-8.0.19-1.el7.x86_64.rpm-bundle.tar の簡単な分析

Baiduクラウドディスク:リンク: https://pan.baidu.com/s/1hv5rUW...

MYSQL ログとバックアップおよび復元の問題の詳細な説明

この記事では、参考までにMYSQLログとバックアップとリストアについて紹介します。具体的な内容は以下...

Windows 7 で Python 3.4 を使って MySQL データベースを使用する

Python 3.4でMySQLデータベースを使用する詳細なプロセスは次のとおりです。 Window...

Dockerを使用してシンプルなJava開発およびコンパイル環境を構築する方法の詳細な説明

Java 言語には多くのバージョンがあります。一般的に使用されている Java 8 に加えて、一部の...

React useEffect の理解と使用

目次繰り返しレンダリングループを避ける副作用の除去についてReact16.8 の新しい useEff...

TypeScript で時間を費やした場所の概要

TS で時間を過ごした場所をいくつか記録します。 (まず、文句を言わせてください。stackover...

WeChatミニプログラムマップの使い方を詳しく解説

この記事の例では、WeChatアプレットマップで使用される具体的な実装コードを参考までに共有していま...

Vueはキャンバスを使用して画像圧縮アップロードを実現します

この記事では、キャンバスを使用して画像圧縮アップロードを実現するVueの具体的なコードを参考までに共...

MySQL 8.0 の非表示インデックスの詳細な説明

言葉MySQL 8.0 は最初のバージョンから 4 年を経てリリースされました。バージョン 8.0 ...

ウェブサイト開発におけるフロントエンド開発者とアーティストの知識の違い

概要: 多くの企業、特にインターネット Web サイトを主な事業とする企業のほとんどが、「アーティス...

MySQL 接続失敗の一般的な障害と原因

==================================================...

Alibaba Cloud Server の詳細な展開 (グラフィック チュートリアル)

最近、Web 開発のフロントエンドとバックエンドの技術を学んだので、その後の管理を容易にするためにプ...