吃透C++ AVL树:原理+完整实现,新手也能轻松上手

news2026/5/14 2:31:15
文章目录前言一、先搞懂什么是AVL树核心特性是什么二、AVL树的C完整实现新手可直接复制运行三、AVL树的删除操作可选进阶内容四、AVL树的性能与应用场景五、新手常见误区避坑总结六、总结前言在C数据结构中AVL树是最经典的平衡二叉搜索树它解决了普通二叉搜索树BST在有序插入时退化为单支树、效率骤降的问题。AVL树通过自动调整树的结构保证树的高度始终维持在O(log₂N)使得插入、查找、删除的时间复杂度稳定在O(log₂N)是面试和实际开发中的高频知识点。很多新手刚接触AVL树时会被“平衡调整”“旋转操作”吓住觉得它复杂难学。其实只要抓住核心——“维持平衡的本质是控制树的高度差”再逐个拆解旋转逻辑就能轻松掌握。今天这篇博客全程以“实用”为导向不搞复杂理论堆砌从原理到代码实现再到实战测试一步步带你搞定AVL树新手跟着敲一遍就能理解并复用代码。一、先搞懂什么是AVL树核心特性是什么AVL树本质是“自平衡的二叉搜索树”它继承了二叉搜索树的所有特性左子树所有节点值根节点值右子树所有节点值根节点值中序遍历有序在此基础上增加了“平衡条件”确保树的结构不会过于倾斜。1. AVL树的核心平衡条件AVL树中每个节点的左右子树高度差称为“平衡因子”的绝对值 ≤ 1。平衡因子 左子树高度 - 右子树高度平衡因子的取值只能是-1、0、1只要有一个节点的平衡因子绝对值1树就失衡需要调整补充树的高度Height定义为“从该节点到最远叶子节点的最长路径上的节点数”叶子节点的高度为1也有定义为0的情况本文统一为1代码中会明确标注避免混淆。2. AVL树的核心价值普通二叉搜索树BST在插入有序数据如1、2、3、4、5时会退化为单支树类似链表此时插入、查找、删除的时间复杂度会退化为O(N)而AVL树通过“旋转调整”始终维持平衡确保时间复杂度稳定在O(log₂N)解决了BST的效率缺陷。3. 失衡场景与调整方式当插入或删除节点后会导致部分节点的平衡因子绝对值1此时需要通过“旋转”操作调整树的结构恢复平衡。根据失衡节点的平衡因子和子节点的平衡因子分为4种失衡场景对应4种旋转方式这是AVL树的核心难点我们逐个拆解1LL型失衡左左失衡场景失衡节点的平衡因子为2左子树比右子树高2且其左孩子的平衡因子为1左孩子的左子树更高。调整方式右旋转围绕失衡节点旋转核心是“将失衡节点的左孩子提升为新根失衡节点作为新根的右孩子新根原来的右子树作为失衡节点的左子树”。2RR型失衡右右失衡场景失衡节点的平衡因子为-2右子树比左子树高2且其右孩子的平衡因子为-1右孩子的右子树更高。调整方式左旋转围绕失衡节点旋转核心是“将失衡节点的右孩子提升为新根失衡节点作为新根的左孩子新根原来的左子树作为失衡节点的右子树”。3LR型失衡左右失衡场景失衡节点的平衡因子为2且其左孩子的平衡因子为-1左孩子的右子树更高。调整方式先左旋转左孩子再右旋转失衡节点——先将左孩子的右子树提升为左孩子的父节点消除左孩子的失衡再对原失衡节点进行右旋转恢复整体平衡。4RL型失衡右左失衡场景失衡节点的平衡因子为-2且其右孩子的平衡因子为1右孩子的左子树更高。调整方式先右旋转右孩子再左旋转失衡节点——先将右孩子的左子树提升为右孩子的父节点消除右孩子的失衡再对原失衡节点进行左旋转恢复整体平衡。关键记住旋转的核心是“降低树的高度恢复平衡因子”LL和RR是单旋转LR和RL是双旋转本质都是通过调整节点位置让每个节点的平衡因子回到-1、0、1。二、AVL树的C完整实现新手可直接复制运行AVL树的实现核心的是节点结构定义、高度计算、平衡因子计算、旋转操作、插入操作插入后调整平衡、删除操作可选新手可先掌握插入、中序遍历验证有序性。下面给出完整代码每个函数都有详细注释重点关注旋转操作和插入后的平衡调整新手跟着注释看就能理解逻辑。1. 节点结构定义AVL树的节点除了存储key、左孩子、右孩子还需要存储当前节点的高度用于计算平衡因子。#include iostream #include algorithm // 用于max函数计算树的高度 using namespace std; // AVL树节点结构 template class K struct AVLNode { K _key; // 节点存储的键值 AVLNodeK* _left; // 左孩子指针 AVLNodeK* _right; // 右孩子指针 int _height; // 当前节点的高度叶子节点高度为1 // 构造函数 AVLNode(const K key) : _key(key) , _left(nullptr) , _right(nullptr) , _height(1) // 新节点默认是叶子节点高度为1 {} };2. AVL树类定义与核心函数AVL树类封装了根节点以及核心操作插入、旋转、高度计算、平衡因子计算、中序遍历等重点实现插入和旋转。template class K class AVLTree { typedef AVLNodeK Node; // 简化节点类型名 public: // 构造函数初始化根节点为空 AVLTree() : _root(nullptr) {} // 析构函数递归销毁所有节点避免内存泄漏 ~AVLTree() { Destroy(_root); } // 对外接口插入节点用户调用无需关心平衡调整 bool Insert(const K key) { return Insert(_root, key); } // 对外接口中序遍历验证AVL树的有序性和BST一致 void InOrder() { InOrder(_root); cout endl; } // 对外接口获取树的高度可选用于调试 int GetHeight() { return GetHeight(_root); } // 对外接口验证AVL树是否平衡可选用于调试 bool IsBalance() { return IsBalance(_root); } private: Node* _root; // 根节点 // 递归销毁节点析构函数调用 void Destroy(Node* root) { if (root nullptr) return; // 后序遍历先销毁左子树再销毁右子树最后销毁当前节点 Destroy(root-_left); Destroy(root-_right); delete root; root nullptr; } // 递归计算节点的高度空节点高度为0叶子节点高度为1 int GetHeight(Node* root) { return root nullptr ? 0 : root-_height; } // 计算节点的平衡因子左子树高度 - 右子树高度 int GetBalanceFactor(Node* root) { if (root nullptr) return 0; return GetHeight(root-_left) - GetHeight(root-_right); } // 1. 右旋转处理LL型失衡 Node* RightRotate(Node* parent) { // 步骤1保存父节点的左孩子newRoot和左孩子的右子树subRight Node* newRoot parent-_left; Node* subRight newRoot-_right; // 步骤2旋转核心操作 newRoot-_right parent; // 父节点成为newRoot的右孩子 parent-_left subRight; // subRight成为父节点的左孩子 // 步骤3更新父节点和newRoot的高度先更新父节点再更新newRoot parent-_height max(GetHeight(parent-_left), GetHeight(parent-_right)) 1; newRoot-_height max(GetHeight(newRoot-_left), GetHeight(newRoot-_right)) 1; // 步骤4返回新的根节点newRoot成为当前子树的根 return newRoot; } // 2. 左旋转处理RR型失衡 Node* LeftRotate(Node* parent) { // 步骤1保存父节点的右孩子newRoot和右孩子的左子树subLeft Node* newRoot parent-_right; Node* subLeft newRoot-_left; // 步骤2旋转核心操作 newRoot-_left parent; // 父节点成为newRoot的左孩子 parent-_right subLeft; // subLeft成为父节点的右孩子 // 步骤3更新父节点和newRoot的高度 parent-_height max(GetHeight(parent-_left), GetHeight(parent-_right)) 1; newRoot-_height max(GetHeight(newRoot-_left), GetHeight(newRoot-_right)) 1; // 步骤4返回新的根节点 return newRoot; } // 递归插入节点核心函数插入后调整平衡 bool Insert(Node* root, const K key) { // 第一步和普通BST一样插入节点 if (root nullptr) { // 空节点创建新节点插入成功 root new Node(key); return true; } // 递归查找插入位置 if (key root-_key) { // 插入到左子树 if (!Insert(root-_left, key)) { return false; // 插入重复key返回失败 } } else if (key root-_key) { // 插入到右子树 if (!Insert(root-_right, key)) { return false; // 插入重复key返回失败 } } else { // key已存在插入失败AVL树不允许重复key return false; } // 第二步插入后更新当前节点的高度从下往上更新 root-_height max(GetHeight(root-_left), GetHeight(root-_right)) 1; // 第三步计算当前节点的平衡因子判断是否失衡 int balanceFactor GetBalanceFactor(root); // 第四步根据失衡类型进行旋转调整恢复平衡 // 4.1 LL型失衡平衡因子2左孩子平衡因子1 if (balanceFactor 1 GetBalanceFactor(root-_left) 1) { root RightRotate(root); } // 4.2 RR型失衡平衡因子-2右孩子平衡因子-1 else if (balanceFactor -1 GetBalanceFactor(root-_right) -1) { root LeftRotate(root); } // 4.3 LR型失衡平衡因子2左孩子平衡因子-1 else if (balanceFactor 1 GetBalanceFactor(root-_left) -1) { // 先左旋转左孩子再右旋转当前节点 root-_left LeftRotate(root-_left); root RightRotate(root); } // 4.4 RL型失衡平衡因子-2右孩子平衡因子1 else if (balanceFactor -1 GetBalanceFactor(root-_right) 1) { // 先右旋转右孩子再左旋转当前节点 root-_right RightRotate(root-_right); root LeftRotate(root); } // 插入成功无论是否调整平衡只要没重复key就返回true return true; } // 递归中序遍历验证有序性 void InOrder(Node* root) { if (root nullptr) return; InOrder(root-_left); // 遍历左子树 cout root-_key ; // 访问当前节点 InOrder(root-_right); // 遍历右子树 } // 递归验证AVL树是否平衡调试用 bool IsBalance(Node* root) { if (root nullptr) return true; // 空树是平衡的 // 计算当前节点的平衡因子 int balanceFactor GetBalanceFactor(root); // 若平衡因子绝对值1树失衡 if (abs(balanceFactor) 1) { cout 失衡节点key root-_key 平衡因子 balanceFactor endl; return false; } // 递归验证左子树和右子树是否平衡只有左右子树都平衡当前树才平衡 return IsBalance(root-_left) IsBalance(root-_right); } };3. 实战测试验证AVL树的插入与平衡编写主函数插入有序数据模拟BST退化场景验证AVL树是否能自动调整平衡同时验证中序遍历的有序性和树的平衡性。// 主函数测试 int main() { AVLTreeint avl; // 测试1插入有序数据1,2,3,4,5模拟BST退化场景 // 若为普通BST会退化为单支树AVL树会自动旋转调整维持平衡 int arr[] {1, 2, 3, 4, 5}; for (auto num : arr) { avl.Insert(num); } // 测试2中序遍历验证有序性应输出1 2 3 4 5 cout AVL树中序遍历; avl.InOrder(); // 测试3获取树的高度5个节点的AVL树高度应为3log₂5≈2.32向上取整为3 cout AVL树高度 avl.GetHeight() endl; // 输出3 // 测试4验证树是否平衡应输出true即平衡 cout AVL树是否平衡 (avl.IsBalance() ? 是 : 否) endl; // 输出是 // 额外测试插入无序数据验证平衡 AVLTreeint avl2; int arr2[] {8, 3, 10, 1, 6, 14, 4, 7, 13}; for (auto num : arr2) { avl2.Insert(num); } cout ------------------------ endl; cout AVL树2中序遍历; avl2.InOrder(); // 输出1 3 4 6 7 8 10 13 14 cout AVL树2高度 avl2.GetHeight() endl; // 输出49个节点log₂9≈3.17向上取整为4 cout AVL树2是否平衡 (avl2.IsBalance() ? 是 : 否) endl; // 输出是 return 0; }运行结果AVL树中序遍历1 2 3 4 5 AVL树高度3 AVL树是否平衡是 ------------------------ AVL树2中序遍历1 3 4 6 7 8 10 13 14 AVL树2高度4 AVL树2是否平衡是从运行结果可以看出AVL树的中序遍历始终有序继承了BST的特性插入有序数据后树的高度为3远小于单支树的高度5说明自动完成了平衡调整IsBalance()函数验证结果为“是”说明所有节点的平衡因子都符合要求。三、AVL树的删除操作可选进阶内容AVL树的删除操作比插入更复杂核心逻辑是先按照BST的删除规则删除节点再从删除节点的父节点开始向上更新高度、判断平衡若失衡则进行旋转调整。删除操作同样需要分4种失衡场景旋转逻辑和插入一致但需要处理更多边界情况如删除叶子节点、删除有一个子节点的节点、删除有两个子节点的节点。新手可先掌握插入操作删除操作作为进阶内容下面给出核心代码直接添加到AVLTree类的private中对外提供接口即可// 递归删除节点核心函数删除后调整平衡 bool Erase(Node* root, const K key) { // 第一步和普通BST一样查找并删除节点 if (root nullptr) { return false; // 未找到待删除节点返回失败 } bool ret false; if (key root-_key) { // 待删除节点在左子树 ret Erase(root-_left, key); } else if (key root-_key) { // 待删除节点在右子树 ret Erase(root-_right, key); } else { // 找到待删除节点分3种情况处理 // 情况1待删除节点只有右子树或无孩子 if (root-_left nullptr) { Node* temp root; root root-_right; delete temp; temp nullptr; ret true; } // 情况2待删除节点只有左子树 else if (root-_right nullptr) { Node* temp root; root root-_left; delete temp; temp nullptr; ret true; } // 情况3待删除节点有两个子树找中序后继节点替代 else { // 找中序后继右子树中最小的节点最左侧节点 Node* subParent root; Node* subCur root-_right; while (subCur-_left) { subParent subCur; subCur subCur-_left; } // 用后继节点的值覆盖待删除节点的值 root-_key subCur-_key; // 删除后继节点后继节点只有右子树或无孩子属于情况1 if (subParent-_left subCur) { subParent-_left subCur-_right; } else { subParent-_right subCur-_right; } delete subCur; subCur nullptr; ret true; } } // 若删除节点后当前子树为空无需调整平衡直接返回 if (root nullptr) return ret; // 第二步更新当前节点的高度 root-_height max(GetHeight(root-_left), GetHeight(root-_right)) 1; // 第三步计算平衡因子判断是否失衡进行旋转调整 int balanceFactor GetBalanceFactor(root); // LL型失衡 if (balanceFactor 1 GetBalanceFactor(root-_left) 0) { root RightRotate(root); } // RR型失衡 else if (balanceFactor -1 GetBalanceFactor(root-_right) 0) { root LeftRotate(root); } // LR型失衡 else if (balanceFactor 1 GetBalanceFactor(root-_left) 0) { root-_left LeftRotate(root-_left); root RightRotate(root); } // RL型失衡 else if (balanceFactor -1 GetBalanceFactor(root-_right) 0) { root-_right RightRotate(root-_right); root LeftRotate(root); } return ret; } // 对外接口删除节点用户调用 bool Erase(const K key) { return Erase(_root, key); }四、AVL树的性能与应用场景1. 性能分析时间复杂度插入、查找、删除的时间复杂度均为O(log₂N)因为AVL树的高度始终维持在log₂N级别空间复杂度O(N)需要存储N个节点每个节点额外存储一个高度值空间开销可忽略优缺点优点是平衡稳定、效率高缺点是插入和删除时需要频繁旋转调整操作相对复杂适合“插入删除不频繁、查找频繁”的场景。2. 实际应用场景AVL树是平衡二叉搜索树的基础实际开发中更多使用红黑树STL中map、set的底层实现因为红黑树的旋转次数更少插入删除效率更高。但AVL树的平衡条件更严格查找效率略高于红黑树适合对查找性能要求极高的场景例如数据库索引部分数据库的索引底层使用AVL树追求查找高效高频查找的缓存系统需要快速定位数据插入删除频率较低编译器的符号表存储变量、函数名需要快速查找和有序遍历。五、新手常见误区避坑总结误区1忘记更新节点高度——插入或删除后必须从当前节点向上更新所有祖先节点的高度否则平衡因子计算错误导致失衡判断失效误区2旋转后不更新高度——旋转操作会改变节点的父子关系必须重新计算旋转涉及节点的高度否则后续平衡判断会出错误区3混淆平衡因子的计算方式——本文平衡因子左子树高度-右子树高度不同资料可能有不同定义如右减左需保持一致否则旋转逻辑会颠倒误区4认为AVL树一定比红黑树好——AVL树平衡更严格查找更快但插入删除旋转次数更多实际开发中需根据场景选择查找多用电AVL插入删除多用红黑树误区5允许重复key——本文实现的AVL树不允许重复key若需要存储重复key需修改插入逻辑如在左子树或右子树插入重复值但会增加平衡调整的复杂度。六、总结AVL树的核心是“自平衡”本质是通过“旋转操作”修复插入、删除导致的失衡维持树的高度在O(log₂N)从而保证高效的查找、插入、删除操作。对于新手来说重点掌握3点1. 平衡条件每个节点的平衡因子绝对值≤1平衡因子左子树高度-右子树高度2. 旋转操作4种失衡场景对应4种旋转方式LL/RR单旋转LR/RL双旋转核心是“提升子节点为新根调整父子关系”3. 插入逻辑先按BST插入再更新高度、判断失衡、旋转调整从下往上修复平衡。AVL树是理解平衡二叉搜索树的基础学好它能帮助你更好地理解红黑树、B树等更复杂的平衡树也能应对面试中的AVL树实现、旋转场景等高频考题。建议新手多敲几遍代码调试插入、旋转的过程就能彻底掌握AVL树的核心逻辑。最后留一个小练习基于本文的代码实现“查找AVL树中最小节点”和“最大节点”的功能动手敲一敲巩固今天的知识点

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

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

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…