当前位置: 首页 > news >正文

C++在实际项目中的应用第二节:C++与网络编程

 

第五章:C++在实际项目中的应用

第二节:C++与网络编程
1. TCP/IP协议详解与C++实现

TCP/IP(传输控制协议/互联网协议)是现代互联网通信的基础协议。理解 TCP/IP 协议对于开发网络应用至关重要。本节将详细介绍 TCP/IP 协议的工作原理以及如何在 C++ 中实现。

1.1 TCP/IP协议栈

TCP/IP 协议栈分为四层:应用层、传输层、网络层和链路层。

  • 应用层:负责处理应用程序之间的通信协议,如 HTTP、FTP 和 SMTP。应用层的协议定义了如何在网络上进行数据交换。

  • 传输层:提供端到端的通信服务,主要有 TCP 和 UDP 两种协议。

    • TCP(传输控制协议):面向连接的协议,提供可靠的数据传输。TCP 通过建立连接、数据流控制和错误检测来确保数据的完整性。
    • UDP(用户数据报协议):无连接的协议,适用于对速度要求高但不需要保证可靠性的应用,如视频流和在线游戏。
  • 网络层:负责数据包的路由和转发,主要协议包括 IP(互联网协议)和 ICMP(互联网控制消息协议)。IP 协议为数据包提供地址和路由信息。

  • 链路层:处理物理网络中的数据传输,包括以太网和无线网络协议。

1.2 TCP连接的建立与关闭

TCP 连接的建立采用三次握手(Three-way Handshake)过程:

  1. SYN:客户端向服务器发送 SYN 数据包,请求建立连接。
  2. SYN-ACK:服务器回复 SYN-ACK 数据包,表示接受连接请求。
  3. ACK:客户端发送 ACK 数据包,完成连接建立。

连接的关闭采用四次挥手(Four-way Handshake)过程:

  1. FIN:客户端发送 FIN 数据包,请求关闭连接。
  2. ACK:服务器回复 ACK 数据包,确认关闭请求。
  3. FIN:服务器发送 FIN 数据包,请求关闭连接。
  4. ACK:客户端回复 ACK 数据包,完成连接关闭。
1.3 C++中TCP/IP的实现

在 C++ 中,我们可以使用 POSIX sockets API 来实现 TCP/IP 通信。以下是一个简单的 TCP 服务器和客户端示例。

TCP服务器示例

#include <iostream>
#include <cstring>
#include <unistd.h>
#include <arpa/inet.h>#define PORT 8080
#define BUFFER_SIZE 1024int main() {int server_fd, new_socket;struct sockaddr_in address;int opt = 1;int addrlen = sizeof(address);char buffer[BUFFER_SIZE] = {0};// 创建套接字if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) == 0) {perror("socket failed");exit(EXIT_FAILURE);}// 绑定套接字if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt))) {perror("setsockopt");exit(EXIT_FAILURE);}address.sin_family = AF_INET;address.sin_addr.s_addr = INADDR_ANY;address.sin_port = htons(PORT);if (bind(server_fd, (struct sockaddr *)&address, sizeof(address)) < 0) {perror("bind failed");exit(EXIT_FAILURE);}// 监听连接if (listen(server_fd, 3) < 0) {perror("listen");exit(EXIT_FAILURE);}// 接受连接if ((new_socket = accept(server_fd, (struct sockaddr *)&address, (socklen_t*)&addrlen)) < 0) {perror("accept");exit(EXIT_FAILURE);}// 读取数据read(new_socket, buffer, BUFFER_SIZE);std::cout << "Message received: " << buffer << std::endl;// 关闭套接字close(new_socket);close(server_fd);return 0;
}

TCP客户端示例

#include <iostream>
#include <cstring>
#include <unistd.h>
#include <arpa/inet.h>#define PORT 8080
#define BUFFER_SIZE 1024int main() {int sock = 0;struct sockaddr_in serv_addr;char *message = "Hello from client";// 创建套接字if ((sock = socket(AF_INET, SOCK_STREAM, 0)) < 0) {std::cout << "Socket creation error" << std::endl;return -1;}serv_addr.sin_family = AF_INET;serv_addr.sin_port = htons(PORT);// 转换IPv4地址从文本到二进制if (inet_pton(AF_INET, "127.0.0.1", &serv_addr.sin_addr) <= 0) {std::cout << "Invalid address/ Address not supported" << std::endl;return -1;}// 连接到服务器if (connect(sock, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) < 0) {std::cout << "Connection Failed" << std::endl;return -1;}// 发送数据send(sock, message, strlen(message), 0);std::cout << "Message sent" << std::endl;// 关闭套接字close(sock);return 0;
}
2. 实际项目中的网络通信示例

