前言

继续今天我们的Java网络编程——TCP和UDP通信

一、TCP和UDP概述

传输层通常以TCP和UDP协议来控制端点与端点的通信

  TCP UDP
协议名称 传输控制协议 用户数据包协议
是否连接 面向连接的协议。数据必须要建立连接 无连接的协议,每个数据报中都给出完整的地址信息,因此不需要事先建立发送方和接受方的连接
是否可靠 可靠协议。确保收方完全地获取发送方所发送的全部数据 不可靠协议。发送方所发送的数据报并不一定以相同的次序到达接收方。
可以传输的数据大小 传输数据大小不受限制。一旦连接建立,双方可以按统一的格式传输大的数据 传输数据时是有大小限制的。每个被传输的数据报必须限定在64KB之类
数据传输方式 IO流 DatagramPacket

二、UDP

1.UDP通信概述

UDP协议是一种对等通信的实现,发送方只需要接受方的IP(地址)和Port(端口),就可以直接向它发送数据,不需要线连接。每个程序都可以作为服务器,也可以作为客户端。UDP是一种无连接的传输协议,每个数据报的大小限定在64KB以内。数据报是一个在网络上发送的独立信息,它的到达。到达时间以及内容本身等都不能得到保证。这种传输方式是无序的,也不能确保绝对的安全可靠,但它很简单也具有较高的效率。
使用UDP协议进行数据传输是,需要将需要传输数据定义为数据报(DatagramPaket),在数据报中指明数据所要到达Socket(主机地址和端口号),然后再将数据报发送出去。实例化DatagramPacket时使用参数port和没有使用参数port的区别在与,提供port的一方可以让别人主动发送消息过来,而没有参数port的则会在发送消息时自动绑定一个本地没有使用的端口。在接收到发送的数据报(DatagramPaket)时,不仅可以获取数据,还可以获得发送方的IP和Port,这样就可以向发送方发送数据,因此,本质上二者是对等的。

2.UDP通信特点

1、UDP是一种无连接的协议,每个数据报都是一个独立的信息,包括完整的原地址或目的地址,它在网络上任何可能的路径传往目的地,因此能否到达目的地,到达目的地的时间以及内容的正确性都是不能被保证的。
2、UDP不属于连接型协议,因而具有资源消耗小,处理速度快的优点,所以通常音频视频和普通数据在传输时使用UDP较多,因为它们即使偶尔丢一两个数据包,也不会对接收结果产生太大的影响。
代码如下(示例):

3.UDP通信传输实现的基石

UDP通信的Socket使用DatagramSocket类实现,数据报使用DatagramPaket实现

3.1、DatagramPake常用方法

InetAddress getAddress() 得到发送方IP地址
int getPort() 得到发送方的端口号
byte[] getData() 返回接收缓冲区,这是一个byte[]
int getLength() 接收字节的真实大小,通常用于从byte[]中提取出有效数据
int getOffset() 返回将要发送或则接收的数据偏移量

3.2、DatagramSocket常用方法

DatagramSocket() 空构造函数
DatagramSocket(int port) 指定通信端口
void receive(DatagramPaket p) 接收数据报
void send(DatagramPaket p) 发送数据报
void close() 关闭Socket

4.UDP通信实现原理

无论一个UDP通信程序的功能多么功能齐全,程序多么复杂,七基本结构都是一样的,都包括以下四个基本步骤
1、在接收端指定一个端口号来创建DatagramSocket,然后创建一个接收数据报(DatagramPaket),使用recevie方法等待发送方请求报文,这将阻塞服务器线程
2、在发送方创建一个DatagramSocket,使用接收方的IP和端口来创建发送数据报(DatagramPaket),使用send方法发送。现在接收方的recevie方法被唤醒,同时会将发送方的数据报内容填充到接收方的DatagramPaket中。
3、接收方从发送方的数据报中获得发送方的IP和端口,使用它们构造一个发送数据报,然后发送给发送方,这样就实现了发送方和接收方的通信
4、在通信完成后,在客服端和服务端中分别关闭Socket

5.UDP通信原理(代码实现)

代码如下(发送端):

import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress; public class SendMessage { public static void main(String[] args) throws Exception {
// 创建发送端Socket对象
DatagramSocket sendSocket = new DatagramSocket();
// 准备需要发送的数据
String message = "hello";
// 创建一个缓冲区
byte[] messageByte = message.getBytes();
// 获取缓冲区中数据的真实长度
int messageLen = message.length();
// InetAddress实例化获取本机通信地址
InetAddress address = InetAddress.getLocalHost();
// 设置通信端口号
int port = 12345;
// 打包数据
DatagramPacket sendPacket = new DatagramPacket(messageByte, messageLen, address, port);
// 发送数据
sendSocket.send(sendPacket);
// 发送端等待接收端成功接收信息后返回的回应
// 创建一个缓冲区,容量尽量设置大一点因为不知道发送过来的信息有多大
byte[] recevieByte = new byte[1024*10];
int len = recevieByte.length;
// 接收数据报
DatagramPacket receivePacket = new DatagramPacket(recevieByte, len);
// 接收数据
sendSocket.receive(receivePacket);
// 获取接收端发送过来的真实长度以及数据
byte[] data = receivePacket.getData();
int length = receivePacket.getLength();
String receiveData = new String(data,0,length);
// 获取发送者的IP
address = receivePacket.getAddress();
String ip = address.getHostAddress();
System.out.println("接收来自:"+ip+"的数据,内容是:"+receiveData);
// 关闭资源
sendSocket.close();
}
}

代码如下(接收端):

import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress; public class RecevieMessage { public static void main(String[] args) throws Exception {
// 接收消息的端口(与发送端保持一致)
int port = 12345;
DatagramSocket recevieSocket = new DatagramSocket(port);
// 设置缓冲区接收发过来的信息
byte[] receiveByte = new byte[1024*10];
int len = receiveByte.length;
// 接收数据报
DatagramPacket receviePacket = new DatagramPacket(receiveByte, len);
// 接收数据
recevieSocket.receive(receviePacket);
// 获取实际接收到的数据及其大小
byte[] data = receviePacket.getData();
int length = receviePacket.getLength();
String receiveData = new String(data,0,length);
// 获取发送者的IP
InetAddress address = receviePacket.getAddress();
String ip = address.getHostAddress();
System.out.println("接收来自:"+ip+"的数据,内容是:"+receiveData);
// 接收端接收到信息后发送一条确认接收的消息到发送端
String message = "OK";
byte[] messageByte = message.getBytes();
int messageLength = message.length();
// 从已收到的数据报中获取IP和port
address = receviePacket.getAddress();
int port1 = receviePacket.getPort();
//构造新数据报
DatagramPacket sendPacket = new DatagramPacket(messageByte, messageLength, address, port1);
// 发送数据
recevieSocket.send(sendPacket);
// 关闭资源
recevieSocket.close();
}
}

发送端输出结果:

接收端输出结果:

总结

**注意:**在发送端与接收端启动测试时最好最好先启动接收端,因为这样才能确保信息能发送出去,接收端鞥接收到信息。

