Java 网络编程(TCP编程 和 UDP编程)
1. Java 网络编程(TCP编程 和 UDP编程)
文章目录
- 1. Java 网络编程(TCP编程 和 UDP编程)
- 2. 网络编程的概念
- 3. IP 地址
- 3.1 IP地址相关的:域名与DNS
- 4. 端口号(port)
- 5. 通信协议
- 5.1 通信协议相关的:OSI 参考模型
- 5.2 通信协议相关的:TCP / IP 参考模型
- 5.3 补充:OSI 参考模型 与 TCP / IP 参考模型 区别
- 6. 网络编程基础类
- 6.1 InetAddress类
- 6.2 URL 类
- 7. TCP 与 UDP协议
- 7.1 Socket 套接字概述
- 7.2 TCP 与 UDP协议的区别
- 7.3 补充:TCP协议的三次握手(通道建立)
- 7.4 补充:TCP协议的四次挥手(通道关闭)
- 8. TCP协议编程
- 8.1 Socket类概述
- 8.2 ServerSocket 类概述
- 9. 基于TCP协议的编程
- 9.1基于 TCP协议的单向通讯的实现
- 9.2 基于 TCP协议的双向通讯的实现
- 10. 基于 UDP 协议的编程
- 10.1 UDP 协议编程概述
- 10.2 DatagramSocket 类的概述
- 10.3 DatagramPacket 类的概述
- 11. 基于UDP协议的编程通信实现
- 12. 总结:
- 13. 最后:
2. 网络编程的概念
什么是网络编程 ?
网络编程是指利用计算机网络实现程序之间通信的一种编程方式。在网络编程中,程序需要通过网络协议(如 TCP/IP)来进行通信,以实现不同计算机之间的数据传输和共享。
在网络编程中,通常有三个基本要素:
- **IP 地址:**定位网络中某台计算机
- 端口号 port:定位计算机上的某个进程(某个应用)
- **通信协议:**通过IP地址和端口号定位后,如何保证数据可靠高效的传输,这就需要依靠通信协议了。
3. IP 地址
IP 地址用于唯一标识网络中的每一台计算机。在 Internet 上,使用 IPv4 或 IPv6 地址来表示 IP 地址。通常 IPv4 地址格式为 xxx.xxx.xxx.xxx,其中每个 xxx 都表示一个 8 位的二进制数(每一个xxx的取值范围是0-255),组合起来可以表示 2^32 个不同的 IP 地址。
IPv4 地址的总数量是4294967296 个,但并不是所有的 IPv4 地址都可以使用。IPv4 地址被分为网络地址和主机地址两部分,前3个字节用于表示网络(省市区),最后1个字节用于表示主机(家门牌)。而一些 IP 地址被保留或者被私有机构使用,不能用于公网的地址分配。另外,一些 IP 地址被用作多播地址,仅用于特定的应用场景。因此实际上可供使用的 IPv4 地址数量要少于总数量,而且随着 IPv4 地址的逐渐枯竭,IPv6 地址已经开始逐渐普及,IPv6 地址数量更是相当巨大。
IPv6使用16个字节表示IP地址(128位),这样就解决了网络地址资源数量不够的问题。IPv6 地址由 8 组 16 位十六进制数表示,每组之间用冒号分隔,如:3ffe:3201:1401:1280:c8ff:fe4d:db39:1984
本机地址:127.0.0.1,主机名:localhost。
192.168.0.0-192.168.255.255为私有地址,属于非注册地址,专门为组织机构内部使用。
3.1 IP地址相关的:域名与DNS
域名:
IP地址毕竟是数字标识,使用时不好记忆和书写,因此在IP地址的基础上又发展出一种符号化的地址方案,来代替数字型的IP地址。每一个符号化的地址都与特定的IP地址对应。这个与网络上的数字型IP地址相对应的字符型地址,就被称为域名。
目前域名已经成为互联网品牌、网上商标保护必备的要素之一,除了识别功能外,还有引导、宣传等作用。如:www.baidu.com
DNS:
在Internet上域名与IP地址之间是一对一(或者多对一)的,域名虽然便于人们记忆,但机器之间只能互相认识IP地址,它们之间的转换工作称为域名解析,域名解析需要由专门的域名解析服务器来完成,DNS(Domain Name System域名系统)就是进行域名解析的服务器,域名的最终指向是IP。
4. 端口号(port)
在计算机中,不同的应用程序是通过端口号区分的。
端口号是用两个字节(无符号)表示的,它的取值范围是0~65535,而这些计算机端口可分为3大类:
- 公认端口:0~1023。被预先定义的服务通信占用(如:HTTP占用端口80,FTP占用端口21,Telnet占用端口23等)
- 注册端口:1024~49151。分配给用户进程或应用程序。(如:Tomcat占用端口8080,MySQL占用端口3306,Oracle占用端口1521等)。
- 动态/私有端口:49152~65535。
通常情况下,服务器程序使用固定的端口号来监听客户端的请求,而客户端则使用随机端口连接服务器。
IP地址好比每个人的地址(门牌号),端口好比是房间号。必须同时指定IP地址和端口号才能够正确的发送数据。接下来通过一个图例来描述IP地址和端口号的作用。
5. 通信协议
通过计算机网络可以使多台计算机实现连接,位于同一个网络中的计算机在进行连接和通信时需要遵守一定的规则。就像两个人想要顺利沟通就必须使用同一种语言一样,如果一个人只懂英语而另外一个人只懂中文,这样就会造成没有共同语言而无法沟通。
在计算机网络中,这些连接和通信的规则被称为网络通信协议,它对数据的传输格式、传输速率、传输步骤等做了统一规定,通信双方必须同时遵守才能完成数据交换。
在计算机网络中,常用的协议有 TCP、UDP、HTTP、FTP 等。这些协议规定了数据传输的格式、传输方式和传输顺序等细节。其中,TCP(传输控制协议)是一种可靠的面向连接的协议,它提供数据传输的完整性保证;而 UDP(用户数据报协议)则是一种无连接的协议,传输效率高。在网络编程中,需要选取合适的协议类型来实现数据传输。
5.1 通信协议相关的:OSI 参考模型
世界上第一个网络体系结构由IBM公司提出(1974年,SNA),以后其他公司也相继提出自己的网络体系结构如:Digital公司的DNA,美国国防部的TCP/IP等,多种网络体系结构并存,其结果是若采用IBM的结构,只能选用IBM的产品,只能与同种结构的网络互联。
为了促进计算机网络的发展,国际标准化组织ISO(International Organization for Standardization)于1977年成立了一个委员会,在现有网络的基础上,提出了不基于具体机型、操作系统或公司的网络体系结构,称为开放系统互连参考模型,即OSI/RM (Open System Interconnection Reference Model)。OSI模型把网络通信的工作分为7层,分别是物理层、数据链路层、网络层、传输层、会话层、表示层和应用层。
5.2 通信协议相关的:TCP / IP 参考模型
OSI 参考模型的初衷是提供全世界范围的计算机网络都要遵循的统一标准,但是由于存在模型和协议自身的缺陷,迟迟没有成熟的产品推出。TCP/IP协议在实践中不断完善和发展取得成功,作为网络的基础,Internet的语言,可以说没有TCP/IP参考模型就没有互联网的今天。
TCP/IP,即Transmission Control Protocol/Internet Protocol的简写,中译名为传输控制协议/因特网互联协议,是Internet最基本的协议、Internet国际互联网络的基础。
TCP/IP协议是一个开放的网络协议簇,它的名字主要取自最重要的网络层IP协议和传输层TCP协议。TCP/IP协议定义了电子设备如何连入因特网,以及数据如何在它们之间传输的标准。TCP/IP参考模型采用4层的层级结构,每一层都呼叫它的下一层所提供的协议来完成自己的需求,这4个层次分别是:网络接口层、互联网层(IP层)、传输层(TCP层)、应用层。
OSI模型与TCP/IP模型的对应关系如图所示:
5.3 补充:OSI 参考模型 与 TCP / IP 参考模型 区别
- OSI 参考模型是理论上的,而 TCP/IP 参考模型是实践上的。TCP/IP 参考模型被许多实际的协议(如 IP、TCP、HTTP 等)所支持和实现,而 OSI 参考模型则主要是作为理论框架和标准进行研究和讨论。
- OSI 参考模型是由国际标准化组织提出的网络通信协议框架,其中分为 7 层,各层之间明确了功能的划分,从物理层到应用层,逐层向上升,每层只对自己下一层提供服务,并依次封装和解封数据。OSI 参考模型是一种理论上的协议框架,用于描述计算机系统间的通信原理和规范。
- TCP/IP 参考模型(也称互联网参考模型)是实际应用中最广泛的协议框架。它将网络协议划分为 4 层:网络接口层、网络层、传输层和应用层。TCP/IP 参考模型与 OSI 参考模型之间有着相对应的层次结构,但是其中的每一层都是实际存在的协议,而不是纯粹的框架。TCP/IP 参考模型被广泛应用于互联网上,是计算机系统间进行通信的重要基础。
6. 网络编程基础类
6.1 InetAddress类
java.net.IntAddress
类用来封装计算机的IP地址和DNS(没有端口信息),它包括一个主机名和一个IP地址,是j ava对IP地址的高层表示。大多数其它网络类都要用到这个类,包括Socket、ServerSocket、URL、DatagramSocket、DatagramPacket等
常用静态方法:
static InetAddress getLocalHost()
得到本机的InetAddress对象,其中封装了IP地址和主机名static InetAddress getByName(String host)
传入目标主机的名字或IP地址得到对应的InetAddress对象,其中封装了IP地址和主机名(底层会自动连接DNS服务器进行域名解析)
常用实例方法:
public String getHostAddress()
获取IP地址public String getHostName()
获取主机名/域名
编写运行测试:
package day34.com.rainbowsea.javase.net;import java.net.InetAddress;
import java.net.UnknownHostException;/*** java.net.IntAddress类用来封装设计计算机IP地址和DNS(没有端口信息)* 它包括一个主机名和一个地址,是Java对IP地址的高层表示,大多数其它* 网络类都要用到这个类,包括 Socket,ServerSocket,URL.DatagramSocket,DatagramPacket等*/
public class InetAddressTest {public static void main(String[] args) throws UnknownHostException {// 获取本机的IP地址和主机名的封装对象: InetAddressInetAddress inetAddress = InetAddress.getLocalHost();// 获取本机的IP地址String hostAddress = inetAddress.getHostAddress();System.out.println("本机IP地址: " + hostAddress);// 获取本机的主机名String hostName = inetAddress.getHostName();System.out.println("本机的主机名: " + hostName);// 通过域名来获取InetAddress 对象InetAddress inetAddress2 = InetAddress.getByName("www.baidu.com");System.out.println(inetAddress2.getHostName()); // www.baidu.comSystem.out.println(inetAddress2.getHostAddress()); // 36.155.132.3}
}
6.2 URL 类
URL 是统一资源定位符 ,对可以从互联网上得到的资源的位置和访问方法的一种简洁的表示,是互联网上标准资源的地址。互联网上的每个文件都有一个唯一的URL,它包含的信息指出文件的位置以及浏览器应该怎么处理它。
URL由4部分组成:协议、存放资源的主机域名、端口号、资源文件名。如果未指定该端口号,则使用协议默认的端口。例如HTTP协议的默认端口为80。在浏览器中访问网页时,地址栏显示的地址就是URL。
URL标准格式为:<协议>://<域名或IP>:<端口>/<路径> 。其中,<协议>://<域名或IP>是必需的,<端口>/<路径>有时可省略。如:https://www.baidu.com。
为了方便程序员编程,JDK中提供了URL类,该类的全名是java.net.URL,该类封装了大量复杂的涉及从远程站点获取信息的细节,可以使用它的各种方法来对URL对象进行分割、合并等处理。
URL类的构造方法:URL url = new URL(“http://127.0.0.1:8080/oa/index.html?name=zhangsan#tip”);
URL类的常用方法:
- 获取协议:url.getProtocol()
- 获取域名:url.getHost()
- 获取默认端口:url.getDefaultPort()
- 获取端口:url.getPort()
- 获取路径:url.getPath()
- 获取资源:url.getFile()
- 获取数据:url.getQuery()
- 获取锚点:url.getRef()
编写运行测试:
package day34.com.rainbowsea.javase.net;import java.net.MalformedURLException;
import java.net.URL;/*** URL包括四部分:协议:IP地址:端口:资源名称* URL是网络中某个资源的地址,某个资源的唯一地址* 通过URL是可以真实的定位到资源的* 在Java中,Java类库提供了一个URL类,来提供对URL的支持* URL的类的构造方法:* URL url = new URL("url");* URL类的常用方法*/
public class URLTest {public static void main(String[] args) throws MalformedURLException {URL url = new URL("http://www.baidu.com/oa/index.html?name=lihua&passwrod=123#tip");// 获取URL中的信息String protocol = url.getProtocol();System.out.println("协议: " + protocol);// 获取资源路径String path = url.getPath();System.out.println("资源路径: " + path);// 获取默认端口(HTTP协议的默认端口是80,HTTPS的协议端口是:443)int defaultPort = url.getDefaultPort();System.out.println("默认端口: " + defaultPort);// 获取当前的端口int port = url.getPort();System.out.println("当前端口号: " + port);// 获取URL中的IP地址String host = url.getHost();System.out.println("主机地址: " + host);// 获取URL准备传送的数据String query = url.getQuery();System.out.println("需要提交给服务器的数据: " + query);String ref = url.getRef();System.out.println("获取锚点: " + ref);// 获取 资源路径 + 数据String file = url.getFile();System.out.println("获取数据资源文件路径: " + file);}
}
使用URL类的 openStream()
方法可以打开到此URL的连接并返回一个用于从该连接读入的InputStream,实现最简单的网络爬虫。
编写运行测试:
package day34.com.rainbowsea.javase.net;import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.MalformedURLException;
import java.net.URL;public class URLTest2 {public static void main(String[] args) throws IOException {// 使用URL类的openStream()方法可以打开到此URL的连接并返回一个用于从该连接读入的InputStream,实现最简单的网络爬虫URL url = new URL("https://tianqi.qq.com/");InputStream inputStream = url.openStream();BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(inputStream));String s = null;while ((s = bufferedReader.readLine()) != null) {System.out.println(s);}bufferedReader.close();}
}
7. TCP 与 UDP协议
7.1 Socket 套接字概述
我们开发的网络应用程序位于应用层,TCP和UDP属于传输层协议,在应用层如何使用传输层的服务呢?在应用层和传输层之间,则是使用套接Socket来进行分离。
套接字就像是传输层为应用层开的一个小口,应用程序通过这个小口向远程发送数据,或者接收远程发来的数据。而这个小口以内,也就是数据进入这个口之后,或者数据从这个口出来之前,是不知道也不需要知道的,也不会关心它如何传输,这属于网络其它层次工作。
Socket实际是传输层供给应用层的编程接口。Socket就是应用层与传输层之间的桥梁。使用Socket编程可以开发客户机和服务器应用程序,可以在本地网络上进行通信,也可通过Internet在全球范围内通信。
TCP协议和UDP协议是传输层的两种协议。Socket是传输层供给应用层的编程接口,所以Socket编程就分为TCP编程和UDP编程两类。
7.2 TCP 与 UDP协议的区别
TCP协议:
- 使用TCP协议,须先建立TCP连接,形成传输数据通道,似于拨打电话
- 传输前,采用“三次握手”方式,属于点对点通信,是面向连接的,效率低。
- 仅支持单播传输,每条TCP传输连接只能有两个端点(客户端、服务端)。
- 两个端点的数据传输,采用的是“字节流”来传输,属于可靠的数据传输。
- 传输完毕,需释放已建立的连接,开销大,速度慢,适用于文件传输、邮件等。
UDP协议:
- 采用数据报(数据、源、目的)的方式来传输,无需建立连接,类似于发短信。
- 每个数据报的大小限制在64K内,超出64k可以分为多个数据报来发送。
- 发送不管对方是否准备好,接收方即使收到也不确认,因此属于不可靠的。
- 可以广播发送,也就是属于一对一、一对多和多对一连接的通信协议。
- 发送数据结束时无需释放资源,开销小,速度快,适用于视频会议、直播等。
7.3 补充:TCP协议的三次握手(通道建立)
TCP(传输控制协议)是一种面向连接的、可靠的传输层协议。它使用三次握手来建立连接,以确保数据在两个设备之间可靠地传输。
三次握手的过程如下:
- 客户端发送 SYN(同步)数据包。这个数据包包含客户端的初始序列号(ISN)。
- 服务器收到 SYN 数据包后,发送 SYN-ACK(同步确认)数据包。这个数据包包含服务器的初始序列号(ISN)和对客户端 ISN 的确认号(ACK)。
- 客户端收到 SYN-ACK 数据包后,发送 ACK(确认)数据包。这个数据包包含对服务器 ISN 的确认号(ACK)。三次握手完成后,客户端和服务器就可以开始交换数据了。
可以四次,五次握手都可以,握手的目的就是为了,确保连接的建立。至于为什么是三次,因为三次握手就足够可以确保连接的成功建立了。多握几次,也是可以,但是会增加时间,效率上的开销。
三次握手的意义:
三次握手可以确保数据在两个设备之间可靠地传输。它可以防止以下情况的发生:
- 不会丢失:如果没有三次握手,客户端和服务器可能会同时发送数据,导致数据丢失。
- 不会重复:如果没有三次握手,客户端和服务器可能会重复发送数据,导致数据重复。
- 不会乱序:如果没有三次握手,客户端和服务器可能会乱序发送数据,导致数据乱序。
7.4 补充:TCP协议的四次挥手(通道关闭)
使用四次挥手来关闭连接,以确保数据在两个设备之间可靠地传输。
四次挥手的过程如下:
- 客户端发送 FIN(结束)数据包。这个数据包表示客户端已经完成数据传输,并希望关闭连接。
- 服务器收到 FIN 数据包后,发送 ACK(确认)数据包。这个数据包表示服务器已经收到客户端的 FIN 数据包,并同意关闭连接。
- 服务器发送 FIN 数据包。这个数据包表示服务器已经完成数据传输,并希望关闭连接。
- 客户端收到 FIN 数据包后,发送 ACK(确认)数据包。这个数据包表示客户端已经收到服务器的 FIN 数据包,并同意关闭连接。四次挥手完成后,客户端和服务器之间的连接就关闭了。
同理,五次,六次…等等挥手都可以,挥手的:目的就是为了,确保连接的关闭,不丢失数据。至于为什么是四次,因为四次挥手就足够可以确保所有的连接关闭了,数据不丢失。多挥几次,也是可以,但是会增加时间,效率上的开销。
四次挥手的意义:
四次挥手可以确保数据在两个设备之间可靠地传输。它可以防止以下情况的发生:
- 如果没有四次挥手,客户端和服务器可能会同时关闭连接,导致数据丢失。
- 如果没有四次挥手,客户端和服务器可能会重复发送数据,导致数据重复。
- 如果没有四次挥手,客户端和服务器可能会乱序发送数据,导致数据乱序。
8. TCP协议编程
套接字是一种进程间的数据交换机制,利用套接字(Socket)开发网络应用程序早已被广泛的采用,以至于成为事实上的标准。
在网络通讯中,第一次主动发起通讯的程序被称作客户端(Client),而在第一次通讯中等待连接的程序被称作服务端(Server)。一旦通讯建立,则客户端和服务器端完全一样,没有本质的区别。
套接字与主机地址和端口号相关联,主机地址就是客户端或服务器程序所在的主机的IP地址,端口地址是指客户端或服务器程序使用的主机的通信端口。在客户端和服务器中,分别创建独立的Socket,并通过Socket的属性,将两个Socket进行连接,这样客户端和服务器通过套接字所建立连接并使用IO流进行通信。
8.1 Socket类概述
Socket类实现客户端套接字(Client),套接字是两台机器间通信的端点
Socket类构造方法:
public Socket(InetAddress a, int p) // 创建套接字并连接到指定IP地址的指定端口号
Socket类实例方法:
public InetAddress getInetAddress() // 返回此套接字连接到的远程 IP 地址。
public InputStream getInputStream() // 返回此套接字的输入流(接收网络消息)。
public OutputStream getOutputStream() // 返回此套接字的输出流(发送网络消息)。
public void shutdownInput() // 禁用此套接字的输入流
public void shutdownOutput() // 禁用此套接字的输出流。
public synchronized void close() // 关闭此套接字(默认会关闭IO流)。
8.2 ServerSocket 类概述
ServerSocket 类用于实现服务器套接字(Server服务端)。服务器套接字等待请求通过网络传入。它基于该请求执行某些操作,然后可能向请求者返回结果。
ServerSocket构造方法:
public ServerSocket(int port)
ServerSocket实例方法:
public Socket accept() // 侦听要连接到此套接字并接受它。
public InetAddress getInetAddress() // 返回此服务器套接字的本地地址。
public void close() // 关闭此套接字。
9. 基于TCP协议的编程
9.1基于 TCP协议的单向通讯的实现
Java语言的基于套接字编程分为服务端编程和客户端编程,其通信模型如图所示
服务器端实现步骤:
- 创建ServerSocket对象,绑定并监听端口;
- 通过accept监听客户端的请求;
- 建立连接后,通过输出输入流进行读写操作;
- 调用close()方法关闭资源。
服务器端的代码编写如下:
package day34.com.rainbowsea.javase.net.onewaycommunication;import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.ServerSocket;
import java.net.Socket;public class Server {public static void main(String[] args) {ServerSocket serverSocket = null;Socket clientSocket = null;BufferedReader bufferedReader = null;try {// 先启动服务端,启动服务器端后,这个应用肯定要对应一个端口// 创建服务器端套接字对象int port = 8888; // 指明端口serverSocket = new ServerSocket(port);System.out.println("服务器端正在启动,请稍后...");System.out.println("服务器端启动成功,端口号: " + port + ",等待客户端的请求");// 开始接收客户端的请求clientSocket = serverSocket.accept();// 后续代码怎么写一会再说?// 服务端接收消息,所以服务端应该获取输入流bufferedReader = new BufferedReader(new InputStreamReader(clientSocket.getInputStream()));// 开始读String s = null;while ((s = bufferedReader.readLine()) != null) {System.out.println(s);}} catch (IOException e) {throw new RuntimeException(e);} finally {// 关闭服务端套接字try {serverSocket.close();} catch (IOException e) {throw new RuntimeException(e);}try {clientSocket.close();} catch (IOException e) {throw new RuntimeException(e);}try {bufferedReader.close();} catch (IOException e) {throw new RuntimeException(e);}}}
}
客户端实现步骤:
- 创建Socket对象,指定服务端的地址和端口号;
- 建立连接后,通过输入输出流进行读写操作;
- 通过输出输入流获取服务器返回信息;
- 调用close()方法关闭资源。
客户端的代码编写如下:
package day34.com.rainbowsea.javase.net.onewaycommunication;import java.io.BufferedWriter;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.net.InetAddress;
import java.net.Socket;
import java.net.UnknownHostException;
import java.util.Scanner;/*** 现在使用Java中的 Socket实现单向通信,基于 TCP协议,属于TCP编程*/
public class Client {public static void main(String[] args) {Socket clientSocket = null;BufferedWriter bufferedWriter = null;Scanner scanner = new Scanner(System.in);// 创建客户端套接字对象// 需要指定服务器的IP地址,和端口号try {InetAddress localHost = InetAddress.getLocalHost();int port = 8888;clientSocket = new Socket(localHost, port);// 客户端给服务器端发送信息// 客户端你是输出流bufferedWriter = new BufferedWriter(new OutputStreamWriter(clientSocket.getOutputStream()));// 发送信息/* bufferedWriter.write("你好,最近怎么样");bufferedWriter.write("\n");bufferedWriter.write("你收到消息了吗");*/// 循环发送信息/* while (true) {bufferedWriter.write("你好,最近怎么样");bufferedWriter.write("\n");bufferedWriter.write("你收到消息了吗");// 因为使用了缓存机制,需要记得刷新bufferedWriter.flush();// 延迟效果Thread.sleep(1000);}*/// 键盘中输入信息,发送给服务器端while (true) {System.out.println("请输入您要发送的信息: ");// 从键盘上接收的消息String msg = scanner.next();// 把消息发送给服务器端bufferedWriter.write(msg);bufferedWriter.write("\n"); // 换行// 刷新bufferedWriter.flush();}// 因为使用了缓存机制,需要记得刷新//bufferedWriter.flush();} catch (UnknownHostException e) {throw new RuntimeException(e);} catch (IOException e) {throw new RuntimeException(e);} finally {try {clientSocket.close();} catch (IOException e) {throw new RuntimeException(e);}try {bufferedWriter.close();} catch (IOException e) {throw new RuntimeException(e);}scanner.close();}}
}
运行测试:
注意:一定是先启动服务器程序,然后再启动客户端程序,先后顺序千万别弄混了!
9.2 基于 TCP协议的双向通讯的实现
在双向通讯的案例中,客户端需要向服务端发送一张图片,服务端收到客户端发送的图片后,则需要向客户端回复收到图片的反馈。在客户端给服务端发送图片的时候,图片发送完毕必须调用shutdownOutput()方法来关闭socket输出流,否则服务端读取数据就会一直阻塞。
服务器端实现步骤:
- 创建ServerSocket对象,绑定监听端口;
- 通过accept()方法监听客户端请求;
- 使用输入流接收客户端发送的图片,然后通过输出流保存图片
- 通过输出流返回客户端图片收到。
- 调用close()方法关闭资源
服务端的代码编写:
package day34.com.rainbowsea.javase.net.twowaycommunication;import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.BufferedWriter;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.net.ServerSocket;
import java.net.Socket;/*** 双向通信*/
public class TwoWayServer {public static void main(String[] args) {ServerSocket serverSocket = null;Socket clientSocket = null;BufferedInputStream bufferedInputStream = null;BufferedOutputStream bufferedOutputStream = null;BufferedWriter bufferedWriter = null;try {// 创建服务器套接字对象int port = 8888; // 端口号serverSocket = new ServerSocket(port);System.out.println("服务器启动成功,正在接收客户端的请求");// 开始接收客户端的请求clientSocket = serverSocket.accept();// 获取输入流bufferedInputStream = new BufferedInputStream(clientSocket.getInputStream());// 新建输出流,输出读取到的信息,到硬盘当中//new BufferedOutputStream(new FileOutputStream("本地服务器硬盘地址"))bufferedOutputStream = new BufferedOutputStream(new FileOutputStream("./test.jpg"));// 开始读byte[] bytes = new byte[1024];int readCount = 0;while ((readCount = bufferedInputStream.read(bytes)) != -1) {// 把客户端发送过来的图片,保存到本地服务器中bufferedOutputStream.write(bytes, 0, readCount);}// 刷新bufferedOutputStream.flush();// 服务器接收完图片之后给客户端回个话bufferedWriter = new BufferedWriter(new OutputStreamWriter(clientSocket.getOutputStream()));bufferedWriter.write("你发的图片我已经收到了");// 刷新bufferedWriter.flush();} catch (IOException e) {throw new RuntimeException(e);} finally {try {serverSocket.close();} catch (IOException e) {throw new RuntimeException(e);}try {clientSocket.close();} catch (IOException e) {throw new RuntimeException(e);}try {bufferedInputStream.close();} catch (IOException e) {throw new RuntimeException(e);}try {bufferedOutputStream.close();} catch (IOException e) {throw new RuntimeException(e);}try {bufferedWriter.close();} catch (IOException e) {throw new RuntimeException(e);}}}
}
客户端实现步骤:
- 创建socket对象,指明需要连接的服务器地址和端口号;
- 建立连接后,通过输出流向服务器端发送图片;
- 通过输入流获取服务器的响应信息;
- 调用close()方法关闭资源
客户端的代码编写:
所在图片的路径如下:
运行测试:
同样注意:注意:一定是先启动服务器程序,然后再启动客户端程序,先后顺序千万别弄混了!
10. 基于 UDP 协议的编程
10.1 UDP 协议编程概述
在UDP通信协议下,两台计算机之间进行数据交互,并不需要先建立连接,发送端直接往指定的IP和端口号上发送数据即可,但是它并不能保证数据一定能让对方收到,也不能确定什么时候可以送达。
java.net.DatagramSocket类
和 java.net.DatagramPacket类
是使用UDP编程中需要使用的两个类,并且发送端
和接收端
都需要使用这个俩类,并且 发送端与接收端是两个独立的运行程序。
- DatagramSocket:负责接收和发送数据,创建接收端时需要指定端口号。
- DatagramPacket:负责把数据打包,创建发送端时需指定接收端的IP地址和端口。
10.2 DatagramSocket 类的概述
DatagramSocket类作为基于UDP协议的Socket,使用DatagramSocket类可以用于接收和发送数据,同时创建接收端时还需指定端口号。
DatagramSocket的构造方法:
public DatagramSocket() // 创建发送端的数据报套接字
public DatagramSocket(int port) // 创建接收端的数据报套接字,并指定端口号
DatagramSocket的实例方法:
public void send(DatagramPacket p) // 发送数据报。
public void receive(DatagramPacket p) // 接收数据报。
public void close() // 关闭数据报套接字。
10.3 DatagramPacket 类的概述
DatagramPacket类负责把发送的数据打包(打包的数据为byte类型的数组),并且创建发送端时需指定接收端的IP地址和端口。
DatagramPacket的构造方法:
public DatagramPacket(byte buf[], int offset, int length) // 创建接收端的数据报。
public DatagramPacket(byte buf[], int offset, int length, InetAddress address, int port) // 创建发送端的数据报,并指定接收端的IP地址和端口号。
DatagramPacket的实例方法:
public synchronized byte[] getData() // 返回数据报中存储的数据
public synchronized int getLength() // 获得发送或接收数据报中的长度
11. 基于UDP协议的编程通信实现
接收端实现步骤:
- 创建DatagramSocket对象(接收端),并指定端口号;
- 创建DatagramPacket对象(数据报);
- 调用receive()方法,用于接收数据报;
- 调用close()方法关闭资源
接收端的代码编写:
package day34.com.rainbowsea.javase.net.udpcommunication;import java.net.DatagramPacket;
import java.net.DatagramSocket;/*** UDP编程,接收端*/
public class Receive {public static void main(String[] args) throws Exception {// 创建 UDP的 Socket 套接字DatagramSocket datagramSocket = new DatagramSocket(8888);byte[] bytes = new byte[1024 * 64];// 准备一个包,这个包接收发送方的信息DatagramPacket datagramPacket = new DatagramPacket(bytes, bytes.length);// 程序执行到这里,停下来,等待发送方的发送datagramSocket.receive(datagramPacket);// 程序执行到这里说明,已经完全将发送方发送的数据接收到了// 从包中取出来数据String msg = new String(bytes, 0, datagramPacket.getLength());System.out.println("接收到发送方发过来的消息: " + msg);datagramSocket.close();}
}
发送端的代码编写:
package day34.com.rainbowsea.javase.net.udpcommunication;import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;/*** UDP编程, 发送端*/
public class Send {public static void main(String[] args) throws Exception {// 创建一个 UDP的Socket 套接字DatagramSocket datagramSocket = new DatagramSocket();// 创建包byte[] bytes = "RainbowSea".getBytes();DatagramPacket datagramPacket = new DatagramPacket(bytes, 0, bytes.length, InetAddress.getLocalHost(), 8888);// 发送消息,将封装到包(datagramPacket) 中的信息发送过去datagramSocket.send(datagramPacket);datagramSocket.close();}
}
运行测试:注意:先启动接收端,再启动发送端 。
12. 总结:
Java SE 中的网络编程主要还是理解:网络协议:TCP协议和UDP协议,特别是 TCP协议的三次握手,和四次挥手。
而Java SE 的网络编程,这些我们后续的 Tomcat Web 框架当中都是封装好了的,并不需要我们真的自己重写这些底层的方法。我们只需要调用就好了。所以关于这部分的内容大家了解即可。
13. 最后:
限于自身水平,其中存在的错误,希望大家给予指教,韩信点兵——多多益善,谢谢大家,江湖再见,后会有期 !!!
相关文章:
Java 网络编程(TCP编程 和 UDP编程)
1. Java 网络编程(TCP编程 和 UDP编程) 文章目录 1. Java 网络编程(TCP编程 和 UDP编程)2. 网络编程的概念3. IP 地址3.1 IP地址相关的:域名与DNS 4. 端口号(port)5. 通信协议5.1 通信协议相关的…...
STM32 | 看门狗+RTC源码解析
点击上方"蓝字"关注我们 作业 1、使用基本定时7,完成一个定时喂狗的程序 01、上节回顾 STM32 | 独立看门狗+RTC时间(第八天)02、定时器头文件 #ifndef __TIM_H#define __TIM_H#include "stm32f4xx.h"void Tim3_Init(void);void Tim7_Init(void);…...
filebeat,kafka,clickhouse,ClickVisual搭建轻量级日志平台
springboot集成链路追踪 springboot版本 <parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.6.3</version><relativePath/> <!-- lookup parent from…...
Django实战项目之进销存数据分析报表——第一天:Anaconda 环境搭建
引言 Anaconda是一个流行的Python和R语言的发行版,它包含了大量预安装的数据科学、机器学习库和科学计算工具。使用Anaconda可以轻松地创建隔离的环境,每个环境都可以有自己的一套库和Python版本,非常适合多项目开发。本文将指导你如何安装A…...
Linux部署Prometheus+Grafana
【Linux】PrometheusGrafana 一、Prometheus(普罗米修斯)1、Prometheus简述2、Prometheus特点3、Prometheus生态组件4、Prometheus工作原理 二、部署Prometheus1、系统架构2、部署Prometheus3、修改配置文件4、配置系统启动文件 三、部署 Node Exporter …...
【视频讲解】神经网络、Lasso回归、线性回归、随机森林、ARIMA股票价格时间序列预测|附代码数据
全文链接:https://tecdat.cn/?p37019 分析师:Haopeng Li 随着我国股票市场规模的不断扩大、制度的不断完善,它在金融市场中也成为了越来越不可或缺的一部分。 【视频讲解】神经网络、Lasso回归、线性回归、随机森林、ARIMA股票价格时间序列…...
低代码前端框架Amis全面教程
什么是Amis? 1.1 Amis的基本概念 Amis是一个基于JSON配置的前端低代码框架,由百度开源。它允许开发者通过简单的JSON配置文件来生成复杂的后台管理页面,从而大大减少了前端开发的工作量。Amis的核心理念是通过配置而非编码来实现页面的构建…...
Windows 如何安装和卸载 OneDrive?具体方法总结
卸载 OneDrive 有人想问 OneDrive 可以卸载吗?如果你不使用当然可以卸载,下面是安装和卸载 OneDrive 中的卸载应用具体操作步骤: 卸载 OneDrive 我们可以从设置面板中的应用选项进行卸载,打开设置面板之后选择应用,然…...
c# .net core中间件,生命周期
某些模块和处理程序具有存储在 Web.config 中的配置选项。但是在 ASP.NET Core 中,使用新配置模型取代了 Web.config。 HTTP 模块和处理程序如何工作 官网地址: 将 HTTP 处理程序和模块迁移到 ASP.NET Core 中间件 | Microsoft Learn 处理程序是…...
Spring后端框架复习总结
之前写的博客太杂,最近想把后端框架的知识点再系统的过一遍,主要是Spring Boot和Mybatis相关,带着自己的理解使用简短的话把一些问题总结一下,尤其是开发中和面试中的高频问题,基础知识点可以参考之前写java后端专栏,这篇不再赘述。 目录 Spring什么是AOP?底层原理?事务…...
基于Llama Index构建RAG应用
前言 Hello,大家好,我是GISer Liu😁,一名热爱AI技术的GIS开发者,本文参与活动是2024 DataWhale AI夏令营;😲 在本文中作者将通过: Gradio、Streamlit和LlamaIndex介绍 LlamaIndex 构…...
SSLRec代码分析
文章目录 encoder-models-general_cfautocf.py data_utilsdata_handler_general_cf.py输入输出说明使用方法 trainertuner.py encoder-models-general_cf autocf.py import torch as t # 导入PyTorch并重命名为t from torch import nn # 从PyTorch导入神经网络模块 import …...
第四节shell条件测试(1)(2)
一,命令执行结果判定 &&在命令执行后如果没有任何报错时会执行符号后面的动作 ||在命令执行后如果命令有报错会执行符号后的动作 示例: vim lee.sh #!/bin/bash ls /mnt/file &> /dev/null &&{echo /mnt/filr is not existecho no }||{echo /mnt/fi…...
申请https证书的具体流程
申请HTTPS证书的具体流程通常涉及以下步骤,不过请注意,具体细节可能因不同的证书颁发机构(CA)而有所差异: 1、确定证书类型: 证书类型:根据需求选择合适的SSL证书类型。常见的有DV(…...
IP溯源工具--IPTraceabilityTool
工具地址:xingyunsec/IPTraceabilityTool: 蓝队值守利器-IP溯源工具 (github.com) 工具介绍: 在攻防演练期间,对于值守人员,某些客户要求对攻击IP都进行分析溯源,发现攻击IP的时候,需要针对攻击IP进行分析…...
字节抖音电商 后端开发岗位 一面
笔者整理答案,以供参考 自我介绍 项目(20分钟) RocketMQ延时消息的底层实现 回答: 延时消息的实现主要依赖于RocketMQ中的定时任务机制。消息被发送到Broker时,会先存储在一个特定的延时消息队列中。Broker会定时扫…...
前端开发日记——在MacBook上配置Vue环境
前言 大家好,我是来自CSDN的寄术区博主PleaSure乐事。今天是开始学习vue的第一天,我使用的编译器是vscode,浏览器使用的是谷歌浏览器,后续会下载webstorm进行使用,当前学习阶段使用vscode也是可以的,不用担…...
测试开发面经总结(三)
TCP三次握手 TCP 是面向连接的协议,所以使用 TCP 前必须先建立连接,而建立连接是通过三次握手来进行的。 一开始,客户端和服务端都处于 CLOSE 状态。先是服务端主动监听某个端口,处于 LISTEN 状态 客户端会随机初始化序号&…...
开始构建我们自己的大语言模型:数据处理部分
关注本专栏(NLP简论:手搓大语言模型实践) 继续学习从头编写、训练自己的大语言模型。 接上集,本章我们将深入说一下大语言模型数据处理部分的细节,并直接提供本部分的完整代码。 【配套资源】 暂时的词汇表࿱…...
springboot系列十: 自定义转换器,处理JSON,内容协商
文章目录 自定义转换器基本介绍应用实例查看源码注意事项和细节 处理JSON需求说明应用实例 内容协商基本介绍应用实例debug源码优先返回xml注意事项和细节 ⬅️ 上一篇: springboot系列九: 接收参数相关注解 🎉 欢迎来到 springboot系列十: 自定义转换器,…...
C++(new与delete操作符)
C中的new与delete new 与 delete定位new表达式 new 与 delete 在C中需要动态申请内存空间时需要使用 new 与 delete 这两个操作符 #include <iostream> using namespace std; int main() {int* p1 new int;//开辟一块int类型大小的空间给p1int* p2 new int(1);//开辟…...
STM32智能工业自动化监控系统教程
目录 引言环境准备智能工业自动化监控系统基础代码实现:实现智能工业自动化监控系统 4.1 数据采集模块 4.2 数据处理与控制模块 4.3 通信与网络系统实现 4.4 用户界面与数据可视化应用场景:工业自动化与管理问题解决方案与优化收尾与总结 1. 引言 智能…...
WPF设置欢迎屏幕,程序启动过度动画
当主窗体加载时间过长,这时候基本都会想添加一个等待操作来响应用户点击,提高用户体验。下面我记录两个方法,一点拙见,仅供参考。 方法1:在App类中使用SplashScreen类。 protected override void OnStartup(StartupEventArgs e)…...
Flink实时开发添加水印的案例分析
在Flink中,处理时间序列数据时,通常需要考虑事件时间和水印(watermarks)的处理。以下是修改前后的代码对比分析: 修改前的代码: val systemDS unitDS.map(dp > {dp.setDeviceCode(DeviceCodeEnum.fro…...
收银系统源码-线上商城diy装修
线下线上一体化收银系统越来越受门店重视,尤其是连锁多门店,想通过线下线上相互带动,相互引流,提升门店营业额。商城商城如何装修呢? 1.收银系统开发语言 核心开发语言: PHP、HTML5、Dart后台接口: PHP7.3后合管理网…...
Linux中nohup(no hang up)不挂起,用于在系统后台不挂断地运行命令,即使退出终端也不会影响程序的运行。
nohup的英文全称是 no hang up,即“不挂起”。这个命令在Linux或Unix系统中非常有用,主要用于在系统后台不挂断地运行命令,即使退出终端也不会影响程序的运行。默认情况下(非重定向时),nohup会将输出写入一…...
【.NET全栈】ASP.NET开发Web应用——站点导航技术
文章目录 前言一、站点地图1、定义站点地图文件2、使用SiteMapPath控件3、SiteMap类4、URL地址映射 二、TreeView控件1、使用TreeView控件2、以编程的方式添加节点3、使用TreeView控件导航4、绑定到XML文件5、按需加载节点6、带复选框的TreeView控件 三、Menu控件1、使用Menu控…...
docker 容器内部UI映射host
方法有很多, 目前我总计一个我自己尝试成功的方法,通过xpra。 Xpra可以看作是screen或tmux的图形版本,支持远程X11应用程序的显示和交互。 在远程服务器上,安装Xpra: sudo apt-get install xpra启动Xpra服务器会话&…...
数仓面试题——DWS层新增维度字段需求
前言 在数据仓库开发中,数据仓库的设计和维护一直是一个备受关注的话题。随着业务需求的不断变化,数据仓库的结构也需要随之调整。 面试过程中,多次被提问:当DWS构建好后,突然来了一个新的需求,需要添加某个…...
Qt实现MDI应用程序
本文记录Qt实现MDI应用程序的相关操作实现 目录 1.MDM模式下窗口的显示两种模式 1.1TabbedView 页签化显示 1.2 SubWindowView 子窗体显示 堆叠cascadeSubWindows 平铺tileSubWindows 2.MDM模式实现记录 2.1. 窗体继承自QMainWindow 2.2.增加组件MdiArea 2.3.定义统一…...
独立商城网站建设/容易被百度收录的网站
Range分区表建表语句如下,其中分区键必须和id构成主键和唯一键CREATE TABLE test1 (id char(32) COLLATE utf8mb4_unicode_ci NOT NULL COMMENT 自增主键(guid),create_time timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 创建时间,partition_key int(8) N…...
网页在线客服系统代码/青岛seo培训
做项目的时候,或多或少需要和其他外部系统或者接口进行数据交互,有些是单向的获取,有些可能是修改状态后再写回去,不管如何,这个都可以称之为数据同步操作,如人员信息同步、业务数据同步、第三方接口数据同…...
网站备案 费用/百度浏览器手机版
Visual Studio Code是Microsoft开发的一款开源免费的现代化轻量级代码编辑器,它体积小、启动快、系统内存占用率低非常适合搭建IDE。VSCode不仅支持几乎所有主流的开发语言的语法高亮、智能代码补全、自定义热键、括号匹配、代码片段、代码对比 Diff、GIT 等特性&am…...
南昌网站建设公司机构/北京做网站的公司排行
1、Timestamp(long)转成日期 Timestamp timestamp new Timestamp(System.currentTimeMillis()); LocalDateTime localDateTime timestamp.toLocalDateTime(); //之后就可以通过java8 新API操作时间了 localDateTime.toLocalDate(); Date da…...
网站建设 电商/关键词排名技巧
今天,我们要来分享的是计算机编程中多任务的处理方式。我们都知道,计算机中的任务执行是由CPU进行的,它以强大的计算能力极其任务处理能力,进行着大量的多任务的处理。简单来讲,计算机中的多任务处理有两种执行方式&am…...
wordpress无限登录密码/百度竞价托管代运营
JVM内存模型总体架构图 程序计数器 多线程时,当线程数超过CPU数量或CPU内核数量,线程之间就要根据时间片轮询抢夺CPU时间资源。因此每个线程有要有一个独立的程序计数器,记录下一条要运行的指令。线程私有的内存区域。如果执行的是JAVA方法&a…...