网络编程(TCP、UDP)

文章目录

  • 一、概念
    • 1.1 什么是网络编程
    • 1.2 网络编程中的基本知识
  • 二、Socket套接字
    • 2.1 概念及分类
    • 2.2 TCP VS UDP
    • 2.3 通信模型
    • 2.4 接口方法
      • UDP数据报套接字编程
      • TCP流套接字编程
  • 三、代码示例
    • 3.1 注意点
    • 3.2 回显服务器
      • 基于UDP
      • 基于TCP

一、概念

首先介绍了什么是网络编程,随后介绍了接收端发送端、请求响应等基本知识

1.1 什么是网络编程

网络编程
(1)网络上的主机,通过不同的进程,以编程的方式实现网络通信(或称为网络数据传输)。(2)在程序员层面上,即要写一个应用程序通过调用传输层提供的API实现网络通信

1.2 网络编程中的基本知识

  1. 发送端和接收端:字面意思,发送端是发送数据的,接收端是接收数据的,注意这个概念只是相对的,主要看数据是从哪发到哪。

  2. 请求和响应
    一般来说,获取一个网络资源,涉及到两次网络数据传输:

    • 第一次:请求数据的发送
    • 第二次:响应数据的发送
  3. 客户端和服务器
    服务端:在常见的网络数据传输场景下,给用户提供服务的一方进程
    客户端:获取服务的一方进程,是指给用户使用的程序

  4. 常见的客户端服务端模型

    • 客户端先发送请求到服务端
    • 服务端根据请求数据,执行相应的业务处理
    • 服务端返回响应:发送业务处理结果
    • 客户端根据响应数据,展示处理结果(展示获取的资源,或提示保存资源的处理结果)

二、Socket套接字

网络编程是基于Socket开发的,传输层上主要涉及TCP、UDP这两种协议,而他俩给出的API都不同,本段主要介绍概念,包括分类和区别。

2.1 概念及分类

(1)Socket套接字,是由系统提供用于网络通信的技术,是基于TCP/IP协议的网络通信的基本操作单元。基于Socket套接字的网络程序开发就是网络编程。

(2)Socket套接字主要针对传输层协议划分为如下三类:

  1. 流套接字:传输层TCP协议(传输控制协议)
  2. 数据报套接字:传输层UDP协议(用户数据报协议)
  3. 原始套接字:原始套接字用于自定义传输层协议,用于读写内核没有处理的IP协议数据

2.2 TCP VS UDP

TCP:有连接;可靠传输;面向字节流;有接收缓冲区和发送缓冲区;大小不限;全双工
UDP:无连接;不可靠传输;面向数据报;有接收缓冲区,无发送缓冲区;大小受限,一次最多传输64k

  1. 连接:客户端和服务器之间是否使用内存保存对端信息,互相承认,连接就建立了。
  2. 传输
    • 可靠传输:A尽可能把消息传过去,传输失败与否都能感知到,会降低传输效率。不可靠传输传输效率更高
    • 注意:这并不意味着,TCP比UDP更安全,因为“网络安全”指的是传输的数据是否容易被黑客截获,以及如果被截获后是否会泄漏一些重要信息
  3. 字节流VS数据报
    • 面向字节流:TCP流式操作
    • 面向数据报:读写的基本单位是一个UDP数据报,且一次只能传输一个
  4. 全双工VS半双工
    • 全双工:一个通道,可以双向通信。网线就是全双工的
    • 半双工:一个通道,只能单向通信

2.3 通信模型

数据报套接字模型

在这里插入图片描述
流套接字模型
在这里插入图片描述

2.4 接口方法

UDP数据报套接字编程

UDP提供的API主要通过两个类来实现,一个是DatagramSocket,一个是DatagramPacket。

DatagramSocket 方法:代表一个Socket对象,即一个网卡的文件
DatagramPacket 方法:代表一个UDP数据报

TCP流套接字编程

因为TCP是字节流式传输,所以不需要一个专门的类来表示TCP数据报

ServerSocket 方法:
(1)给服务器使用的
(2)负责拉客,在外场广撒网。

Socket 方法:
(1)服务端和客户端都能用
(2)像是一个耳麦,可以直接与对方进行通信,负责在内场细致服务。

三、代码示例

