1.DatagramSocket:这个类代表的是数据报,是数据报套接字发送或者接收点的分组传送服务,它本身只是码头,不维护状态,不能产生IO流,它的唯一作用就是接收和发送数据报。
类的继承关系
public class DatagramSocket
extends Object
implements Closeable
2.DatagramPacket:这个类表示一个数据报包,数据包是用来实现一个无连接的分组传送服务。每个消息都是从一台机器路由到另一个完全基于包含在该数据包内的信息。从一
台机器发送到另一台机器的多个数据包可能会被不同的路由,并可能以任何顺序到达。
类的继承关系
public final class DatagramPacket
extends Object
InetSocketAddress:这个类主要作用是封装端口 他是在在InetAddress基础上加端口,但它是有构造器的
类的继承关系
public class InetSocketAddress
extends SocketAddress
3.InetAddress:类的主要作用是封装IP及DNS,因为这个类没有构造器,所以我们要用他的一些方法来获得对象常用的有
- 使用getLocalHost方法为InetAddress创建对象;
- 根据域名得到InetAddress对象
- 根据ip得到InetAddress对象
UDP通过DatagramSocket和DatagramPacket传输数据时可以进行如下的对比理解:
udp传输数据就好像在大海上运输货物,DatagramSocket就像的两个通信港口的码头,DatagramPacket就像在两端运输货物的轮船
发送端和接收端代码实现如下
public class UDPChatDemo {
public static void main(String[] args) {
//开启线程
new Thread(new Sender()).start();
new Thread(new Receiver()).start();
}
}
//代表线程的类---发送端
class Sender implements Runnable{
@Override
public void run() {
//run 方法不能抛异常,不能抛给CPU
try {
DatagramSocket ds = new DatagramSocket();
Scanner sc = new Scanner(System.in);
while (true){
byte[] bs = sc.nextLine().getBytes();
DatagramPacket dp = new DatagramPacket(bs,0,bs.length
,new InetSocketAddress("255.255.255.255",9090));
//发送数据包
ds.send(dp);
}
} catch (SocketException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}
//代表线程的类--接收端
class Receiver implements Runnable{
@Override
public void run() {
try {
DatagramSocket ds = new DatagramSocket(9090);
DatagramPacket dp = new DatagramPacket(new byte[1024],1024);
//接收,while 保证可以持续进行输入传输
while (true){
ds.receive(dp);
//获取并展示发送端的IP
System.out.println(dp.getAddress());
//打印发送的内容
System.out.println(new String(dp.getData(),0,dp.getLength()));
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
运行结果如下: