C++ AVL树详解(含模拟实现)

目录

AVL树的概念

AVL树节点的定义

AVL树的插入

AVL树的旋转(难点)

AVL树的验证

AVL树的删除(本文不做具体的模拟实现)

AVL树的性能

AVL树的模拟实现


AVL树的概念

二叉搜索树虽可以缩短查找的效率,但如果数据有序或接近有序二叉搜索树将退化为单支树,查找元素相当于在顺序表中搜索元素,效率低下。

因此,两位俄罗斯的数学家G.M.Adelson - Velskii和E.M.Landis在1962年发明了一种解决上述问题的方法:

当向二叉搜索树中插入新结点后,如果能保证每个结点的左右子树高度之差的绝对值不超过1(需要对树中的结点进行调整),即可降低树的高度,从而减少平均搜索长度。

一棵AVL树或者是空树,或者是具有以下性质的二叉搜索树:

  1. 它的左右子树都是AVL树
  2. 左右子树高度之差(简称平衡因子)的绝对值不超过1(-1 / 0 / 1) -- 注:之所以保持 1 是因为:左右高度差没办法保证能一直相等(例如只有两个结点呢?)

注:这里的 平衡因子 并不是必须的,这只是它的一种实现方式。

如果一棵二叉搜索树是高度平衡的,它就是AVL树。

如果它有n个结点,其高度可保持在O(logN),搜索时间复杂度O(logN)。

AVL树节点的定义

template<class T>
struct AVLTreeNode
{AVLTreeNode(const T& data): _pLeft(nullptr), _pRight(nullptr), _pParent(nullptr), _data(data), _bf(0){}AVLTreeNode<T>* _pLeft;   // 该节点的左孩子AVLTreeNode<T>* _pRight;  // 该节点的右孩子AVLTreeNode<T>* _pParent; // 该节点的双亲T _data;int _bf;				  // 该节点的平衡因子
};

AVL树的插入

AVL树就是在二叉搜索树的基础上引入了平衡因子,因此AVL树也可以看成是二叉搜索树。

那么AVL树的插入过程可以分为两步:

1. 按照二叉搜索树的方式插入新节点

2. 调整节点的平衡因子 (插入一个结点,只会影响到它的祖先的平衡因子)

  • a、如果插入在双亲的左边,双亲平衡因子--
  • b、如果插入在双亲的右边,双亲的平衡因子++

   当经过a、b两步情况过后,此时又可以分为接下来的三种情况

  • c、双亲的平衡因子 == 0,此时双亲所在的子树高度不变,就不再需要继续往上更新平衡因子了,本次插入就算结束了。
  • d、双亲的平衡因子 == 1 or -1,此时双亲所在的子树高度变了,即需要继续往上更新平衡因子(按照 a、b 的规则继续更新),直到到达 c 的情况/出现 e 的情况。
  • e、双亲的平衡因子 == 2 or -2,此时双亲所在的子树已经不平衡了,需要旋转处理。

以上就是 AVL树的插入 的主要流程。

AVL树的旋转(难点)

如果在一棵原本是平衡的AVL树中插入一个新节点,可能造成不平衡,此时必须调整树的结构,使之平衡化。

根据节点插入位置的不同,AVL树的旋转分为四种:

1. 新节点插入较高左子树的左侧---左左:右单旋

2. 新节点插入较高右子树的右侧---右右:左单旋

3. 新节点插入较高左子树的右侧---左右:先左单旋再右单旋

4. 新节点插入较高右子树的左侧---右左:先右单旋再左单旋

注:旋转首先要保证一件事,就是旋转完后这棵树还是符合搜索二叉树的规则的树!!

总结:

假如以pParent为根的子树不平衡,即pParent的平衡因子为2或者 - 2,分以下情况考虑:

1. pParent的平衡因子为2,说明pParent的右子树高,设pParent的右子树的根为pSubR

  • a、当pSubR的平衡因子为 1 时,执行左单旋
  • b、当pSubR的平衡因子为 -1 时,执行右左双旋

