网络编程—Socket套接字(UDP)

打印 上一主题 下一主题

主题 1746|帖子 1746|积分 5238

上篇文章:
网络编程—网络概念
https://blog.csdn.net/sniper_fandc/article/details/146923380?fromshare=blogdetail&sharetype=blogdetail&sharerId=146923380&sharerefer=PC&sharesource=sniper_fandc&sharefrom=from_link
目录
1 概念
2 Socket套接字
3 UDP数据报套接字


1 概念

        (1)网络编程:指网络上的主机,通过不同的进程,以编程的方式实现网络通讯(网络数据传输)。
           进程具有隔离性,因此不同的进程间要通讯就得通过共享的公共区域来交换数据,网卡便是一种共享公共区域。
          (2)客户端(Client)/服务器(Server):客户端是数据发送方,服务器是数据接收方,由于服务器是被动接收数据,不知道客户端什么时候发送数据,因此服务器经常是24小时运行。
        (3)哀求(request)和相应(response):客户端发送给服务器的数据,服务器返回给客户端的数据。
        (4)客户端/服务器交互方式:一问一答(浏览网页)、多问一答(上传文件)、一问多答(下载文件)和多问多答(远程控制、游戏串流(手机毗连电脑玩电脑游戏))。
2 Socket套接字

        Socket便是传输层(操纵体系)提供给步调员的API,有基于TCP和UDP两种范例,原始的API是C语言的,经过JVM包装后,形成了Java的Socket API。
3 UDP数据报套接字

        基于UDP的Socket重要是DatagramSocket,用于发送和接收UDP数据报,作用和操纵方式都类似文件(打开Socket文件(对应网卡),发送/接收数据报,关闭文件)。
构造方法/方法

含义

DatagramSocket()

构造方法,创建一个UDP数据报套接字的Socket,绑定到本机任意一个随机端口(一般用于客户端)

DatagramSocket(int port)

构造方法,创建一个UDP数据报套接字的Socket,绑定到本机指定的端口(一般用于服务端)

void receive(DatagramPacket p)

从套接字接收数据报(如果没有接收到数据报,该方法会阻塞等候)

void send(DatagramPacket p)

从套接字发送数据报包(不会阻塞等候,直接发送)

void close()

关闭数据报套接字

        DatagramPacket是数据报的API,用来构造必要发送和接收的数据。
构造方法/方法

含义

DatagramPacket(byte[] buf, int length)

构造方法,创建一个DatagramPacket用来接收数据报,接收的数据生存在buf,length是数据长度

DatagramPacket(byte[] buf, int offset, int length, SocketAddress address

构造方法,创建一个DatagramPacket用来发送数据报,发送的数据生存在buf,offset是起始下标,length是数据长度,address指定目标主机的IP和端标语

InetAddress getAddress()

从接收的数据报中,获取发送端主机IP地点,或从发送的数据报中,获取接收端主机IP地点

int getPort()

从接收的数据报中,获取发送端主机的端标语,或从发送的数据报中,获

取接收端主机端标语

byte[] getData()

获取数据报中的数据

        而使用InetSocketAddress(InetAddress addr, int port)即可创建一个Socket地点,包罗IP地点和端标语,它是SocketAddress的子类。
        下面利用UDP数据报套接字实现一个回显服务器,回显服务器是指客户端发送什么,服务器就返回什么。
  1. public class UdpServer {
  2.     //回显服务器端口号
  3.     private  final int PORT = 8000;
  4.     public  DatagramSocket datagramSocket = null;
  5.     //通过端口创建udp服务器(本质是打开一个socket文件)
  6.     public UdpServer() throws SocketException {
  7.         datagramSocket = new DatagramSocket(PORT);
  8.     }
  9.     //启动服务器
  10.     public void start() throws IOException {
  11.         System.out.println("服务器已启动");
  12.         //服务器需要24小时启动因此写成死循环
  13.         while(true){
  14.             //1.创建接收请求的数据报(分配字节数组空间)并接收请求
  15.             DatagramPacket datagramPacket = new DatagramPacket(new byte[4096],4096);
  16.             //该方法会一直阻塞等待直到客户端发送请求
  17.             datagramSocket.receive(datagramPacket);
  18.             //2.解析请求构造响应数据报
  19.             String request = new String(datagramPacket.getData(),0,datagramPacket.getLength());
  20.             //3.根据请求构造响应(服务器业务逻辑)
  21.             String response = process(request);
  22.             //要用指定发送地址的构造方法(发送地址可以从接收的请求数据报中获取)
  23.             DatagramPacket datagramPacketResponse = new DatagramPacket(response.getBytes(),response.getBytes().length,
  24.                     datagramPacket.getSocketAddress());
  25.             //4.返回响应
  26.             datagramSocket.send(datagramPacketResponse);
  27.             System.out.printf("[%s:%d] request=%s;response=%s\n",datagramPacket.getAddress().toString(),datagramPacket.getPort(),
  28.                     request,response);
  29.         }
  30.     }
  31.     //回显服务器就是发送什么请求返回什么响应
  32.     private String process(String request) {
  33.         return request;
  34.     }
  35.     public static void main(String[] args) throws IOException {
  36.         //启动服务器
  37.         UdpServer server = new UdpServer();
  38.         server.start();
  39.     }
  40. }
  41. public class UdpClient {
  42.     //创建客户端
  43.     private DatagramSocket socket = null;
  44.     public UdpClient() throws SocketException {
  45.         //客户端一般随机生成端口号
  46.         socket = new DatagramSocket();
  47.     }
  48.     //启动客户端
  49.     public void start() throws IOException {
  50.         Scanner scanner = new Scanner(System.in);
  51.         while(true){
  52.             //1.构造请求
  53.             System.out.print(">");
  54.             String req = scanner.next();
  55.             //构造的请求数据报既要包含数据又要包含地址(IP+端口)
  56.             DatagramPacket request = new DatagramPacket(req.getBytes(),req.getBytes().length,
  57.                     InetAddress.getByName("127.0.0.1"),8000);
  58.             //2.发送请求
  59.             socket.send(request);
  60.             //3.接受响应并解析响应
  61.             DatagramPacket response = new DatagramPacket(new byte[4096],4096);
  62.             socket.receive(response);
  63.             String resp = new String(response.getData(),0,response.getLength());
  64.             //4.将响应返回给用户
  65.             System.out.printf("request=%s,response=%s\n",req,resp);
  66.         }
  67.     }
  68.     public static void main(String[] args) throws IOException {
  69.         UdpClient udpClient = new UdpClient();
  70.         udpClient.start();
  71.     }
  72. }
复制代码
        在服务器代码中,必要注意response.getBytes().length不等于response.length(),前一个方法获取的是字节长度,而后一个方法获取的是字符长度,当字符中有中文时,字节长度和字符长度并不相等,因此不能替换。而字符串“127.0.0.1”是环回IP,表示本机(自己发给自己)。


        根据结果显示,虽然发送的一条哀求hello World,但是却被拆成hello和World两条哀求发送了,这是因为这里使用Scanner类的next()方法,该方法以空格作为结束举行分割,因此发送两条哀求。如果向发送一条,就可以使用nextLine(),该方法以换行符结束。
下篇文章:
网络编程—Socket套接字(TCP)
https://blog.csdn.net/sniper_fandc/article/details/146923783?fromshare=blogdetail&sharetype=blogdetail&sharerId=146923783&sharerefer=PC&sharesource=sniper_fandc&sharefrom=from_link

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

兜兜零元

论坛元老
这个人很懒什么都没写!
快速回复 返回顶部 返回列表