在实际项目中,网络通信经常用于实现客户端与服务器之间的交互。以下是一个简单的聊天室示例,展示了如何使用 C++ 实现多客户端 TCP 聊天功能。

2.1 聊天服务器实现

聊天服务器需要能够处理多个客户端的连接,可以使用 select 函数或多线程来实现。下面是一个简单的多线程聊天室服务器示例。

#include <iostream>
#include <cstring>
#include <unistd.h>
#include <arpa/inet.h>
#include <thread>
#include <vector>
#include <mutex>#define PORT 8080
#define BUFFER_SIZE 1024std::vector<int> clients;
std::mutex clients_mutex;void handle_client(int client_socket) {char buffer[BUFFER_SIZE];while (true) {int bytes_read = read(client_socket, buffer, BUFFER_SIZE);if (bytes_read <= 0) {break; // 连接关闭}// 广播消息std::lock_guard<std::mutex> lock(clients_mutex);for (int sock : clients) {if (sock != client_socket) {send(sock, buffer, bytes_read, 0);}}}// 清理客户端{std::lock_guard<std::mutex> lock(clients_mutex);clients.erase(std::remove(clients.begin(), clients.end(), client_socket), clients.end());}close(client_socket);
}int main() {int server_fd;struct sockaddr_in address;int opt = 1;int addrlen = sizeof(address);// 创建套接字if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) == 0) {perror("socket failed");exit(EXIT_FAILURE);}// 绑定套接字if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt))) {perror("setsockopt");exit(EXIT_FAILURE);}address.sin_family = AF_INET;address.sin_addr.s_addr = INADDR_ANY;address.sin_port = htons(PORT);if (bind(server_fd, (struct sockaddr *)&address, sizeof(address)) < 0) {perror("bind failed");exit(EXIT_FAILURE);}// 监听连接if (listen(server_fd, 3) < 0) {perror("listen");exit(EXIT_FAILURE);}std::cout << "Chat server started on port " << PORT << std::endl;while (true) {int new_socket = accept(server_fd, (struct sockaddr *)&address, (socklen_t*)&addrlen);if (new_socket < 0) {perror("accept");continue;}// 添加新客户端{std::lock_guard<std::mutex> lock(clients_mutex);clients.push_back(new_socket);}// 创建新线程处理客户端std::thread(handle_client, new_socket).detach();}close(server_fd);return 0;
}
2.2 聊天客户端实现

聊天客户端负责连接到服务器并发送和接收消息。以下是一个简单的聊天客户端示例。

#include <iostream>
#include <cstring>
#include <unistd.h>
#include <arpa/inet.h>
#include <thread>#define PORT 8080
#define BUFFER_SIZE 1024void receive_messages(int sock) {char buffer[BUFFER_SIZE];while (true) {int bytes_read = read(sock, buffer, BUFFER_SIZE);if (bytes_read <= 0) {std::cout << "Disconnected from server." << std::endl;break;}std::cout << "Message received: " << buffer << std::endl;}
}int main() {int sock;struct sockaddr_in serv_addr;// 创建套接字if ((sock = socket(AF_INET, SOCK_STREAM, 0)) < 0) {std::cout << "Socket creation error" << std::endl;return -1;}serv_addr.sin_family = AF_INET;serv_addr.sin_port = htons(PORT);// 转换IPv4地址从文本到二进制if (inet_pton(AF_INET, "127.0.0.1", &serv_addr.sin_addr) <= 0) {std::cout << "Invalid address/ Address not supported" << std::endl;return -1;}// 连接到服务器if (connect(sock, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) < 0) {std::cout << "Connection Failed" << std::endl;return -1;}// 启动接收消息的线程std::thread(receive_messages, sock).detach();std::string message;while (true) {std::cout << "Enter message: ";std::getline(std::cin, message);send(sock, message.c_str(), message.size(), 0);}close(sock);return 0;
}
3. 常见问题及解决方案

在网络编程中,我们可能会遇到各种问题。以下是一些常见问题及其解决方案。

3.1 网络延迟与带宽问题

网络延迟会影响应用程序的性能。可以采取以下措施来优化网络延迟:

  • 使用异步 I/O:通过使用异步 I/O 模型,可以提高应用程序的响应速度。
  • 压缩数据:在发送大量数据时,考虑使用数据压缩来减少传输时间。
3.2 连接超时问题

连接超时可能会导致用户体验不佳。可以通过设置合理的超时参数来避免这种情况:

struct timeval timeout;
timeout.tv_sec = 5; // 超时5秒
timeout.tv_usec = 0;
setsockopt(sock, SOL_SOCKET, SO_SNDTIMEO, (const char*)&timeout, sizeof(timeout));
setsockopt(sock, SOL_SOCKET, SO_RCVTIMEO, (const char*)&timeout, sizeof(timeout));
3.3 安全性问题

网络安全是一个重要的问题。在网络编程中,可以考虑以下安全措施:

  • 加密传输:使用 SSL/TLS 等加密协议来保护数据传输的安全性。
  • 输入验证:对用户输入进行验证,防止 SQL 注入和其他攻击。
3.4 错误处理

在网络编程中,正确的错误处理机制至关重要。以下是一个简单的错误处理示例:

if (send(sock, message.c_str(), message.size(), 0) < 0) {perror("Send failed");close(sock);return -1;
}

以上是关于《C++与网络编程》的详细分析,涵盖了 TCP/IP 协议的实现、实际项目中的网络通信示例,以及常见问题及其解决方案。接下来,我们将总结本章内容。

小结

在本课中,我们深入探讨了 C++ 在网络编程中的应用,包括 TCP/IP 协议的详解与实现、实际项目中的网络通信示例,以及解决常见问题的方法。通过具体的代码示例和项目分析,读者应该能够理解如何在实际项目中有效利用 C++ 进行网络编程。接下来,我们将继续探索更多关于 C++ 的高级话题和实践。

相关文章:

C++在实际项目中的应用第二节:C++与网络编程

第五章&#xff1a;C在实际项目中的应用 第二节&#xff1a;C与网络编程 1. TCP/IP协议详解与C实现 TCP/IP&#xff08;传输控制协议/互联网协议&#xff09;是现代互联网通信的基础协议。理解 TCP/IP 协议对于开发网络应用至关重要。本节将详细介绍 TCP/IP 协议的工作原理以…...

依赖关系是危险的

依赖, 我们需要它们&#xff0c;但如何有效安全地使用它们&#xff1f;在本周的节目中&#xff0c;Kris 与 Ian 和 Johnny 一起讨论了 polyfill.io 供应链攻击、Go 中依赖管理和使用的历史&#xff0c;以及 Go 谚语“一点复制胜过一点依赖”。当然&#xff0c;我们用一些不受欢…...

ipguard与Ping32如何加密数据防止泄露?让企业信息更安全

在信息化时代&#xff0c;数据安全已成为企业运营的重中之重。数据泄露不仅会导致经济损失&#xff0c;还可能损害企业声誉。因此&#xff0c;选择合适的数据加密工具是保护企业敏感信息的关键。本文将对IPGuard与Ping32这两款加密软件进行探讨&#xff0c;了解它们如何有效加密…...

gitlab 的备份与回复

一、gitlab备份 1.确定备份目录 gitlab 默认的备份目录为/var/opt/gitlab/backups&#xff0c;可通过配置gitlab.rb配置文件进行修改&#xff0c;如&#xff1a; [rootlocalhost ~]# vim /etc/gitlab/gitlab.rb #若要修改备份文件的存储目录话&#xff0c;打开下面选项的注释…...

创建型模式-----建造者模式

目录 背景&#xff1a; 构建模式UML 代码示例 房子成品&#xff1a; 构建器抽象&#xff1a; 具体构建器&#xff1a; 建筑师&#xff1a; 测试部…...

威胁 Windows 和 Linux 系统的新型跨平台勒索软件:Cicada3301

近年来&#xff0c;网络犯罪世界出现了新的、日益复杂的威胁&#xff0c;能够影响广泛的目标。 这一领域最令人担忧的新功能之一是Cicada3301勒索软件&#xff0c;最近由几位网络安全专家进行了分析。他们有机会采访了这一危险威胁背后的勒索软件团伙的成员。 Cicada3301的崛…...

Go 语言基础教程:7.Switch 语句

在这篇教程中&#xff0c;我们将学习 Go 语言中的 switch 语句&#xff0c;它是条件分支的重要结构。我们将通过一个示例程序逐步解析 switch 的不同用法。 package mainimport ("fmt""time" )func main() {i : 2fmt.Print("Write ", i, " …...

mysql原理、部署mysql主从+读写分离、监控mysql主从脚本

mysql&#xff1a;工作原理 从库生成两个线程&#xff0c;一个I/O线程&#xff0c;一个SQL线程&#xff1b; i/o线程去请求主库 的binlog&#xff0c;并将得到的binlog日志写到relay log&#xff08;中继日志&#xff09; 文件中&#xff1b; 主库会生成一个 log dump 线程&…...

模型选择拟合

1.通过多项式拟合交互探索概念 import math import numpy as np import torch from torch import nn from d2l import torch as d2l 2.使用三阶多项式来生成训练和测试数据的标签 max_degree 20 # 多项式的最大阶数 n_train, n_test 100, 100 # 训练和测试数据集大小 true…...

文案语音图片视频管理分析系统-视频矩阵

文案语音图片视频管理分析系统-视频矩阵 1.产品介绍 产品介绍方案 产品名称&#xff1a; 智驭视频矩阵深度分析系统&#xff08;SmartVMatrix&#xff09; 主要功能&#xff1a; 深度学习驱动的视频内容分析多源视频整合与智能分类高效视频检索与编辑实时视频监控与异常预警…...

ArcGIS计算落入面图层中的线的长度或面的面积

本文介绍在ArcMap软件中&#xff0c;计算落入某个指定矢量面图层中的另一个线图层的长度、面图层的面积等指标的方法。 如下图所示&#xff0c;现在有2个矢量要素集&#xff0c;其中一个为面要素&#xff0c;表示某些区域&#xff1b;另一个为线要素&#xff0c;表示道路路网。…...

ctfshow-web入门-web172

//拼接sql语句查找指定ID用户 $sql "select username,password from ctfshow_user2 where username !flag and id ".$_GET[id]." limit 1;"; 联合查询 该题目与上一个题目不是同一个类型&#xff0c;该题目需要进行sql联合查询。 第一步&#xff1a;确…...

Keep健身TV版 3.3.0 | 针对智能电视的健身塑形软件

Keep健身TV版是专为智能电视设计的功能强大的健身塑形软件。该软件根据用户的不同需求提供多种器械和阶段健身目标组合编排&#xff0c;为用户提供科学、规范、专业的实时指导。即便是在家没有健身器械的情况下&#xff0c;也能跟随教练的语音指导一步步完成训练。软件涵盖从有…...

推荐一些关于计算机网络和 TCP/IP 协议的书籍

以下是一些关于计算机网络和 TCP/IP 协议的优秀书籍推荐: 《TCP/IP 详解》: 作者为 W.Richard Stevens,这是一套经典之作,分为三卷。《TCP/IP 详解卷 1:协议》:详细解析了 TCP/IP 协议的工作原理和实现细节,对协议族中的各个层次和协议,如 IP、TCP、UDP 等进行了深入剖…...

生成式AI浪潮下的商业机遇与经济展望 —— 与互联网时代的比较

一、引言 近年来,生成式人工智能(AI)技术迅速崛起,不仅吸引了大量资本的关注,同时也催生了诸多创新商业模式。与互联网早期阶段类似,AI领域同样面临着前所未有的发展机遇。本文将探讨生成式AI与互联网时代的异同,并分析当前AI行业的经济状况及其增长潜力。 二、经济形…...

Go 标准库

本篇内容是根据2016年9月份The Go Standard Library音频录制内容的整理与翻译, BoltDB 的创建者 Ben Johnson 参加了节目&#xff0c;讨论 NoSQL 与 SQL 数据库、两者之间的权衡以及选择其中之一。我们还讨论了 Ben 的数据秘密生活项目&#xff0c;可视化数据结构&#xff0c;…...

AUTOSAR_EXP_ARAComAPI的6章笔记(5)

☞返回总目录 相关总结&#xff1a;AUTOSAR 通信组的使用方法总结 6.5 通信组的使用方法 6.5.1. 设置 本节描述了使用 Communication Group Template&#xff08;类别为 COMMUNICATION_GROUP&#xff09;定义通信组的配置步骤。定义一个通信组需要指定三个项目&#xff1a;…...

Photoshop中的混合模式公式详解

图层混合简介 图层混合&#xff08;blend&#xff09;顾名思义&#xff0c;就是把两个图层混合成一个。 最基本的混合是alpha融合&#xff08;alpha compositing&#xff09;&#xff0c;这是一个遵循光的反射与透射等&#xff08;简化版&#xff09;物理学原理的混合方式。 各…...

Vue 自定义指令 Directive 的高级使用与最佳实践

前言 Vue.js 是一个非常流行的前端框架&#xff0c;它的核心理念是通过声明式的方式来描述 UI 和数据绑定。除了模板语法和组件系统&#xff0c;Vue 还提供了一个强大的功能——自定义指令。 自定义指令可以让我们对 DOM 元素进行底层操作&#xff0c;下面让我们通过一个有趣的…...

万字图文实战:从0到1构建 UniApp + Vue3 + TypeScript 移动端跨平台开源脚手架

&#x1f680; 作者主页&#xff1a; 有来技术 &#x1f525; 开源项目&#xff1a; youlai-mall &#x1f343; vue3-element-admin &#x1f343; youlai-boot &#x1f343; vue-uniapp-template &#x1f33a; 仓库主页&#xff1a; Gitee &#x1f4ab; Github &#x1f…...

在WebStorm遇到Error: error:0308010C:digital envelope routines::unsupported报错时的解决方案

作者&#xff1a;CSDN-PleaSure乐事 欢迎大家阅读我的博客 希望大家喜欢 使用环境&#xff1a;WebStorm 目录 介绍 解决 分析 方法一&#xff1a;设置环境变量 使用WebStorm 使用其他编译器 方法二&#xff1a;使用nvm切换nodejs版本 方法三&#xff1a;更新依赖版本 介…...

数据库产品中SQL注入防护功能应该包含哪些功能

数据库产品中 SQL 注入防护功能应包含以下几方面&#xff1a; 输入验证与过滤功能&#xff1a; 数据类型和格式验证&#xff1a;检查用户输入的数据是否符合预期的数据类型&#xff0c;比如对于一个应该是整数类型的字段&#xff0c;检查输入是否为整数&#xff1b;对于字符串…...

Ribbon客户端负载均衡策略测试及其改进

文章目录 一、目的概述二、验证步骤1、源码下载2、导入IDE3、运行前修改配置4、策略说明5、修改策略 三、最终结论四、改进措施1. 思路分析2. 核心代码3. 测试页面 一、目的概述 为了验证Ribbon客户端负载均衡策略在负载节点失效的情况下&#xff0c;是否具有故障转移的功能&a…...

linux网络编程5——Posix API和网络协议栈,使用TCP实现P2P通信

文章目录 Posix API和网络协议栈&#xff0c;使用TCP实现P2P通信1. socket()2. bind()3. listen()4. connect()5. accept()6. read()/write(), recv()/send()7. 内核tcp数据传输7.1 TCP流量控制7.2 TCP拥塞控制——慢启动/拥塞避免/快速恢复/快速重传 8. shutdown()9. close()9…...

低代码平台中的功能驱动开发:模块化与领域设计

在现代软件开发中&#xff0c;尤其是在低代码平台的背景下&#xff0c;清晰地定义功能和模块是成功的关键。功能驱动开发强调功能的优先性&#xff0c;模块化设计则确保系统的可维护性和可扩展性。本文将探讨如何在低代码平台中有效地将功能与模块结合起来&#xff0c;形成一个…...

HTTP和HTTPS基本概念,主要区别,应用场景

HTTP和 HTTPS是用于在网络中传输数据的协议&#xff0c;虽然它们的功能类似&#xff0c;但在安全性上存在显著差异。 1. HTTP 的基本概念 定义&#xff1a;HTTP 是一种无状态的、面向请求-响应的协议&#xff0c;用于客户端&#xff08;如浏览器&#xff09;和服务器之间传输…...

node.js使用Sequelize ORM操作数据库

一、什么是ORM ORM是在数据库和编程语言之间建立一种映射关系&#xff0c;这样可以让我们有非常简单的代码&#xff0c;来实现各种数据库的操作。 例如&#xff1a;使用mysql去查找表&#xff08;表名称为Articles&#xff09; SELECT * FROM Articles;但是我们使用ORM的话&…...

STM32-Modbus协议(一文通)

Modbus协议原理 RT-Thread官网开源modbus RT-Thread官方提供 FreeModbus开源。 野火有移植的例程。 QT经常用 libModbus库。 Modbus是什么&#xff1f; Modbus协议&#xff0c;从字面理解它包括Mod和Bus两部分&#xff0c;首先它是一种bus&#xff0c;即总线协议&#xff0c;和…...

100. 不同方向的投影视图

本节课给大家讲解&#xff0c;通过UI按钮界面交互改变threejs相机的观察视角。 x轴方向观察 // 通过UI按钮改变相机观察角度 document.getElementById(x).addEventListener(click, function () {camera.position.set(500, 0, 0); //x轴方向观察camera.lookAt(0, 0, 0); //重新…...

Appium中的api(三)

目录 Appium中的api(三) 1.输入和清空内容 1--输入内容 2--清空内容 2.获取文本内容 3.获取文本位置 4.获取文本的大小&#xff08;即获取控件的宽和高&#xff09; 5.滑动api 6.拖拽api 7.如何获取手机分辨率 8.如何截图 9.模拟按键事件api 10.操作通知栏 案例:App自动化模拟 …...

怎么样能够为一个网站做推广/贵阳网络推广外包

2019独角兽企业重金招聘Python工程师标准>>> C#操作MySQL的增、删、改、查代码&#xff1a; using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; using MySql.Data.MySqlClient; namespace MyS…...

复制代码做网站/百度关键词优化多少钱

现在阿凉给大家带来的是小组赛第四场中的开辟鸿蒙VS斜月三星开辟鸿蒙的综合实力是209430斜月三星的综合实力是231117首先我们看到开辟使用的地阵&#xff0c;小克斜月的蛇阵不过值得关注的是斜月使用了新门派&#xff0c;月宫第一回合&#xff0c;开辟四只灵符女蜗开场&#xf…...

高端做网站/上海优化公司有哪些

以前一直以为IIS应用程序的默认文档只能设置根目录下的文件&#xff0c;像index.html&#xff0c;default.aspx等&#xff0c;后来经同事指点&#xff0c;原来子目录或者子应用程序下的文件也可以添加到根应用程序的默认文档列表中。之 前为了实现这样的需求&#xff0c;当用户…...

武汉住建局/深圳seo优化推广

程序源代码中的注释经常是一个卧虎藏龙的地方&#xff0c;来看看这一辑国外某公司产品中的注释。注意&#xff1a;看的时候严禁喝水或进食。亲爱的代码维护人员&#xff1a;当您尝试优化这段代码但发现这是一个极端错误的决定的时候&#xff0c;请修改下面的计时器&#xff0c;…...

网站建设见站分析和准备论文/网络营销有什么特点

RemoteViews中的setxxx方法 比如setCharSequence(int viewId, String methodName, CharSequence value); views.setString(R.id.textview01, "setText", battery "%"); 其中views是RomoteViews的实例, 第一个参数就是ID了, 第二个参数,是一个方法名字,比如…...

网站建设联盟/线下广告投放渠道都有哪些

这里为了弄清楚JobIntentService&#xff0c;这边我将androidx的JobIntentService源码&#xff0c;doc都进行了翻译并拆分了2个类&#xff0c;分别为BelowOJobIntentService(android8.0以下版本)&#xff0c;和JobIntentXService&#xff08;8.0和以上版本&#xff09;。以便充…...