DatagramPacket与DatagramSocket 详解

2022-08-18 12:52:20 浏览数 (1)

大家好,又见面了,我是你们的朋友全栈君。

1.基本概念:

  a.DatagramPacket与DatagramSocket位于java.net包中

  b.DatagramPacket表示存放数据的数据报,DatagramSocket表示接受或发送数据报的套接字

  c.由这两个类所有构成的网络链接是基于UDP协议,是一种不可靠的协议。

  之所以不可靠是因为发送方不负责数据是否发送成功,接收方收到数据时也不会   向发送方反馈成功消息,容易导致信息的丢失。

  但是这种协议却是快捷的,比如CS(半条命)游戏就是基于UDP协议,否则我们估计   要玩一次游戏后就会把机子砸了,所以我们可以看到游戏有时会失帧。

  2.使用方法:

  要建立基于UDP协议的链接,我们应该先建立套接字<DatagramSocket>(即发送站或接收站),之后通过套接字发送或接受数据<DatagramPacket>。

  我们先了解一下DatagramSocket和DatagramPacket类

  DatagramSocket类:

  构造方法:

  DatagramSocket():

  表示创建一个默认的套接字,并绑定到本地地址和一个随机的端口号

  DatagramSocket(int port):

  与上面不同的是,绑定到特定端口号,其它不变

  DatagramSocket(int port, InetAddress iad):

  表示创建一个套接字,绑定到特定的端口号及指定地址

  DatagramSocket(SocketAddress sad);

  表示创建一个套接字,绑定到特定的套接字地址

  小提示:所谓的本地地址我们可以这样来理解,我们通过“控制面板-本地连接-支持”中可以看到一个IP地址,这就是本地的IP地址,也暂且理解为本地地址。

  基本方法:

  close():

  关闭套接字

  recevie(DatagramPacket dp):

  接受数据报

  send(DatagramPacket dp):

 发送数据报

  这里触及了数据报,下面介绍数据报:

  DatagramPacket类:

  构造方法:(只列出主要的)

  接受类型:

  DatagramPacket(byte[] buf, int length):

  用来接受长度为length的buf数据(即数据存于字节数组buf中)

  发送类型:

  DatagramPacket(byte[] buf, int length, InetAddress address, int port);

  将length长的buf数据发送到指定的地址的端口号处

  DatagramPacket(byte[] buf, int length, SocketAddress address);

   将length长的buf数据发送到指定的套接字地址处

  有上面可知,我们发送数据时构造的数据报应该是发送类型的,而接受数据报时应该是接受类型的    

  开始书写代码

  代码意图:

  1.一个接收方,一个发送方

  2.当接收方收到发送方发送的消息后,打印发送的消息及发送方的地址和端口号,之后向发送反馈一条信息“接受方:我收到了!”

  3.发送方打印出接收方反馈的消息

  详细思路见代码:

  1. import java.net.DatagramPacket;
  2. import java.net.DatagramSocket;
  3. import java.net.InetAddress;
  4. public class Sender {
  5. public static void main(String[] args) {
  6. try {
  7. // 创建发送方的套接字,IP默认为本地,端口号随机
  8. DatagramSocket sendSocket = new DatagramSocket();
  9. // 确定要发送的消息:
  10. String mes = “你好!接收方!”;
  11. // 由于数据报的数据是以字符数组传的形式存储的,所以传转数据
  12. byte[] buf = mes.getBytes();
  13. // 确定发送方的IP地址及端口号,地址为本地机器地址
  14. int port = 8888;
  15. InetAddress ip = InetAddress.getLocalHost();
  16. // 创建发送类型的数据报:
  17. DatagramPacket sendPacket = new DatagramPacket(buf, buf.length, ip,
  18. port);
  19. // 通过套接字发送数据:
  20. sendSocket.send(sendPacket);
  21. // 确定接受反馈数据的缓冲存储器,即存储数据的字节数组
  22. byte[] getBuf = new byte[1024];
  23. // 创建接受类型的数据报
  24. DatagramPacket getPacket = new DatagramPacket(getBuf, getBuf.length);
  25. // 通过套接字接受数据
  26. sendSocket.receive(getPacket);
  27. // 解析反馈的消息,并打印
  28. String backMes = new String(getBuf, 0, getPacket.getLength());
  29. System.out.println(“接受方返回的消息:” backMes);
  30. // 关闭套接字
  31. sendSocket.close();
  32. } catch (Exception e) {
  33. e.printStackTrace();
  34. }
  35. }
  36. }
  37. import java.net.DatagramPacket;
  38. import java.net.DatagramSocket;
  39. import java.net.InetAddress;
  40. import java.net.SocketAddress;
  41. public class Receive {
  42. public static void main(String[] args) {
  43. try {
  44. // 确定接受方的IP和端口号,IP地址为本地机器地址
  45. InetAddress ip = InetAddress.getLocalHost();
  46. int port = 8888;
  47. // 创建接收方的套接字,并制定端口号和IP地址
  48. DatagramSocket getSocket = new DatagramSocket(port, ip);
  49. // 确定数据报接受的数据的数组大小
  50. byte[] buf = new byte[1024];
  51. // 创建接受类型的数据报,数据将存储在buf中
  52. DatagramPacket getPacket = new DatagramPacket(buf, buf.length);
  53. // 通过套接字接收数据
  54. getSocket.receive(getPacket);
  55. // 解析发送方传递的消息,并打印
  56. String getMes = new String(buf, 0, getPacket.getLength());
  57. System.out.println(“对方发送的消息:” getMes);
  58. // 通过数据报得到发送方的IP和端口号,并打印
  59. InetAddress sendIP = getPacket.getAddress();
  60. int sendPort = getPacket.getPort();
  61. System.out.println(“对方的IP地址是:” sendIP.getHostAddress());
  62. System.out.println(“对方的端口号是:” sendPort);
  63. // 通过数据报得到发送方的套接字地址
  64. SocketAddress sendAddress = getPacket.getSocketAddress();
  65. // 确定要反馈发送方的消息内容,并转换为字节数组
  66. String feedback = “接收方说:我收到了!”;
  67. byte[] backBuf = feedback.getBytes();
  68. // 创建发送类型的数据报
  69. DatagramPacket sendPacket = new DatagramPacket(backBuf,
  70. backBuf.length, sendAddress);
  71. // 通过套接字发送数据
  72. getSocket.send(sendPacket);
  73. // 关闭套接字
  74. getSocket.close();
  75. } catch (Exception e) {
  76. e.printStackTrace();
  77. }
  78. }
  79. }

测试已通过

  先启动接收方,然后启动发送方即可

发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/134597.html原文链接:https://javaforall.cn

0 人点赞