3.1 注意点

  1. 服务器使用的端口需要手动指定,客户端的则需要系统自动分配
    • 因为服务器需要能被客户端百分百找到,万一服务器随便改变端口,那客户端按照以往的连接是无法连接到服务器的。
    • 客户端则没有这个需求,且客户端一旦指定端口号很可能回合其他程序冲突,系统自动分配的端口号一定是空闲的。
  2. flush() ------ 手动刷新
    • IO操作是比较有开销的,相比于访问内存,进行IO次数越多,程序的速度就越慢。为了解决这个问题,我们会使用一块内存作为缓冲区,写数据的时候,先写到缓冲区里。攒了一波数据之后,统一进行IO。
    • printWriter 内置了缓冲区
    • flush()可以实现手动刷新,可以确保这里的数据是真的通过网卡发出去了,而不是残留在内存缓冲区中的
    • 加上flush()可以让代码更稳妥,但是不加不一定会出错,因为缓冲区内置了一定的刷新策略
  3. 关于 close()
    close() 主要是删除文件描述符表中的记录。当然也并不是每个流对象都有文件描述符表,只有那种调用了操作系统提供的open()方法,涉及到了用户态和用户态。

3.2 回显服务器

基于UDP

(1)InetAddress 表示的是IP地址
(2)客户端要设置要传给的服务器的IP地址和端口号是多少,即示例中的 serverIp 和 serverPort 。服务器则是要手动指定一个端口号。
(3)难点在于 DatagramPacket 的构造,涉及了String 和 DatagramPacket 之间的转换。
(4)数据包的接收和发送,依靠 DatagramSocket 的 send 和 receive 方法。

//客户端
public class UdpEchoClient {private DatagramSocket socket = null;private String serverIp;private int serverPort;public UdpEchoClient(String ip, int port) throws SocketException {serverIp = ip;serverPort = port;socket = new DatagramSocket();}public void start() throws IOException {//获取用户输入的内容Scanner scanner = new Scanner(System.in);System.out.println("客户端启动!");//给服务器发送响应while (true){System.out.print("-> ");String request = scanner.next();//不能用request.length(),因为这是指字符长度,我们需要的是字节长度//除非是纯英文且由utf-8编码,那么两者会相同。Socket是按照字节来处理的。DatagramPacket requestPacket = new DatagramPacket(request.getBytes(), request.getBytes().length,InetAddress.getByName(serverIp), serverPort);socket.send(requestPacket);//接收响应DatagramPacket responsePacket = new DatagramPacket(new byte[4096], 4096);socket.receive(responsePacket);String response = new String(responsePacket.getData(), 0, responsePacket.getLength());System.out.println(response);}}public static void main(String[] args) throws IOException {UdpEchoClient client = new UdpEchoClient("127.0.0.1", 9090);client.start();}
}
//服务器
public class UdpEchoServer {private DatagramSocket socket = null;public UdpEchoServer(int port) throws SocketException {socket = new DatagramSocket(port);}public void start() throws IOException {System.out.println("服务器启动");while(true){DatagramPacket requestPacket = new DatagramPacket(new byte[4096], 4096);socket.receive(requestPacket);String request = new String(requestPacket.getData(), 0, requestPacket.getLength());String response = process(request);DatagramPacket responsePacket = new DatagramPacket(response.getBytes(), response.getBytes().length,requestPacket.getSocketAddress());socket.send(responsePacket);System.out.printf("[%s:%d req:%s, resp:%s\n]", requestPacket.getAddress().toString(),requestPacket.getPort(), request, response);}}public String process(String request){return request;}public static void main(String[] args) throws IOException {UdpEchoServer server= new UdpEchoServer(9090);server.start();}
}

基于TCP

