1、堵塞原因分析
Java使用DatagramSocket代表UDP协议的Socket,DatagramSocket本身只是码头,不维护状态,不能产生IO流,它的唯一作用就是接收和发送数据报,Java使用DatagramPacket来代表数据报,DatagramSocket接收和发送的数据都是通过DatagramPacket对象完成的。
在接收数据前,通常需要生成一个DatagramPacket对象,给出数据字节数组及其长度。然后调用DatagramSocket的receive()方法等待数据报的到来,该方法会一直阻塞调用该方法的线程,直到收到一个数据报为止。
如下:
// 创建一个接收数据的DatagramPacket对象
DatagramPacket data_packet=new DatagramPacket(buf, 1024);
// 接收数据报
socket.receive(data_packet);
2、解决方法
在Java中,可以使用DatagramSocket的setSoTimeout(int timeout)方法来设置超时时间(单位是毫秒)。当在receive()方法调用时,如果在指定的超时时间内没有接收到数据报文,则会抛出SocketTimeoutException。
如下:
DatagramSocket socket = new DatagramSocket(9999)
// 设置超时时间为1000毫秒
socket.setSoTimeout(1000);
byte[] buffer = new byte[1024];
DatagramPacket packet = new DatagramPacket(buffer, buffer.length);
try {
// 尝试接收数据报文
socket.receive(packet);
// 处理接收到的数据
String data = new String(packet.getData(), 0, packet.getLength());
System.out.println("Received data: " + data);
} catch (SocketTimeoutException e) {
// 处理超时异常
System.out.println("Timeout occurred, no data received within the timeout period.");
}
} catch (IOException e) {
e.printStackTrace();
}
小结:这样使用socket接收时间时就不会一直堵塞在那里,会抛出异常,可以在finally中进行超时异常处理,如sleep将cpu资源进行释放。