• Android(java)学习笔记20:UDP协议发送数据


    1. UDP协议发送数据:我们总是先运行接收端,再运行发送端
    发送端:

     1 package cn.itcast_02;
     2 
     3 import java.io.IOException;
     4 import java.net.DatagramPacket;
     5 import java.net.DatagramSocket;
     6 import java.net.InetAddress;
     7 /*
     8  * UDP协议发送数据:
     9  * A:创建发送端Socket对象
    10  * B:创建数据,并把数据打包
    11  * C:调用Socket对象的发送方法发送数据包
    12  * D:释放资源
    13  */
    14 public class SendDemo {
    15     public static void main(String[] args) throws IOException {
    16         // 创建发送端Socket对象
    17         // DatagramSocket()
    18         DatagramSocket ds = new DatagramSocket();
    19 
    20         // 创建数据,并把数据打包
    21         // DatagramPacket(byte[] buf, int length, InetAddress address, int port)
    22         // 创建数据
    23         byte[] bys = "hello,udp,我来了".getBytes();//转码
    24         // 长度
    25         int length = bys.length;
    26         // IP地址对象
    27         InetAddress address = InetAddress.getByName("192.168.12.92");
    28         // 端口
    29         int port = 10086;
    30         DatagramPacket dp = new DatagramPacket(bys, length, address, port);
    31 
    32         // 调用Socket对象的发送方法发送数据包
    33         // public void send(DatagramPacket p)
    34         ds.send(dp);
    35 
    36         // 释放资源
    37         ds.close();
    38     }
    39 }

    接收端:

     1 package cn.itcast_02;
     2 
     3 import java.io.IOException;
     4 import java.net.DatagramPacket;
     5 import java.net.DatagramSocket;
     6 import java.net.InetAddress;
     7 
     8 /*
     9  * UDP协议接收数据:
    10  * A:创建接收端Socket对象
    11  * B:创建一个数据包(接收容器)
    12  * C:调用Socket对象的接收方法接收数据
    13  * D:解析数据包,并显示在控制台
    14  * E:释放资源
    15  */
    16 public class ReceiveDemo {
    17     public static void main(String[] args) throws IOException {
    18         // 创建接收端Socket对象
    19         // DatagramSocket(int port)
    20         DatagramSocket ds = new DatagramSocket(10086);
    21 
    22         // 创建一个数据包(接收容器)
    23         // DatagramPacket(byte[] buf, int length)
    24         byte[] bys = new byte[1024];
    25         int length = bys.length;
    26         DatagramPacket dp = new DatagramPacket(bys, length);
    27 
    28         // 调用Socket对象的接收方法接收数据
    29         // public void receive(DatagramPacket p)
    30         ds.receive(dp); // 阻塞式
    31 
    32         // 解析数据包,并显示在控制台
    33         // 获取对方的ip
    34         // public InetAddress getAddress()
    35         InetAddress address = dp.getAddress();
    36         String ip = address.getHostAddress();
    37         // public byte[] getData():获取数据缓冲区
    38         // public int getLength():获取数据的实际长度
    39         byte[] bys2 = dp.getData();
    40         int len = dp.getLength();
    41         String s = new String(bys2, 0, len);
    42         System.out.println(ip + "传递的数据是:" + s);
    43 
    44         // 释放资源
    45         ds.close();
    46     }
    47 }

    这里ds.close():

    Java的内存回收机制,也是要等到资源达到一定限度才开始回收,也是有生命周期的。用close()可以及时回收资源,更加高效.使用close()后就可以及时释放资源,不必非等到最后资源占用完了才开始痛苦的回收过程,而且从良好的编程习惯来说,创建了对象,就应该考虑到用完后就要释放内存资源,要养成一个良好的编程习惯。

    这里首先我们是运行接收端,因为如果不先运行接收端,先运行发送端的话,数据也不会接收到。但是与此同时,如果先运行接收端,可是没有接收到数据,不可能解析数据和显示数据,所以:先运行接收端,后运行发送端,同时我们也定义接收端为阻塞式,(也就是等待数据发送过来)

    UDP发送数据和接收数据图解:

    UDP发送数据和接收数据代码的优化

    UDP协议发送数据:我们总是先运行接收端,再运行发送端
    发送端:

     1 package cn.itcast_03;
     2 
     3 import java.io.IOException;
     4 import java.net.DatagramPacket;
     5 import java.net.DatagramSocket;
     6 import java.net.InetAddress;
     7 
     8 public class SendDemo {
     9     public static void main(String[] args) throws IOException {
    10         // 创建发送端的Socket对象
    11         DatagramSocket ds = new DatagramSocket();
    12 
    13         // 创建数据并打包
    14         byte[] bys = "helloworld".getBytes();
    15         DatagramPacket dp = new DatagramPacket(bys, bys.length,
    16                 InetAddress.getByName("192.168.12.92"), 12345);
    17 
    18         // 发送数据
    19         ds.send(dp);
    20 
    21         // 释放资源
    22         ds.close();
    23     }
    24 }

     接收端:

     1 package cn.itcast_03;
     2 
     3 import java.io.IOException;
     4 import java.net.DatagramPacket;
     5 import java.net.DatagramSocket;
     6 
     7 /*
     8  * 多次启动接收端:
     9  *         java.net.BindException: Address already in use: Cannot bind
    10  *         端口被占用。
    11  */
    12 public class ReceiveDemo {
    13     public static void main(String[] args) throws IOException {
    14         // 创建接收端的Socket对象
    15         DatagramSocket ds = new DatagramSocket(12345);
    16 
    17         // 创建一个包裹
    18         byte[] bys = new byte[1024];
    19         DatagramPacket dp = new DatagramPacket(bys, bys.length);
    20 
    21         // 接收数据
    22         ds.receive(dp);
    23 
    24         // 解析数据
    25         String ip = dp.getAddress().getHostAddress();
    26         String s = new String(dp.getData(), 0, dp.getLength());
    27         System.out.println("from " + ip + " data is : " + s);
    28 
    29         // 释放资源
    30         ds.close();
    31     }
    32 }

    2. 发送端数据来自于键盘录入的案例:(注意这里我们是如何更改上面的代码的)

    发送端:

     1 package cn.itcast_04;
     2 
     3 import java.io.BufferedReader;
     4 import java.io.IOException;
     5 import java.io.InputStreamReader;
     6 import java.net.DatagramPacket;
     7 import java.net.DatagramSocket;
     8 import java.net.InetAddress;
     9 
    10 /*
    11  * 数据来自于键盘录入
    12  * 键盘录入数据要自己控制录入结束。
    13  */
    14 public class SendDemo {
    15     public static void main(String[] args) throws IOException {
    16         // 创建发送端的Socket对象
    17         DatagramSocket ds = new DatagramSocket();
    18 
    19         // 封装键盘录入数据
    20         BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
    21         String line = null;
    22         while ((line = br.readLine()) != null) {
    23             if ("886".equals(line)) { //键盘录入数据终止符定义为:886
    24                 break;
    25             }
    26 
    27             // 创建数据并打包
    28             byte[] bys = line.getBytes();
    29             // DatagramPacket dp = new DatagramPacket(bys, bys.length,
    30             // InetAddress.getByName("192.168.12.92"), 12345);//发送给特定的用户电脑
    31             DatagramPacket dp = new DatagramPacket(bys, bys.length,
    32                     InetAddress.getByName("192.168.12.255"), 12345);//发送给网内所有电脑
    33 
    34             // 发送数据
    35             ds.send(dp);
    36         }
    37 
    38         // 释放资源
    39         ds.close();
    40     }
    41 }

    接收端:

     1 package cn.itcast_04;
     2 
     3 import java.io.IOException;
     4 import java.net.DatagramPacket;
     5 import java.net.DatagramSocket;
     6 
     7 /*
     8  * 多次启动接收端:
     9  *         java.net.BindException: Address already in use: Cannot bind
    10  *         端口被占用。
    11  */
    12 public class ReceiveDemo {
    13     public static void main(String[] args) throws IOException {
    14         // 创建接收端的Socket对象
    15         DatagramSocket ds = new DatagramSocket(12345);
    16 
    17         while (true) {
    18             // 创建一个包裹
    19             byte[] bys = new byte[1024];
    20             DatagramPacket dp = new DatagramPacket(bys, bys.length);
    21 
    22             // 接收数据
    23             ds.receive(dp);
    24 
    25             // 解析数据
    26             String ip = dp.getAddress().getHostAddress();
    27             String s = new String(dp.getData(), 0, dp.getLength());
    28             System.out.println("from " + ip + " data is : " + s);
    29         }
    30 
    31         // 释放资源
    32         // 接收端应该一直开着等待接收数据,是不需要关闭,就好比百度服务器是一直开着着,一样的,接收端好比服务器端
    33         // ds.close();
    34     }
    35 }

    这里我们知道如果这个程序要完成通信的话,我们就必须打开两个界面,一个发送端一个接收端:

    但是现实生活中我们都是在一个界面下发送和接收数据,例如如下的qq聊天界面:

    于是我们这里引入对上面程序的进一步优化:这里就是利用多线程改进程序

     1 package cn.itcast_05;
     2 
     3 import java.io.IOException;
     4 import java.net.DatagramSocket;
     5 
     6 /*
     7  * 通过多线程改进刚才的聊天程序,这样我就可以实现在一个窗口发送和接收数据了
     8  */
     9 public class ChatRoom {
    10     public static void main(String[] args) throws IOException {
    11         DatagramSocket dsSend = new DatagramSocket();
    12         DatagramSocket dsReceive = new DatagramSocket(12306);
    13 
    14         SendThread st = new SendThread(dsSend);
    15         ReceiveThread rt = new ReceiveThread(dsReceive);
    16 
    17         Thread t1 = new Thread(st);
    18         Thread t2 = new Thread(rt);
    19 
    20         t1.start();
    21         t2.start();
    22     }
    23 }
     1 package cn.itcast_05;
     2 
     3 import java.io.IOException;
     4 import java.net.DatagramPacket;
     5 import java.net.DatagramSocket;
     6 
     7 public class ReceiveThread implements Runnable {
     8     private DatagramSocket ds;
     9 
    10     public ReceiveThread(DatagramSocket ds) {
    11         this.ds = ds;
    12     }
    13 
    14     @Override
    15     public void run() {
    16         try {
    17             while (true) {
    18                 // 创建一个包裹
    19                 byte[] bys = new byte[1024];
    20                 DatagramPacket dp = new DatagramPacket(bys, bys.length);
    21 
    22                 // 接收数据
    23                 ds.receive(dp);
    24 
    25                 // 解析数据
    26                 String ip = dp.getAddress().getHostAddress();
    27                 String s = new String(dp.getData(), 0, dp.getLength());
    28                 System.out.println("from " + ip + " data is : " + s);
    29             }
    30         } catch (IOException e) {
    31             e.printStackTrace();
    32         }
    33     }
    34 
    35 }
     1 package cn.itcast_05;
     2 
     3 import java.io.BufferedReader;
     4 import java.io.IOException;
     5 import java.io.InputStreamReader;
     6 import java.net.DatagramPacket;
     7 import java.net.DatagramSocket;
     8 import java.net.InetAddress;
     9 
    10 public class SendThread implements Runnable {
    11 
    12     private DatagramSocket ds;
    13 
    14     public SendThread(DatagramSocket ds) {
    15         this.ds = ds;
    16     }
    17 
    18     @Override
    19     public void run() {
    20         try {
    21             // 封装键盘录入数据
    22             BufferedReader br = new BufferedReader(new InputStreamReader(
    23                     System.in));
    24             String line = null;
    25             while ((line = br.readLine()) != null) {
    26                 if ("886".equals(line)) {
    27                     break;
    28                 }
    29 
    30                 // 创建数据并打包
    31                 byte[] bys = line.getBytes();
    32                 // DatagramPacket dp = new DatagramPacket(bys, bys.length,
    33                 // InetAddress.getByName("192.168.12.92"), 12345);
    34                 DatagramPacket dp = new DatagramPacket(bys, bys.length,
    35                         InetAddress.getByName("192.168.12.255"), 12306);
    36 
    37                 // 发送数据
    38                 ds.send(dp);
    39             }
    40 
    41             // 释放资源
    42             ds.close();
    43         } catch (IOException e) {
    44             e.printStackTrace();
    45         }
    46     }
    47 
    48 }

    执行效果图如下:

  • 相关阅读:
    实用脚本 2 Linux下定时执行脚本
    SSH原理与运用(一):远程登录
    SSH原理与运用(二):远程操作与端口转发
    注册页面
    注册页面2
    CSS實現網頁漸變背景
    SQL Server Log文件“減肥”方法
    发布一个jquery插件在Web下输入密码时提示大写锁定键
    新增流水号
    ftp图片上传
  • 原文地址:https://www.cnblogs.com/hebao0514/p/4513682.html
Copyright © 2020-2023  润新知