UDP连接对象原理解析及使用实例

 更新时间:2020年10月20日 11:32:44   作者:cuisuqiang  
这篇文章主要介绍了UDP连接对象原理解析及使用实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下

之前写了一个简单的UDP服务端和客户端示例,用于入门UDP,当我实际使用时发生了一点问题!

上次使用中我也把连接对象 DatagramSocket 写成了静态的,在类的初始化时使用,可是系统中有很多地方使用,难道我要不断的创建这个类的对象吗?

可以这么做,当时有后果,后果就是内存溢出。

UDP是没有状态的,DatagramSocket 创建一次即可,就是开始指向某个地址的端口,而不用每次创建。

由于UDP是无状态的,在创建 DatagramSocket 对象时只是创建了一个指向网络的对象,就像你架设一个大喇叭对着某个方向,可是你并不知道这个方向到底有没有人在听。

如果,即使你没有开服务端,创建连接对象并向这个地址放松数据,都是没有问题。你用喇叭向某个方向喊没人听这没有什么!可是当你需要回应时如果一直没有接到响应,超时之后就会报错!

package udp; 
 
import java.net.*; 
 
/** 
 * @说明 UDP客户端程序,用于对服务端发送数据,并接收服务端的回应信息 
 * @author cuisuqiang 
 * @version 1.0 
 * @since <a href="mailto:cuisuqiang@163.com" rel="external nofollow" >cuisuqiang@163.com</a> 
 */ 
public class UdpClientSocket { 
  /** 
   * 连接对象 
   */ 
  private static DatagramSocket ds = null; 
  /** 
   * 地址对象 
   */ 
  private static SocketAddress address = null; 
   
  /** 
   * 测试客户端发包和接收回应信息的方法 
   */ 
  public static void main(String[] args) throws Exception { 
    init(); 
    while(true){ 
      UdpClientSocket.send(address,"你好,亲爱的!".getBytes()); 
      UdpClientSocket.receive(); 
      try { 
        Thread.sleep(3 * 1000); 
      } catch (Exception e) { 
        e.printStackTrace(); 
      } 
    } 
  } 
   
  /** 
   * 对连接和地址初始化 
   */ 
  public static void init(){ 
    try { 
      ds = new DatagramSocket(8899); // 邦定本地端口作为客户端 
      ds.setSoTimeout(2 * 1000); 
      address = new InetSocketAddress("127.0.0.1",3344); 
    } catch (Exception e) { 
      e.printStackTrace(); 
    } 
  } 
   
  /** 
   * 向指定的服务端发送数据信息 
   */ 
  public static void send(SocketAddress address,byte[] bytes){ 
    try { 
      DatagramPacket dp = new DatagramPacket(bytes, bytes.length, address); 
      ds.send(dp); 
    } catch (Exception e) { 
      e.printStackTrace(); 
    } 
  } 
 
  /** 
   * 接收从指定的服务端发回的数据 
   */ 
  public static void receive(){ 
    try { 
      byte[] buffer = new byte[1024]; 
      DatagramPacket dp = new DatagramPacket(buffer, buffer.length); 
      ds.receive(dp);    
      byte[] data = new byte[dp.getLength()]; 
      System.arraycopy(dp.getData(), 0, data, 0, dp.getLength());  
      System.out.println("服务端回应数据:" + new String(data)); 
    } catch (Exception e) { 
      e.printStackTrace(); 
    } 
  } 
} 

执行以代码运行结果如下:

java.net.SocketTimeoutException: Receive timed out
at java.net.PlainDatagramSocketImpl.receive0(Native Method)
at java.net.PlainDatagramSocketImpl.receive(PlainDatagramSocketImpl.java:136)
at java.net.DatagramSocket.receive(DatagramSocket.java:712)
at udp.UdpClientSocket.receive(UdpClientSocket.java:69)
at udp.UdpClientSocket.main(UdpClientSocket.java:28)

运行超时,但是报错的地方不是创建对象和发送数据,而是接收数据时超时!

这个程序一直运行,我们来搞一个服务端:

package udp;

import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetSocketAddress;
import java.net.SocketAddress;

/**
 * @说明 UDP服务类
 * @author cuisuqiang
 * @version 1.0
 * @since cuisuqiang@163.com
 */
public class UdpServerSocket {
	
	private static DatagramSocket ds = null;
	private static SocketAddress address = null;
	