2. pParent的平衡因子为 -2,说明pParent的左子树高,设pParent的左子树的根为pSubL

  • a、当pSubL的平衡因子为 -1 时,执行右单旋
  • b、当pSubL的平衡因子为 1 时,执行左右双旋

旋转完成后,原pParent为根的子树个高度降低,已经平衡(新的pParent的平衡因子为0),不需要再向上更新。

补充:如果是单纯的单旋,旋转完之后参与旋转的两个根结点的平衡因子都会变成0。如果是双旋,旋转完之后参与旋转的三个根结点的平衡因子会受一开始新节点插入的位置影响。

AVL树的验证

AVL树是在二叉搜索树的基础上加入了平衡性的限制,因此要验证AVL树,可以分两步:

1.验证其为二叉搜索树

   如果中序遍历可得到一个有序的序列,就说明为二叉搜索树。

2.验证其为平衡树

   每个节点子树高度差的绝对值不超过1(注意节点中如果没有平衡因子),节点的平衡因子是否计算正确。

AVL树的删除(本文不做具体的模拟实现)

因为AVL树也是二叉搜索树,可按照二叉搜索树的方式将节点删除,然后再更新平衡因子,最差情况下一直要调整到根节点的位置。

具体实现可参考《算法导论》(基本都是伪代码,不一定好看懂) 或《数据结构 - 用面向对象方法与C++描述》殷人昆版。

注:删除调整平衡因子的判断条件和插入不同,删除是当平衡因子为 0 时,就继续往上更新,为 1 或 -1 才停止,为 2 或 -2 就要旋转。

主要逻辑:

  1. 先按搜索树的规则进行删除
  2. 再进行平衡因子的更新(出现不平衡旋转一下)

AVL树的性能

AVL树是一棵绝对平衡的二叉搜索树,其要求每个节点的左右子树高度差的绝对值都不超过1,这样可以保证查询时高效的时间复杂度,即log(N)。但是如果要对AVL树做一些结构修改的操作时,性能非常低下,比如:插入时要维护其绝对平衡,旋转的次数比较多,更差的是在删除时,有可能一直要让旋转持续到根的位置。

因此:如果需要一种查询高效且有序的数据结构,而且数据的个数为静态的(即不会改变),可以考虑AVL树,但一个结构经常修改,就不太适合。

AVL树的模拟实现

AVLTree.h:

