Socket通信
优质博文IT-BLOG-CN
一、简介
Socket
套接字:描述了计算机的IP
地址和端口,运行在计算机中的程序之间采用socket
进行数据通信。通信的两端都有socket
,它是一个通道,数据在两个socket
之间进行传输。socket
把复杂的TCP/IP
协议族隐藏在socket
接口后面,对程序员来说,只要用好socket
相关的函数,就可以完成数据通信。
TCP
提供了流stream
和数据报datagram
两种通信机制,所以套接字也分为流套接字和数据报套接字:
【1】流套接字的类型是SOCK_STREAM
,它提供的是一个有序、可靠、双向字节流的连接,因此发送的数据可以确保不会丢失、重复或乱序到达,而且它还有出错后重新发送的机制。
【2】数据报套接的类型是SOCK_DGRAM
,它不需要建立和维持一个连接,采用UDP/IP
协议实现。它对可以发送的数据的长度有限制,数据报作为一个单独的网络消息被传输,它可能会丢失、复制或错乱到达,UDP
不是一个可靠的协议,但是它的速度比较高,因为它不需要建立和维持连接。
常见面试题:TCP与UDP的区别
TCP(传输控制协议)和UDP(用户数据报协议)是两种常见的互联网传输协议,它们在数据传输方面有以下区别:
【1】连接性: TCP
是面向连接的协议,而UDP
是无连接的协议。TCP
在传输数据之前需要先建立连接,而UDP
则直接发送数据包。
【2】可靠性: TCP
提供可靠的数据传输,它使用确认、重传和流量控制等机制来确保数据的完整性和顺序性。而UDP
不提供可靠性保证,它只是简单地将数据包发送出去,无法保证数据的到达和顺序。
【3】速度: 由于TCP
提供了可靠性保证,它的传输速度相对较慢。而UDP
没有额外的机制,传输速度相对较快。
【4】数据量限制: TCP
没有固定的数据包大小限制,可以处理任意大小的数据。而UDP
的数据包大小有限制,最大为64KB
。
【5】适用场景: TCP
适用于对数据传输可靠性要求较高的场景,例如文件传输、网页浏览等。而UDP
适用于对实时性要求较高,但对数据可靠性要求相对较低的场景,例如音视频流媒体、实时游戏等。
总结来说,TCP
提供可靠的、面向连接的数据传输,适用于对数据完整性和顺序性要求较高的场景。而UDP
提供快速的、无连接的数据传输,适用于对实时性要求较高的场景。
二、Socket通信过程
【1】服务端程序将一个套接字绑定到指定的ip
地址和端口,并通过此套接字等待和监听客户的连接请求。
【2】客户程序向服务端程序绑定的地址和端口发出连接请求。
【3】服务端接受连接请求。
【4】客户端和服务端通过读写套接字进行通信。
TCP链接
服务端工作流程:
【1】创建服务端的socket
。
【2)把服务端用于通信的地址和端口绑定到socket
上。
【3】把socket
设置为监听模式。
【4】接受客户端的连接。
【5】与客户端通信,接收客户端发过来的报文后,回复处理结果。
【6)不断的重复第5)步,直到客户端断开连接。
【7】关闭socket
,释放资源。
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;public class Server {public static void main(String[] args) {try {// 创建ServerSocket对象,指定监听的端口号ServerSocket serverSocket = new ServerSocket(45442);System.out.println("等待客户端连接...");// 监听客户端的连接请求Socket clientSocket = serverSocket.accept();System.out.println("客户端已连接");// 获取输入流和输出流 输入流和输出流是通过socket对象来进行数据传输的。BufferedReader in = new BufferedReader(new InputStreamReader(clientSocket.getInputStream()));PrintWriter out = new PrintWriter(clientSocket.getOutputStream(), true);String message;while (true) {// 读取客户端发送的信息message = in.readLine();if (message.equalsIgnoreCase("exit")) {// 如果接收到终止标志,退出循环break;}System.out.println("收到客户端消息:" + message);// 发送响应给客户端out.println("已收到你的消息:" + message);}// 关闭连接clientSocket.close();serverSocket.close();} catch (IOException e) {e.printStackTrace();}}
}
客户端的工作流程:
【1】创建客户端的socket
。
【2】向服务器发起连接请求。
【3】与服务端通信,发送一个报文后等待回复,然后再发下一个报文。
【4】不断的重复第3)步,直到全部的数据被发送完。
【5】第4步:关闭socket
,释放资源。
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.Socket;public class Client {public static void main(String[] args) {try {// 创建Socket对象,指定服务端的IP地址和端口号Socket socket = new Socket("127.0.0.1", 45442);// 获取输入流和输出流 输入流和输出流是通过socket对象来进行数据传输的。BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream()));PrintWriter out = new PrintWriter(socket.getOutputStream(), true);// 从控制台读取用户输入BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));String message;while (true) {System.out.println("请输入要发送的信息(输入 'exit' 退出):");message = reader.readLine();if (message.equalsIgnoreCase("exit")) {// 如果用户输入 'exit',发送终止标志给服务端并退出循环out.println("exit");break;}// 将用户输入的信息发送给服务端out.println(message);// 接收服务端的响应并打印String response = in.readLine();System.out.println("服务端响应:" + response);}// 关闭连接socket.close();} catch (IOException e) {e.printStackTrace();}}
}
测试数据:
请输入要发送的信息(输入 'exit' 退出):
Hello Word
服务端响应:已收到你的消息:Hello Word
在运行程序之前,必须保证服务器的防火墙已经开通了网络访问策略
socket
是系统资源,操作系统打开的socket
数量是有限的,在程序退出之前必须关闭已打开的socket
,就像关闭文件指针一样,就像delete
已分配的内存一样,极其重要。
UDP链接
使用UDP
通信,使用java.net.DatagramSocket
创建了一个DatagramSocket
对象。
【1】服务端
package socket.UDP;import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.*;public class UDPClient {public static void main(String[] args) {try {// 创建DatagramSocket对象DatagramSocket socket = new DatagramSocket();InetAddress serverAddress = InetAddress.getByName("127.0.0.1");int serverPort = 45442;// 从控制台读取用户输入BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));String message;while (true) {System.out.println("请输入要发送的信息(输入 'exit' 退出):");message = reader.readLine();if (message.equalsIgnoreCase("exit")) {// 如果用户输入 'exit',退出循环break;}byte[] sendData = message.getBytes();// 创建发送数据的DatagramPacket对象DatagramPacket sendPacket = new DatagramPacket(sendData, sendData.length, serverAddress, serverPort);// 发送数据socket.send(sendPacket);byte[] receiveData = new byte[1024];// 创建接收数据的DatagramPacket对象DatagramPacket receivePacket = new DatagramPacket(receiveData, receiveData.length);// 接收服务端的响应socket.receive(receivePacket);// 将接收到的数据转换为字符串并打印String response = new String(receivePacket.getData(), 0, receivePacket.getLength());System.out.println("服务端响应:" + response);}// 关闭Socket连接socket.close();} catch (IOException e) {e.printStackTrace();}}
}
【2】客户端
package socket.UDP;import java.io.IOException;
import java.net.*;public class UDPServer {public static void main(String[] args) {try {// 创建DatagramSocket对象,并指定监听的端口号DatagramSocket socket = new DatagramSocket(45442);System.out.println("等待客户端连接...");byte[] receiveData = new byte[1024];while (true) {// 创建接收数据的DatagramPacket对象DatagramPacket receivePacket = new DatagramPacket(receiveData, receiveData.length);// 接收客户端发送的数据socket.receive(receivePacket);// 获取客户端的IP地址和端口号InetAddress clientAddress = receivePacket.getAddress();int clientPort = receivePacket.getPort();// 将接收到的数据转换为字符串String message = new String(receivePacket.getData(), 0, receivePacket.getLength());System.out.println("收到客户端消息:" + message);if (message.equalsIgnoreCase("exit")) {// 如果接收到终止标志,退出循环break;}// 构造发送数据的字节数组String response = "已收到你的消息:" + message;byte[] sendData = response.getBytes();// 创建发送数据的DatagramPacket对象DatagramPacket sendPacket = new DatagramPacket(sendData, sendData.length, clientAddress, clientPort);// 发送响应给客户端socket.send(sendPacket);}// 关闭Socket连接socket.close();} catch (IOException e) {e.printStackTrace();}}
}
测试数据:
请输入要发送的信息(输入 'exit' 退出):
UPD Test
服务端响应:已收到你的消息:UPD Test
相关文章:

Socket通信
优质博文IT-BLOG-CN 一、简介 Socket套接字:描述了计算机的IP地址和端口,运行在计算机中的程序之间采用socket进行数据通信。通信的两端都有socket,它是一个通道,数据在两个socket之间进行传输。socket把复杂的TCP/IP协议族隐藏在…...
TCP 如何保证有效传输及拥塞控制
TCP(传输控制协议)可以通过以下机制保证有效传输和拥塞控制: 确认机制:TCP使用确认机制来保证数据的有效传输。发送方在发送数据的同时还会发送一个确认请求,接收方收到数据后会回复确认响应。如果发送方没有收到确认响…...

PyQt5+Qt设计师初探
在上一篇文章中我们搭建好了PyQt5的开发环境,打铁到趁热我们基于搭建好的环境来简单实战一把 一:PyQt5包模块简介 PyQt5包括的主要模块如下。 QtCore模块——涵盖了包的核心的非GUI功能,此模块被用于处理程序中涉及的时间、文件、目录、数…...
rust cargo
一、cargo是什么 Cargo是Rust的构建工具和包管理器。 Cargo除了创建工程、构建工程、运行工程等功能,还具有下载依赖库、编译依赖等功能。 真正编写程序时,我们不直接用rustc,而是用cargo。 二、使用cargo (一)使用…...

CANoe.Diva生成测试用例
Diva目录 一、CANoe.Diva打开CDD文件二、导入CDD文件三、ECU Information四、时间参数设置五、选择是否测试功能寻址六、勾选需要测试服务项七、生成测试用例 一、CANoe.Diva打开CDD文件 CANoe.Diva可以通过导入cdd或odx文件,自动生成全面的测试用例。再在CANoe中导…...

openGauss学习笔记-89 openGauss 数据库管理-内存优化表MOT管理-内存表特性-使用MOT-MOT使用查询原生编译
文章目录 openGauss学习笔记-89 openGauss 数据库管理-内存优化表MOT管理-内存表特性-使用MOT-MOT使用查询原生编译89.1 查询编译:PREPARE语句89.2 运行命令89.3 轻量执行支持的查询89.4 轻量执行不支持的查询89.5 JIT存储过程89.6 MOT JIT诊断89.6.1 mot_jit_detai…...

python获取时间戳
使用 datetime 库获取时间。 获取当前时间: import datetime print(datetime.datetime.now()) . 后面的是微秒,也是一个时间单位,1秒1000000微秒。 转为时间戳: import datetimedate datetime.datetime.now() timestamp date…...

2023年山东省安全员C证证考试题库及山东省安全员C证试题解析
题库来源:安全生产模拟考试一点通公众号小程序 2023年山东省安全员C证证考试题库及山东省安全员C证试题解析是安全生产模拟考试一点通结合(安监局)特种作业人员操作证考试大纲和(质检局)特种设备作业人员上岗证考试大…...
Java中的Unicode字符编码与占用比特位解析
本文将详细介绍Java中Unicode字符编码与占用比特位的相关知识。我们将首先介绍Unicode字符集的基本概念,然后深入探讨Java中Unicode字符的编码方式以及占用比特位的特点。最后,我们将讨论一些特殊字符的编码情况,并给出一些在Java中处理Unico…...

分布式事务-TCC案例分析流程图
防止cancel方法在最后执行出现问题,用户收到提示已经退款成功但是由于cancel过慢或者出现问题(虽然最后会重试成功但是用户体验很差),可以做以下的业务sql模型优化(增加一个冻结金额)。...

究竟是什么样的讲解数组算法的博客让我写了三小时???
版本说明 当前版本号[20231004]。 版本修改说明20231004初版 目录 文章目录 版本说明目录二. 基础数据结构2.1 数组1) 概述2) 动态数组1)插入addlast 方法测试: addlast 方法 add 方法测试:add方法 addlast 方法与 add 方法合并版get 方法测试&#x…...

Day-05 CentOS7.5 安装docker
参考 : Install Docker Engine on CentOS | Docker DocsLearn how to install Docker Engine on CentOS. These instructions cover the different installation methods, how to uninstall, and next steps.https://docs.docker.com/engine/install/centos/ Doc…...

Makefile
目录 Makefile Makefile格式 Makefile函数:foreach和wildcard $(foreach var,list,text) $(wildcard pattern) 完善Makefile Makefile 在Linux中使用make工具来编译程序(特别是大程序),而make工具所执行的动作依赖于Makefil…...
c语言练习77:公因⼦的数⽬
公因⼦的数⽬ 题⽬描述: 给你两个正整数 a 和 b ,返回 a 和 b 的公因⼦的数⽬。 如果 x 可以同时整除 a 和 b ,则认为 x 是 a 和 b 的⼀个公因⼦ 。 • ⽰例 1: 输⼊:a 12, b 6 输出:4 解释&#…...

【C++】C++11——右值引用和移动语义、左值引用和右值引用、右值引用使用场景和意义、完美转发、新的类功能
文章目录 C115.右值引用和移动语义5.1左值引用和右值引用5.2左值引用与右值引用比较5.3右值引用使用场景和意义5.4右值引用引用左值及其一些更深入的使用场景分析5.5完美转发 6.新的类功能 C11 5.右值引用和移动语义 右值引用是C11引入的一个新特性,用于支持移动语义…...

Spring Boot的创建和使用(JavaEE进阶系列2)
目录 前言: 1.什么是Spring Boot?为什么要学习Spring Boot? 2.Spring Boot优点 3.创建Spring Boot项目 3.1准备工作 3.2Spring Boot创建 3.2.1通过idea的方式创建 3.2.2通过网页创建 4.Spring Boot中的配置文件 4.1Spring Boot配置…...

【OLSR路由协议】链路状态路由(OLSR)协议中选择多点中继节点算法研究(Matlab代码实现)
💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。 ⛳️座右铭&a…...

双重差分模型(DID)论文写作指南与操作手册
手册链接:双重差分模型(DID)论文写作指南与操作手册https://www.cctalk.com/m/group/90983583?xh_fshareuid60953990 简介: 当前,对于准应届生们来说,毕设季叠加就业季,写作时间显得十分宝贵…...
ping 的工作原理
ping 是一个常用于网络诊断的命令行工具,用于测试两台计算机之间的网络连通性。它的工作原理如下: 发出 ICMP Echo 请求: 当你在终端中运行 ping 命令并指定目标主机的IP地址或域名时,计算机会创建一个特殊的ICMP(In…...
93. 复原 IP 地址
有效 IP 地址 正好由四个整数(每个整数位于 0 到 255 之间组成,且不能含有前导 0),整数之间用 . 分隔。 例如:"0.1.2.201" 和 "192.168.1.1" 是 有效 IP 地址,但是 "0.011.255.24…...

Lombok 的 @Data 注解失效,未生成 getter/setter 方法引发的HTTP 406 错误
HTTP 状态码 406 (Not Acceptable) 和 500 (Internal Server Error) 是两类完全不同的错误,它们的含义、原因和解决方法都有显著区别。以下是详细对比: 1. HTTP 406 (Not Acceptable) 含义: 客户端请求的内容类型与服务器支持的内容类型不匹…...
Java如何权衡是使用无序的数组还是有序的数组
在 Java 中,选择有序数组还是无序数组取决于具体场景的性能需求与操作特点。以下是关键权衡因素及决策指南: ⚖️ 核心权衡维度 维度有序数组无序数组查询性能二分查找 O(log n) ✅线性扫描 O(n) ❌插入/删除需移位维护顺序 O(n) ❌直接操作尾部 O(1) ✅内存开销与无序数组相…...
OpenPrompt 和直接对提示词的嵌入向量进行训练有什么区别
OpenPrompt 和直接对提示词的嵌入向量进行训练有什么区别 直接训练提示词嵌入向量的核心区别 您提到的代码: prompt_embedding = initial_embedding.clone().requires_grad_(True) optimizer = torch.optim.Adam([prompt_embedding...

(转)什么是DockerCompose?它有什么作用?
一、什么是DockerCompose? DockerCompose可以基于Compose文件帮我们快速的部署分布式应用,而无需手动一个个创建和运行容器。 Compose文件是一个文本文件,通过指令定义集群中的每个容器如何运行。 DockerCompose就是把DockerFile转换成指令去运行。 …...
C++八股 —— 单例模式
文章目录 1. 基本概念2. 设计要点3. 实现方式4. 详解懒汉模式 1. 基本概念 线程安全(Thread Safety) 线程安全是指在多线程环境下,某个函数、类或代码片段能够被多个线程同时调用时,仍能保证数据的一致性和逻辑的正确性…...

GO协程(Goroutine)问题总结
在使用Go语言来编写代码时,遇到的一些问题总结一下 [参考文档]:https://www.topgoer.com/%E5%B9%B6%E5%8F%91%E7%BC%96%E7%A8%8B/goroutine.html 1. main()函数默认的Goroutine 场景再现: 今天在看到这个教程的时候,在自己的电…...

【 java 虚拟机知识 第一篇 】
目录 1.内存模型 1.1.JVM内存模型的介绍 1.2.堆和栈的区别 1.3.栈的存储细节 1.4.堆的部分 1.5.程序计数器的作用 1.6.方法区的内容 1.7.字符串池 1.8.引用类型 1.9.内存泄漏与内存溢出 1.10.会出现内存溢出的结构 1.内存模型 1.1.JVM内存模型的介绍 内存模型主要分…...

消息队列系统设计与实践全解析
文章目录 🚀 消息队列系统设计与实践全解析🔍 一、消息队列选型1.1 业务场景匹配矩阵1.2 吞吐量/延迟/可靠性权衡💡 权衡决策框架 1.3 运维复杂度评估🔧 运维成本降低策略 🏗️ 二、典型架构设计2.1 分布式事务最终一致…...

数据结构第5章:树和二叉树完全指南(自整理详细图文笔记)
名人说:莫道桑榆晚,为霞尚满天。——刘禹锡(刘梦得,诗豪) 原创笔记:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 上一篇:《数据结构第4章 数组和广义表》…...

【iOS】 Block再学习
iOS Block再学习 文章目录 iOS Block再学习前言Block的三种类型__ NSGlobalBlock____ NSMallocBlock____ NSStackBlock__小结 Block底层分析Block的结构捕获自由变量捕获全局(静态)变量捕获静态变量__block修饰符forwarding指针 Block的copy时机block作为函数返回值将block赋给…...