	/**
	 * 测试方法
	 */
	public static void main(String[] args) throws Exception {
		init();
		System.out.println("---->服务开始监听!<----");
		while (true) {
			UdpServerSocket.receive();
			UdpServerSocket.response(address,"你好,吃了吗!");
		}		
	}
	
	public static void init(){
		try {
			ds = new DatagramSocket(3344);
			ds.setSoTimeout(0);
			address = new InetSocketAddress("127.0.0.1",8899);
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

	/**
	 * 接收数据包,该方法会造成线程阻塞
	 */
	public static void receive() {
		try {
			byte[] buffer = new byte[1024];
			DatagramPacket packet = new DatagramPacket(buffer, buffer.length);
			ds.receive(packet);
			String info = new String(packet.getData(), 0, packet.getLength());
			System.out.println("接收信息:" + info);
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

	/**
	 * 将响应包发送给请求端
	 */
	public static void response(SocketAddress address,String info){
		try {
			DatagramPacket dp = new DatagramPacket(info.getBytes(), info.getBytes().length, address);
			dp.setData(info.getBytes());
			ds.send(dp);
		} catch (Exception e) {
			e.printStackTrace();
		}		
	}
}

运行后客户端可以正常发送和接收数据!

如果在实际运用中,我是设置一个系统启动项,来初始化 init 连接对象和地址,具体使用时进行异常捕获就可以了!

如果你的连接对象每次创建,且使用频繁,一般几分钟系统即可搞挂!

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • 手机怎么访问电脑服务器上的网页

    手机怎么访问电脑服务器上的网页

    这篇文章主要介绍了手机怎么访问电脑服务器上的网页的相关资料,需要的朋友可以参考下
    2016-02-02
  • Memcached简介_动力节点Java学院整理

    Memcached简介_动力节点Java学院整理

    这篇文章主要介绍了Memcached简介,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-08-08
  • 服务器重启不能启动的几种常见的解决方法

    服务器重启不能启动的几种常见的解决方法

    服务器重启不能启动的几种常见的解决方法,碰到此问题的朋友可以参考下。
    2011-04-04
  • Webstorm2017上SVN插件安装的方法步骤

    Webstorm2017上SVN插件安装的方法步骤

    这篇文章主要介绍了Webstorm2017上SVN插件安装的方法步骤,安装Webstorm2017后,要使用svn版本控制,主要介绍如何安装配置SVN,分享给大家,有需要的可以了解一下
    2018-05-05
  • ibmx335/ibmx336服务器做RAID阵列的图文方法(包括删除RAID阵列)

    ibmx335/ibmx336服务器做RAID阵列的图文方法(包括删除RAID阵列)

    X服务器中有一些集成LSI SCSI控制器的机型,在开机自检时按CTRL C可以配置两个硬盘的镜像。但是当升级BIOS之后,CTRL C中的一些设置发生了变化,配置方法也较以前的版本有些差异
    2012-06-06
  • 树莓派搭建web服务器的方法实现(无需公网IP)

    树莓派搭建web服务器的方法实现(无需公网IP)

    本文主要介绍了树莓派搭建web服务器的方法实现(无需公网IP),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-08-08
  • 网站压力测试工具-ab工具apache bench使用过程

    网站压力测试工具-ab工具apache bench使用过程

    apache bench是apache自带的压力测试工具。ab不仅可以对apache服务器进行网站访问压力测试,也可以对或其它类型的服务器进行压力测试。ab工具上手学习较快,可以提供需要的基本性能指标,但没有图形化结果,不能监控。因此可以用作临时紧急任务和简单测试。
    2022-11-11
  • Nexus使用Api进行操作

    Nexus使用Api进行操作

    今天小编就为大家分享一篇关于Nexus使用Api进行操作,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-12-12
  • 使用.htaccess设置图片防盗链的详细方法

    使用.htaccess设置图片防盗链的详细方法

    对于虚拟主机用户来说,最方便的莫过于利用.htaccess设置图片防盗链了。上网搜了一下,.htaccess规则写法很多,但是大部分的区别在于最后一行,下面我们先看看正确的.htaccess防盗链写法,稍后我再详细解释一下最后一行的重点
    2018-02-02
  • Cloudflare免费免备案cdn加速优化设置方法指南

    Cloudflare免费免备案cdn加速优化设置方法指南

    CDN加速是利用节点让用户建成访问时,直接将缓存好的静态文件交给用户。还可以隐藏我们网站的IP地址,变得更安全。Cloudflare CDN免费加速你的网站,合适的配置会使我们网站的访问速度和安全性都得到进一步的提升。
    2023-04-04

最新评论