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 }

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

发送端:

 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 bind10  *         端口被占用。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     @Override15     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     @Override19     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 }

执行效果图如下:

更多相关文章

  1. android 系统internt
  2. android上的漏洞挖掘方式和Fuzz
  3. Android(安卓)BroadcastReceiver 简介
  4. 如何查看USB方式连接Android设备的外接设备信息。
  5. android 访问SD卡的方法
  6. android 检测网络或wifi是否开启
  7. android 使用post方法请求网址
  8. Android之Bitmap学习总结
  9. android线程调度工具类SchedulerUtils

随机推荐

  1. Android(安卓)Fatal signal 6 (SIGABRT)
  2. Android客制化------在设置中加入RAM fla
  3. Android插件—H.A.X.M(硬件加速执行管理器
  4. Android中WebView与Javascript的交互
  5. Android(安卓)N Preview前瞻
  6. Unity 游戏框架搭建 (十二) 简易AssetBun
  7. SDK-IOS 关于SDK接入 MAC平台笔记
  8. Android(安卓)壁纸设置 总结
  9. Android(安卓)RecycleView(二)——添加分
  10. Android(安卓)国际化之多语言切换