Java网络编程UDP通信原理的更多相关文章

  1. JAVA网络编程--UDP通信

    首先网络传输数据需了解例如以下三点 1.找到对方IP 2.数据要发送到对方指定的应用程序上,为了标识这些应用程序,所以给这些网络应用程序用数字进行了标识.为了方便称呼这个数字,叫做port,逻辑por ...

  2. java网络编程Socket通信详解

    Java最初是作为网络编程语言出现的,其对网络提供了高度的支持,使得客户端和服务器的沟通变成了现实,而在网络编程中,使用最多的就是Socket.像大家熟悉的QQ.MSN都使用了Socket相关的技术. ...

  3. java 网络编程 UDP TCP

    网络编程 网络编程主要用于解决计算机与计算机(手机.平板..)之间的数据传输问题. 网络编程: 不需要基于html页面就可以达到数据之间的传输. 比如: feiQ , QQ , 微信....网页编程: ...

  4. UNIX网络编程——Socket通信原理和实践

    我们深谙信息交流的价值,那网络中进程之间如何通信,如我们每天打开浏览器浏览网页时,浏览器的进程怎么与web服务器通信的?当你用QQ聊天时,QQ进程怎么与服务器或你好友所在的QQ进程通信?这些都得靠so ...

  5. java网络编程(UDP详解)

    UDP详解 一,TCP/IP协议栈中,TCP协议和UDP协议的联系和区别? 联系: TCP和UDP是TCP/IP协议栈中传输层的两个协议,它们使用网络层功能把数据包发送到目的地,从而为应用层提供网络服 ...

  6. Java网络编程——UDP聊天程序

    UDP简介 UDP协议的全称是用户数据报,在网络中它与TCP协议一样用于处理数据报.在OSI模型中,UDP位于第四层--传输层,处于IP协议额上一层.UDP有不提供数据报分组.组装以及不能对数据报排序 ...

  7. JAVA网络编程TCP通信

    Socket简介: Socket称为"套接字",描述IP地址和端口.在Internet上的主机一般运行多个服务软件,同时提供几种服务,每种服务都打开一个Socket,并绑定在一个端 ...

  8. Java网络编程Socket通信

        TCP(Transmission Control Protocol 传输控制协议)是一种面向连接的.可靠的.基于字节流的传输层通信协议     UDP (User Datagram Proto ...

  9. Java网络编程(UDP协议:接收端)

    package WebProgramingDemo; import java.io.IOException; import java.net.DatagramPacket; import java.n ...

随机推荐

  1. XV6学习(11)Lab thread: Multithreading

    代码放在github上. 这一次实验感觉挺简单的,特别是后面两个小实验.主要就是对多线程和锁进行一个学习. Uthread: switching between threads 这一个实验是要实现一个 ...

  2. Educational Codeforces Round 17

    Educational Codeforces Round 17 A. k-th divisor 水题,把所有因子找出来排序然后找第\(k\)大 view code //#pragma GCC opti ...

  3. Codeforces Round #680 (Div. 2, based on Moscow Team Olympiad) D. Divide and Sum (思维,数学,逆元)

    题意:有一个长度为\(2n\)数组,从中选分别选\(n\)个元素出来组成两个序列\(p\)和\(q\),(\(p\)和\(q\)中只要有任意一个元素在\(a\)的原位置不同,就算一个新的情况),选完后 ...

  4. Codeforces Round #672 (Div. 2) A. Cubes Sorting (思维)

    题意:有一长度为\(n\)的一组数,每次可以交换两个数的位置,问能否在\(\frac{n*(n-1)}{2}-1\)次操作内使得数组非递减. 题解:不难发现,只有当整个数组严格递减的时候,操作次数是\ ...

  5. Codeforces Round #531 (Div. 3) B. Array K-Coloring (结构体排序)

    题意:给你\(n\)个数字,用\(k\)种颜色给他们涂色,要求每个数字都要涂,每种颜色都要用,相同的数字不能涂一样的颜色. 题解:用结构体读入每个数字和它的位置,然后用桶记录每个数字出现的次数,判断是 ...

  6. VMX - block by NMI和 NMI unblockinig due to IRET 之间的关系

    相关SDM章节: 27.2.3- Information About NMI Unblocking Due to IRET 最近收到同事发来的一个问题,即: VMCS 中的 Guest Interru ...

  7. 文件的读写(cpp)

    文件的读写(cpp) c++中要进行文件的读入,首先要包含一个头文件 fstream . 输出到文件 为打开一个可供输出的文件需要定义一个ofstream 对象并将文件名传入: std::ofstre ...

  8. httprunner(8)用例调用-RunTestCase

    前言 一般我们写接口自动化的时候,遇到复杂的逻辑,都会调用API方法来满足前置条件,Pytest的特性是无法用例之间相互调动的,我们一般只调用自己封装的API方法. 而httprunner支持用例之间 ...

  9. Linux-字符处理命令

    目录 1.sort(排序) 2.uniq(不相邻的两行重复不会去除) 3.cut(取列,截取字段) 4.wc(统计行.单词.字符数) 1.sort(排序) 选项: -t # 指定分隔符 -k # 指定 ...

  10. struct 和 class的区别

    struct和class如果按照在C的时代,还是有很大差别的. c中struct的定义如下: struct  结构名 {   成员表 }: 因为struct是一种数据类型,那么就肯定不能定义函数,所以 ...