#pragma once#include <iostream>
#include <vector>
#include <assert.h>template <class K, class V>
struct AVLTreeNode
{AVLTreeNode<K, V> *_left;AVLTreeNode<K, V> *_right;AVLTreeNode<K, V> *_parent; // 注意:AVL树 这里还需要多一个 _parent,方便往上更新 平衡因子 // 没有 _parent 会很麻烦std::pair<K, V> _kv;int _bf; // balance factorAVLTreeNode(const std::pair<K, V> &kv): _left(nullptr), _right(nullptr), _parent(nullptr), _kv(kv), _bf(0){}
};template <class K, class V>
class AVLTree
{typedef AVLTreeNode<K, V> Node;public:bool insert(const std::pair<K, V> &kv){if (_root == nullptr){_root = new Node(kv);return true;}Node *parent = nullptr;Node *cur = _root;while (cur){parent = cur;if ((cur->_kv).first < kv.first){cur = cur->_right;}else if ((cur->_kv).first > kv.first){cur = cur->_left;}else{return false;}}cur = new Node(kv);if ((parent->_kv).first > kv.first){parent->_left = cur;}else{parent->_right = cur;}cur->_parent = parent; // AVL树的 插入这里要比 普通搜索二叉树 的插入多一步// 到这里,上面的代码仅仅只是完成了插入一个结点的工作// 接下来,开始更新平衡因子 _bfwhile (parent) // 可能存在一直更新都没触发下面的两个break的情况,根据分析一直更新下去,最后 parent 是变成 nullptr。{// 向左插入,--_bfif (cur == parent->_left){--parent->_bf;}// 向右插入,++_bfelse{++parent->_bf;}if (parent->_bf == 0) // 调整好了{break;}else if (parent->_bf == 1 || parent->_bf == -1) // 继续往上更新{cur = parent;parent = parent->_parent;}else if (parent->_bf == 2 || parent->_bf == -2) // 当前子树出问题了,需要旋转调整一下平衡{if (parent->_bf == -2 && cur->_bf == -1) // 右单旋{RotateR(parent);}else if (parent->_bf == 2 && cur->_bf == 1) // 左单旋{RotateL(parent);}// 注:一正一负就是双旋else if (parent->_bf == 2 && cur->_bf == -1) // 右左双旋 {RotateRL(parent);}else if (parent->_bf == -2 && cur->_bf == 1) // 左右双旋{RotateLR(parent);}// 旋转完了,也可以直接结束循环// 原因:旋转之后,该出问题的子树变平衡了,并且该子树的高度和未插入这个最新结点时的高度是一样的。break;}else // 理论上不可能还有其他情况,出现其他情况证明你代码写出问题了{assert(false); // 直接断言报错}}return true;}void RotateR(Node *parent) // 右单旋 // 这里的 parent 的 _bf == -2{// 要抬高左边,降低右边Node *subL = parent->_left; // 这里的 subL 的 _bf 为 -1Node *subLR = subL->_right; // subL 的右子树 要给给 parent,当 parent 的左子树// subL的值 < subLR的值 < parent的值// 处理一个节点的同时也要记得处理它的父亲parent->_left = subLR; // 将 subL 的右子树给给 parentif (subLR)			   // 这边要注意 subLR 可能是 nullptrsubLR->_parent = parent;subL->_right = parent; // 提高左边subL->_parent = parent->_parent; // 要注意,要先把 parent 的 parent 给给 subLif (parent->_parent)			 // 还要注意改 parent 的 parent  // 因为 parent 可能就是根,那么 parent 的 parent 就是 nullptr,所以这里要 if 判断一下{if (parent->_parent->_left == parent){parent->_parent->_left = subL;}else{parent->_parent->_right = subL;}}else // 如果原本的 parent 是根的话,就更新一下根{_root = subL;}parent->_parent = subL; // 降低右边// 根据图分析,旋转完之后,subL的_bf 和 parent的_bf 都为 0// 所以,更新一下两者的_bfparent->_bf = subL->_bf = 0;}void RotateL(Node *parent) // 左单旋 // 这里的 parent 的 _bf == 2{// 要抬高右边,降低左边Node *subR = parent->_right; // 这里的 subR 的 _bf 为 1Node *subRL = subR->_left;	 // subR 的左子树 要给给 parent,当 parent 的右子树// parent的值 < subRL的值 < subR的值// 处理一个节点的同时也要记得处理它的父亲parent->_right = subRL; // 将 subR 的右子树给给 parentif (subRL)				// 这边要注意 subRL 可能是 nullptrsubRL->_parent = parent;subR->_left = parent; // 提高右边subR->_parent = parent->_parent; // 要注意,要先把 parent 的 parent 给给 subRif (parent->_parent)			 // 还要注意改 parent 的 parent  // 因为 parent 可能就是根,那么 parent 的 parent 就是 nullptr,所以这里要 if 判断一下{if (parent->_parent->_left == parent){parent->_parent->_left = subR;}else{parent->_parent->_right = subR;}}else // 如果原本的 parent 是根的话,就更新一下根{_root = subR;}parent->_parent = subR; // 降低右边// 根据图分析,旋转完之后,subR的_bf 和 parent的_bf 都为 0// 所以,更新一下两者的_bfparent->_bf = subR->_bf = 0;}// 注:双旋的难点在于对节点的平衡因子的调节void RotateRL(Node *parent) // 右左双旋(就是对 左单旋和右单旋 的封装) // 这里的 parent 的 _bf == 2{// 因为单旋会导致这几个结点的平衡因子都直接变为0,这可能和实际分析的最终结果不符,需要一些特殊处理Node *subR = parent->_right; // 这里的 subR 的 _bf 为 -1Node *subRL = subR->_left;int bf = subRL->_bf;// 先对 parent->_right 进行右单旋,然后这颗子树就变为了单纯的右边高RotateR(parent->_right);// 再对 parent 自己本身进行左单旋即可RotateL(parent);// 调整为正确的平衡因子(虽然上面的旋转函数对平衡因子的改变有一部分是正确的,但我们下面还是最好全都给它更新一下,做到万无一失)subRL->_bf = 0; // subRL 去做新的根了,根据分析,不管什么情况新的根的_bf都为0。if (bf == -1)	// 代表一开始在 subRL 的左边插入{subR->_bf = 1;	 // subRL 的右子树(h-1)给了 subR 做左子树parent->_bf = 0; // subRL 的左子树(h)给了 parent 做右子树}else if (bf == 1) // 代表一开始在 subRL 的右边插入{subR->_bf = 0;	  // subRL 的右子树(h)给了 subR 做左子树parent->_bf = -1; // subRL 的左子树(h-1)给了 parent 做右子树}else if (bf == 0) // 代表 subRL 自己就是新增的结点{subR->_bf = 0;parent->_bf = 0;}else // 理论不应该还有其他的情况{assert(false);}}void RotateLR(Node *parent) // 左右双旋(就是对 左单旋和右单旋 的封装) // 这里的 parent 的 _bf == -2{// 因为单旋会导致这几个结点的平衡因子都直接变为0,这可能和实际分析的最终结果不符,需要一些特殊处理Node *subL = parent->_left; // 这里的 subL 的 _bf 为 1Node *subLR = subL->_right;int bf = subLR->_bf;// 先对 parent->_left 进行左单旋,然后这颗子树就变为了单纯的左边高RotateL(parent->_left);// 再对 parent 自己本身进行右单旋即可RotateR(parent);// 调整为正确的平衡因子(虽然上面的旋转函数对平衡因子的改变有一部分是正确的,但我们下面还是最好全都给它更新一下,做到万无一失)subLR->_bf = 0; // subLR 去做新的根了,根据分析,不管什么情况新的根的_bf都为0。if (bf == -1)	// 代表一开始在 subLR 的左边插入{subL->_bf = 0;	 // subLR 的左子树(h)给了 subL 做右子树parent->_bf = 1; // subLR 的右子树(h-1)给了 parent 做左子树}else if (bf == 1) // 代表一开始在 subLR 的右边插入{subL->_bf = -1;	 // subLR 的左子树(h-1)给了 subL 做右子树parent->_bf = 0; // subLR 的右子树(h)给了 parent 做左子树}else if (bf == 0) // 代表 subLR 自己就是新增的结点{subL->_bf = 0;parent->_bf = 0;}else // 理论不应该还有其他的情况{assert(false);}}Node *Find(const std::pair<K, V> &kv){Node *cur = _root;while (cur){if ((cur->_kv).first < kv.first){cur = cur->_right;}else if ((cur->_kv).first > kv.first){cur = cur->_left;}else{return cur;}}return nullptr;}void InOrder(){_InOrder(_root);std::cout << std::endl;}bool IsBalance() // 下面的遍历打印,只能证明这是一颗搜索二叉树,并不能说明它是否平衡{return _IsBalance(_root);}int Height() // 返回树的高度{return _Height(_root);}int Size() // 计算树的大小{return _Size(_root);}private:bool _IsBalance(Node *root) // 我们要做的事就是不断的去看每个结点的平衡因子就好(直接前序遍历就行){if (root == nullptr)return true;if (root->_bf <= -2 || 2 <= root->_bf) // 这也算剪枝return false;bool left = _IsBalance_k(root->_left);if (left == false)return false; // 剪枝bool right = _IsBalance_k(root->_right);if (right == false)return false; // 剪枝return left && right;}int _Size(Node *root){return root == nullptr ? 0 : _Size(root->_left) + _Size(root->_right) + 1;}int _Height(Node *root){if (root == nullptr)return 0;return std::max(_Height(root->_left), _Height(root->_right)) + 1;}void _InOrder(Node *root /* = _root */) // 注意:这里不能给缺省值 _root,因为 _root 需要this指针调用,但是this指针本身就是形参,这样写玩不了。{if (root == nullptr)return;_InOrder(root->_left);													  // 左std::cout << (root->_kv).first << ":" << (root->_kv).second << std::endl; // 根_InOrder(root->_right);													  // 右}private:Node *_root = nullptr;
};void TestAVL_Insert_Balance1()
{int a[] = {8, 3, 1, 10, 6, 4, 7, 14, 13};int a1[] = {16, 3, 7, 11, 9, 26, 18, 14, 15};int a2[] = {4, 2, 6, 1, 3, 5, 15, 7, 16, 14};AVLTree<int, int> t;for (auto &x : a2){t.insert({x, x});std::cout << x << "->" << t.IsBalance() << std::endl;}t.InOrder(); // 这里有个问题,没法给 InOrder() 这个函数传参,因为 _root 是私有函数,你在这里调不动。// 那么该怎么解决呢?// 给个缺省值吗? 这是不行的,给不了// 那该怎么办?// 三种方法:1、把这个测试函数定义成友元。(这个方法很不好,就一个测试函数又不是要经常用,定义成友元有点太没边界感了)//           2、学Java,弄一个 Get() 函数,把 _root 拿出来。//			 3、看上面的操作。(封装一下,套一层)std::cout << t.IsBalance() << std::endl;
}
// 补充:对于这种比较复杂的程序,如果哪里出了bug,有个好方法就是在某些关键的步骤或循环里加个打印,这种方式比较容易知道哪里出错了。void TestAVL_Insert_Balance2()
{const int N = 100000000; // 注:32位环境下,这里插入 1亿 个节点会失败,因为空间不够,new 爆了srand((unsigned int)time(nullptr));std::vector<int> v(N);AVLTree<int, int> t;for (int i = 0; i < N; ++i){v[i] = rand() + i;}for (auto x : v){t.insert({x, x});}std::cout << "t.Height():" << t.Height() << std::endl;std::cout << "t.Size():" << t.Size() << std::endl;std::cout << "t.IsBalance():" << t.IsBalance() << std::endl;size_t begin = clock();for (auto x : v){t.Find({ x,x });}size_t end = clock();std::cout << "t.Find():" << end - begin << std::endl;
}
// 注:程序主要时间消耗在于插入时的 new 节点。

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

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

相关文章

网络安全学习中,web渗透的测试流程是怎样的?

渗透测试是什么&#xff1f;网络安全学习中&#xff0c;web渗透的测试流程是怎样的&#xff1f; 渗透测试就是利用我们所掌握的渗透知识&#xff0c;对网站进行一步一步的渗透&#xff0c;发现其中存在的漏洞和隐藏的风险&#xff0c;然后撰写一篇测试报告&#xff0c;提供给我…

Bitbucket 设置SSH KEY方法

0 Preface/Foreword SSH具有传输安全特点&#xff0c;受到广泛使用。 1 添加方法 Bitbucket也是代码托管平台&#xff0c;跟GitLab类似。SSH key的设置方法也跟GitLab类似。 在个人profile设置界面&#xff0c;添加SSH KEY。

和鲸科技推出人工智能通识课程解决方案,助力AI人才培养

2025年2月&#xff0c;教育部副部长吴岩应港澳特区政府邀请&#xff0c;率团赴港澳宣讲《教育强国建设规划纲要 (2024—2035 年)》。在港澳期间&#xff0c;吴岩阐释了教育强国目标的任务&#xff0c;并与特区政府官员交流推进人工智能人才培养的办法。这一系列行动体现出人工智…

Ollama下载安装+本地部署DeepSeek+UI可视化+搭建个人知识库——详解!(Windows版本)

目录 1️⃣下载和安装Ollama 1. &#x1f947;官网下载安装包 2. &#x1f948;安装Ollama 3.&#x1f949;配置Ollama环境变量 4、&#x1f389;验证Ollama 2️⃣本地部署DeepSeek 1. 选择模型并下载 2. 验证和使用DeepSeek 3️⃣使用可视化工具 1. Chrome插件-Page …

STM32中使用PWM对舵机控制

目录 1、硬件JIE 2、PWM口配置 3、角度转换 4、main函数中应用 5、工程下载连接 1、硬件介绍 单片机&#xff1a;STM32F1 舵机&#xff1a;MG995 2、PWM口配置 20毫秒的PWM脉冲占空比&#xff0c;对舵机控制效果较好 计算的公式&#xff1a; PSC、ARR值的选取&#xf…

Java+Vue+uniapp微信小程序校园自助打印系统(程序+论文+讲解+安装+调试+售后)

感兴趣的可以先收藏起来&#xff0c;还有大家在毕设选题&#xff0c;项目以及论文编写等相关问题都可以给我留言咨询&#xff0c;我会一一回复&#xff0c;希望帮助更多的人。 系统介绍 在当今时代&#xff0c;网络与科学技术正以前所未有的速度迅猛发展&#xff0c;这股强大…

如何利用爬虫测试1688商品详情接口

在电商数据分析、市场调研以及商品信息管理等领域&#xff0c;获取1688商品详情数据具有重要意义。虽然1688开放平台提供了官方API接口&#xff0c;但通过爬虫技术获取数据也是一种高效且灵活的方式。本文将详细介绍如何利用爬虫测试1688商品详情接口&#xff0c;包括环境搭建、…

期权帮|国内期权交易投资人做卖出期权价差交易收取的保证金是单边的还是双向的?

锦鲤三三每日分享期权知识&#xff0c;帮助期权新手及时有效地掌握即市趋势与新资讯&#xff01; 国内期权交易投资人做卖出期权价差交易收取的保证金是单边的还是双向的? 在国内期权交易中&#xff0c;投资人做卖出期权价差交易时收取的保证金通常是单边的&#xff0c;但具…

spring security

DefaultLoginPageGeneratingFilter 生成默认的登录页 只有当 登录请求、登录错误、退出登录成功时&#xff0c;才返回登录页面 DefaultLogoutPageGeneratingFilter 退出登录页 只有 logout时返回 spring security 开箱即用&#xff0c;主要是做一些配置&#xff0c;下面是基本…

vue2版本elementUI的table分页实现多选逻辑

1. 需求 我们需要在表格页上实现多选要求&#xff0c;该表格支持分页逻辑。 2. 认识属性 表格属性 参数说明类型可选值默认值data显示的数据array——row-key行数据的 Key&#xff0c;用来优化 Table 的渲染&#xff1b;在使用 reserve-selection 功能与显示树形数据时&…

专业的UML开发工具StarUML

专业的UML开发工具StarUML 可靠的软件建模软件StarUML StarUML 是一款支持统一建模语言 (UML)框架的开源建模软件。它提供了几种类型的图表&#xff0c;并允许用户生成多种语言的代码。在它的帮助下&#xff0c;软件开发人员可以创建设计、概念和编码解决方案。但是&#xff0…

wav格式的音频压缩,WAV 转 MP3 VBR 体积缩减比为 13.5%、多个 MP3 格式音频合并为一个、文件夹存在则删除重建,不存在则直接建立

&#x1f947; 版权: 本文由【墨理学AI】原创首发、各位读者大大、敬请查阅、感谢三连 &#x1f389; 声明: 作为全网 AI 领域 干货最多的博主之一&#xff0c;❤️ 不负光阴不负卿 ❤️ 文章目录 问题一&#xff1a;wav格式的音频压缩为哪些格式&#xff0c;网络传输给用户播放…

利用node.js搭配express框架写后端接口(一)

Node.js 凭借其高效的非阻塞 I/O 操作、事件驱动架构以及轻量级的特点&#xff0c;成为了开发高性能服务器应用的热门选择。Express 框架作为 Node.js 上最流行的 Web 应用框架之一&#xff0c;以其简洁的 API 和丰富的中间件生态系统&#xff0c;极大地简化了 Web 后端开发流程…

黑马Java面试教程_P5_微服务

系列博客目录 文章目录 系列博客目录1.引言2.Spring Cloud2.1 Spring Cloud 5大组件有哪些?面试文稿 2.2 服务注册和发现是什么意思?Spring Cloud 如何实现服务注册发现?面试文稿 2.3 我看你之前也用过nacos、你能说下nacos与eureka的区别?面试文稿 2.4 你们项目负载均衡如…

深入了解 Python 中的 MRO(方法解析顺序)

文章目录 深入了解 Python 中的 MRO&#xff08;方法解析顺序&#xff09;什么是 MRO&#xff1f;如何计算 MRO&#xff1f;C3 算法的合并规则C3 算法的合并步骤示例&#xff1a;合并过程解析 MRO 解析失败的场景使用 mro() 方法查看 MRO示例 1&#xff1a;基本用法 菱形继承与…

信息系统的安全防护

文章目录 引言**1. 物理安全****2. 网络安全****3. 数据安全****4. 身份认证与访问控制****5. 应用安全****6. 日志与监控****7. 人员与管理制度****8. 其他安全措施****9. 安全防护框架**引言 从技术、管理和人员三个方面综合考虑,构建多层次、多维度的安全防护体系。 信息…

Tailwind CSS 4【实用教程】

官网 https://tailwindcss.com/docs/installation/using-vite Tailwind CSS 是一个实用优先的 CSS 框架 特色 原子化样式类名可深度定制主题插件丰富 安装配置导入 vite 中 pnpm add tailwindcss tailwindcss/vitevite.config.ts 中配置 import tailwindcss from tailwindcs…

ChatGPT 提示词框架

作为一个资深安卓开发工程师&#xff0c;我们在日常开发中经常会用到 ChatGPT 来提升开发效率&#xff0c;比如代码优化、bug 排查、生成单元测试等。 但要想真正发挥 ChatGPT 的潜力&#xff0c;我们需要掌握一些提示词&#xff08;Prompt&#xff09;的编写技巧&#xff0c;并…

毕业项目推荐:基于yolov8/yolo11的苹果叶片病害检测识别系统(python+卷积神经网络)

文章目录 概要一、整体资源介绍技术要点功能展示&#xff1a;功能1 支持单张图片识别功能2 支持遍历文件夹识别功能3 支持识别视频文件功能4 支持摄像头识别功能5 支持结果文件导出&#xff08;xls格式&#xff09;功能6 支持切换检测到的目标查看 二、数据集三、算法介绍1. YO…

【Python 入门基础】—— 人工智能“超级引擎”,AI界的“瑞士军刀”,

欢迎来到ZyyOvO的博客✨&#xff0c;一个关于探索技术的角落&#xff0c;记录学习的点滴&#x1f4d6;&#xff0c;分享实用的技巧&#x1f6e0;️&#xff0c;偶尔还有一些奇思妙想&#x1f4a1; 本文由ZyyOvO原创✍️&#xff0c;感谢支持❤️&#xff01;请尊重原创&#x1…