  1. 服务器对客户是多对多的关系,一个服务器内核会存许多客户端的连接(建立连接的过程,系统内核会自己完成)。连接就像是一个个代办事项,等待应用程序一个个完成。
  2. serverSocket.accept():将在内核中已经建立好的连接取出来给应用程序
  3. 原理是一个【生产者消费者模型】
    在这里插入图片描述
  4. TCP中的长短连接
    • 短连接:每次接收到数据并返回响应后,都关闭连接,即只能一次收发数据
    • 长连接:不关闭连接,一直保持连接状态,双方不停的收发数据
public class TcpServer {private ServerSocket socket = null;private ExecutorService service = Executors.newCachedThreadPool();public TcpServer(int port) throws IOException {socket = new ServerSocket(9090);}public void start() throws IOException {System.out.println("服务端启动!");while (true){Socket clientSocket = socket.accept();//为了能够接收多个客户端,这里使用了线程池//当客户端进一步增加,线程数目也会增加,系统的负担也就越来越重,响应速度也就越来越慢//为了解决这个问题,就需要开源(引入更多的硬件资源)节流(减少每种硬件资源占用的资源)service.submit(new Runnable() {@Overridepublic void run() {try {processConnection(clientSocket);} catch (IOException e) {throw new RuntimeException(e);}}});}}public void processConnection(Socket clientSocket) throws IOException {try(InputStream inputStream = clientSocket.getInputStream();OutputStream outputStream = clientSocket.getOutputStream()){while(true){Scanner scannerClient = new Scanner(inputStream);String request = scannerClient.next();String response = process(request);PrintWriter writer = new PrintWriter(outputStream);writer.println(response);writer.flush();System.out.printf("[%s:%d] req:%s, resp:%s\n", clientSocket.getInetAddress().toString(),clientSocket.getPort(), request, response);}} catch (IOException e) {throw new RuntimeException(e);}finally {clientSocket.close();}}public String process(String request){return request;}public static void main(String[] args) throws IOException {TcpServer server = new TcpServer(9090);server.start();}
}
public class TcpClient {private Socket socket = null;public TcpClient(String serverIp, int serverPort) throws IOException {socket = new Socket(serverIp, serverPort);}public void start(){System.out.println("客户端启动");Scanner scanner = new Scanner(System.in);try(InputStream inputStream = socket.getInputStream();OutputStream outputStream = socket.getOutputStream()){while (true){System.out.print("---->");String request = scanner.next();PrintWriter writer = new PrintWriter(outputStream);writer.println(request);writer.flush();Scanner scammerConsole = new Scanner(inputStream);String response = scammerConsole.next();System.out.println(response);}} catch (IOException e) {throw new RuntimeException(e);}}public static void main(String[] args) throws IOException {TcpClient client = new TcpClient("127.0.0.1", 9090);client.start();}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/793065.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Emacs之实现复制当前已打开文件buffer(一百三十五)

简介: CSDN博客专家,专注Android/Linux系统,分享多mic语音方案、音视频、编解码等技术,与大家一起成长! 优质专栏:Audio工程师进阶系列【原创干货持续更新中……】🚀 优质专栏:多媒…

ruoyi-nbcio-plus基于vue3的flowable流程元素选择区面板的升级修改

更多ruoyi-nbcio功能请看演示系统 gitee源代码地址 前后端代码: https://gitee.com/nbacheng/ruoyi-nbcio 演示地址:RuoYi-Nbcio后台管理系统 http://122.227.135.243:9666/ 更多nbcio-boot功能请看演示系统 gitee源代码地址 后端代码&#xff1a…

51单片机实验01-点亮LED小灯

目录 一,软件下载 二,单片机概述 1,单片机内部资源 1)flash 2)ram 3)sfr 2,51单片机 3,单片机最小系统 三,点亮最右边的小灯 1,指出满足小灯点亮的有…

适用于车载设备无钥匙进入系统汽车用晶振FA-238A

汽车用晶振FA-238A是一款适用于车载设备无钥匙进入系统的耐高温晶振。汽车用晶振FA-238A是爱普生推出一的款MHz表贴式晶体单元,具有很好的预率性能,符合AEC-0200标准,其封装尺寸仅为3.2x2.5x0.7mm,工作温度范围在-40℃~125℃之间&…

【计算机考研】408到底有多难?值得冲吗?

考408就必须要面对的现实!拒绝眼高手低!! 408其实想达到110并不难,但是想上130是比较困难的。 几个必须要面对的现实: 1.如果备考的是11408,除非基础特别好或者学习能力特别强,否则一定要尽早…

非关系型数据库-----------探索Redis支持五种数据类型

目录 一、Redis支持五种数据类型 1.String(字符串) 2.Hash(哈希) 3.List(列表) 4.Set(集合) 5.sorted set(有序集合) 二、Redis的字符串类型string 1、 SET/GET/APPEND/STRL…

Vue关键知识点

watch侦听器 Vue.js 中的侦听器(Watcher)是 Vue提供的一种响应式系统的核心机制之一。 监听数据的变化,并在数据发生变化时执行相应的回调函数。 目的:数据变化能够自动更新到视图中 原理: Vue 的侦听器通过观察对象的属性&#…

Red Hat Enterprise Linux release 8.4安装Jenkins

1. 查看安装 1.1 显示 Linux 系统的详细信息,包括内核版本、操作系统版本和其他相关信息 uname -a1.2 查看 Red Hat Linux 系统的版本 cat /etc/redhat-release # 或者 cat /etc/os-release1.3 查看 JDK 是否安装 java -version #查看安装路径 echo $JAVA_HOME1…

【保姆级介绍Oracle】

🎥博主:程序员不想YY啊 💫CSDN优质创作者,CSDN实力新星,CSDN博客专家 🤗点赞🎈收藏⭐再看💫养成习惯 ✨希望本文对您有所裨益,如有不足之处,欢迎在评论区提出…

Linux云计算之Linux基础2——Linux发行版本的安装

目录 一、彻底删除VMware 二、VMware-17虚拟机安装 三、MobaXterm 安装 四、Centos 发行版 7.9的安装 五、rockys 9.1的安装 六、ubuntu2204的安装 一、彻底删除VMware 在卸载VMware虚拟机之前,要先把与VMware相关的服务和进程终止 1. 在windows中按下【Windo…

windows安装OpenUSD

一、下载OpenUSD git clone https://github.com/PixarAnimationStudios/OpenUSDOpenUSD,原名USD(Universal Scene Description,通用场景描述),是由皮克斯动画工作室开发的一种开放数据格式。OpenUSD主要用于在虚拟世界…

上位机图像处理和嵌入式模块部署(qmacvisual获取边界点)

【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing 163.com】 在图像处理中,解决了分割的问题之后,下面就是属性信息的提取。在这其中,有一种属性是非常重要的 ,那…

ChatGPT 与 OpenAI 的现代生成式 AI(上)

原文:Modern Generative AI with ChatGPT and OpenAI Models 译者:飞龙 协议:CC BY-NC-SA 4.0 序言 本书以介绍生成式 AI 领域开始,重点是使用机器学习算法创建新的独特数据或内容。它涵盖了生成式 AI 模型的基础知识&#xff0c…

Jackson @JsonUnwrapped注解扁平化 序列化反序列化数据

参考资料 Jackson 2.x 系列【7】注解大全篇三JsonUnwrapped 以扁平的数据结构序列化/反序列化属性Jackson扁平化处理对象 目录 一. 前期准备1.1 前端1.2 实体类1.3 Controller层 二. 扁平化序列反序列化数据2.1 序列化数据2.2 反序列化数据 三. 前缀后缀处理属性同名四. Map数…

尚硅谷50道Java面试题笔记 写的不全

b站链接:https://www.bilibili.com/video/BV1Bb411d7SL/?p4&vd_source714a8042f058b82c668750a0930ff9b0 1 mysql使用innodb引擎,请简述mysql索引的最左前缀如何优化orderby语句。 关键点: 如果排序字段不在索引列上,file…

2006-2022年各省研发投入强度数据/研究与试验发展(RD)经费投入强度数据(无缺失)

2006-2022年各省研发投入强度数据/研究与试验发展(R&D)经费投入强度数据(无缺失) 1、时间:2006-2022年 2、范围:31省 3、来源:科技年鉴 4、指标:研发投入强度/研究与试验发展(R&D)经费投入强度 5、指标解释:研发投入…

spring boot后端controller中接收表单参数校验

校验分为两部分,一部分是前端的输入时就校验,一部分时后端接收参数时的校验。本文提到的是后端接收参数时的校验。这个后端校验的存在有什么意义呢? 比如我们设置前端在输入参数时限制输入不能为空,应该为3-20位非空字符&#xf…

十分钟掌握在 PyTorch 中构建一个深度神经网络,基本组件、步骤和代码实现,从导入模块和定义网络结构到训练和评估网络性能。

🍉 CSDN 叶庭云:https://yetingyun.blog.csdn.net/ 深度神经网络(Deep Neural Networks, DNNs),也被称为人工神经网络(Artificial Neural Networks,ANNs),已成为当今机器学习任务中最流行、最成功的方法之一。这些网络能够表示数据中的复杂关系,并在图像分类、自然…

python调用java中的jar

一、基于IDEA生成可执行jar包 1、编写class的代码,注意一定要有main()方法才可以生成jar包,main()方法可以没有内容。例如下Java 代码: package Project;public class Demo {public static void main(String[] args){Demo t2 new Demo();S…

Redux Toolkit+TypeScript最佳实践

Redux-Toolkit是为了简化使用Redux繁琐的步骤,可以j降低使用useReducer与useContext管理状态的频率,而且起到项目中状态管理规范和约束化的效果。 阅读本文需要的前置知识:React、Redux、Typescript、Redux hooks。 Redux-Toolkit使用步骤 …