返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C++ AVL树(四种旋转,插入)
  • 171
分享到

C++ AVL树(四种旋转,插入)

c++AVL树高度平衡二叉搜索树 2023-12-25 12:12:42 171人浏览 薄情痞子
摘要

c++ AVL树[四种旋转,插入] 一.AVL树的概念及性质二.我们要实现的大致框架1.AVL树的节点定义2.AVL树的大致框架 三.插入1.插入逻辑跟BST相同的那一部分2.修改平衡因子1.前置说明2.画图演示1.情况1(一直

一.AVL树的概念及性质

AVL树又称高度平衡二叉搜索树,它的高度接近log[2]N(以2为底N的对数),整棵树的形状接近完全二叉树
增删查改的时间复杂度是O(log[2]N)
在这里插入图片描述
本节我们实现的是Key-Value模型的AVL树

二.我们要实现的大致框架

1.AVL树的节点定义

这里我们的AVL树节点比起普通的二叉树的节点来说多了两个成员
第一个是平衡因子,这里我们定义的平衡因子是右子树高度-左子树高度
第二个是指向父节点的指针,方便找到父亲节点从而方便旋转的实现

template<class K,class V>struct AVLTreenode{AVLTreeNode(const pair<K,V>& data = pair<K,V>()): _pLeft(nullptr), _pRight(nullptr), _pParent(nullptr), _data(data), _bf(0){}AVLTreeNode<K,V>* _pLeft;AVLTreeNode<K,V>* _pRight;AVLTreeNode<K,V>* _pParent;pair<K,V> _data;int _bf;   // 节点的平衡因子};

存放的数据类型是一个pair
pair的第一个值是Key
第二个值是Value

2.AVL树的大致框架

// AVL: 二叉搜索树 + 平衡因子的限制template<class K,class V>class AVLTree{typedef AVLTreeNode<K,V> Node;public://构造函数AVLTree(): _pRoot(nullptr){}// 在AVL树中插入值为data的节点bool Insert(const pair<K,V>& data);// AVL树的验证bool IsAVLTree(){return _IsAVLTree(_pRoot);}void InOrder(){_InOrder(_pRoot);}private://中序遍历void _InOrder(Node* root);// 根据AVL树的概念验证pRoot是否为有效的AVL树bool _IsAVLTree(Node* pRoot);//求树的高度size_t _Height(Node* pRoot);// 右单旋void RotateR(Node* pParent);// 左单旋void RotateL(Node* pParent);// 右左双旋void RotateRL(Node* pParent);// 左右双旋void RotateLR(Node* pParent);private:Node* _pRoot;};

三.插入

它的插入的大体逻辑跟二叉搜索树(BST)的插入逻辑很像
只不过需要考虑平衡因子的修改以及旋转
我们先把跟二叉搜索树一样的部分写出来

1.插入逻辑跟BST相同的那一部分

// 在AVL树中插入值为data的节点bool Insert(const pair<K,V>& data){if (_pRoot == nullptr){_pRoot = new Node(data);return true;}Node* cur = _pRoot, * parent = nullptr;//1.找插入位置while (cur){if (cur->_data > data){parent = cur;cur = cur->_pLeft;}else if (cur->_data < data){parent = cur;cur = cur->_pRight;}else{return false;}}//2.new一个新节点,开始链接cur=new Node(data);//下面就是链接节点,修改平衡因子了,这也是AVL树相比于普通的BST的插入逻辑不同的地方return true;}

2.修改平衡因子

1.前置说明

首先要说明:
1.新插入的节点的平衡因子是0,是在AVL树节点的构造函数当中进行初始化的
2.在插入节点的过程中,我们一直维持平衡因子的值为-1/0/1
3.一旦平衡因子到达了2或者-2,说明此时不满足AVL树的性质,需要进行旋转来调整平衡因子,使平衡因子恢复到-1/0/1的状态

2.画图演示

下面我们来画图演示一下插入节点后对于整棵树的节点的平衡因子的影响

1.情况1(一直影响到根节点为止)

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
那不就是一直向上影响不就行了吗?
不是的,比如下面这种情况

2.情况2(在影响到根节点之前影响消失了)

为了方便演示,把节点8的值改成了9
新插入的值为8

在这里插入图片描述
在这里插入图片描述

3.深剖情况1和2

第一种情况时:
在这里插入图片描述
在这里插入图片描述
第二种情况时:
在这里插入图片描述
在这里插入图片描述

4.总结

在这里插入图片描述

3.考虑旋转

在修改平衡因子时,如果某个节点的平衡因子修改之后变为了2或者-2
说明以这个节点为根节点的子树已经不是AVL树了,需要旋转这个节点

关于如何旋转我们后面会介绍
这里先说明一下什么情况下进行哪种旋转

而旋转分为4种情况:
1.左单旋

1.左单旋的介绍

在这里插入图片描述
在这里插入图片描述
总结:
在这里插入图片描述

2.右单旋的介绍

了解了左单旋之后,右单旋也就能够很好的理解了
在这里插入图片描述
在这里插入图片描述
总结:
在这里插入图片描述

3.右左双旋的介绍

右左双旋的基础条件跟左单旋的基础条件很像
只不过有一点不一样,我们来看看吧
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
总结:
在这里插入图片描述

4.左右双旋的介绍

同理,左右双旋跟右单旋的基础条件也很像
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
总结:
在这里插入图片描述

5.旋转条件的总结:

在这里插入图片描述

4.插入逻辑的完善

因此我们就能够写出这样的insert代码

// 在AVL树中插入值为data的节点bool Insert(const pair<K,V>& data){if (_pRoot == nullptr){_pRoot = new Node(data);return true;}Node* cur = _pRoot, * parent = nullptr;//1.找插入位置while (cur){if (cur->_data > data){parent = cur;cur = cur->_pLeft;}else if (cur->_data < data){parent = cur;cur = cur->_pRight;}else{return false;}}//2.插入节点调整平衡因子cur = new Node(data);//在parent左边插入节点if (parent->_data > data){parent->_pLeft = cur;parent->_bf--;}//在parent的右边插入节点else{parent->_pRight = cur;parent->_bf++;}cur->_pParent = parent;//3.向上影响平衡因子//影响的结束条件://1.cur到达_pRoot,也就是parent到达nullptr//2.parent的bf调整之后变为0//因为0只能由1或者-1变过来//而且由1或者-1变成0时,parent这个树的高度没有发生变化,因此不会在往上去影响了//3.parent这棵树需要旋转//旋转之后会达到1个目的://降低parent这棵树的高度,降为插入这个结点之前的高度//因此此时就不会在往上去影响了while (parent){//此时无需在往上去影响if (parent->_bf == 0){break;}//此时需要再往上去影响//因为1或者-1只能由0变过来,因此parent这个树的高度变高,需要往上去影响else if (parent->_bf == 1 || parent->_bf == -1){cur = parent;parent = parent->_pParent;if (parent != nullptr){//说明parent是左子树,因此会让祖父的bf--if (parent->_pLeft == cur){parent->_bf--;}//说明parent是右子树,因此会让祖父的bf++else{parent->_bf++;}}}else if (parent->_bf == 2 || parent->_bf == -2){//左单旋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){RotateR(parent);}//左右双旋else if (parent->_bf == -2 && cur->_bf == 1){RotateLR(parent);}break;}else{assert(false);}}return true;}

四.旋转的动图演示和代码实现

我们在上面已经介绍完什么是左、右单旋,右左、左右双旋转了
下面我们来看一下如何实现旋转呢?

1.左单旋

1.步骤+注意事项

在这里插入图片描述

2.动图演示

旋转之前:
在这里插入图片描述
旋转过程:
在这里插入图片描述
旋转完成之后
在这里插入图片描述

3.代码实现

// 左单旋void RotateL(Node* pParent){Node* subR = pParent->_pRight;Node* subRL = subR->_pLeft;Node* grandParent = pParent->_pParent;pParent->_pParent = subR;subR->_pLeft = pParent;pParent->_pRight = subRL;if (subRL)subRL->_pParent = pParent;//说明此时pParent是_pRootif (grandParent == nullptr){_pRoot = subR;_pRoot->_pParent = nullptr;}//说明此时pParent所在的树是一颗子树,需要跟父亲链接else{if (pParent == grandParent->_pLeft){grandParent->_pLeft = subR;}else{grandParent->_pRight = subR;}subR->_pParent = grandParent;}//调整平衡因子pParent->_bf = subR->_bf = 0;}

2.右单旋

同理,右单旋也是类似的思路
而且注意事项跟左单旋如出一辙

1.动图演示

旋转之前:
在这里插入图片描述
旋转过程:
在这里插入图片描述
旋转完成之后:
在这里插入图片描述

2.代码实现

// 右单旋void RotateR(Node* pParent){Node* subL = pParent->_pLeft;Node* subLR = subL->_pRight;Node* grandParent = pParent->_pParent;subL->_pRight = pParent;pParent->_pParent = subL;pParent->_pLeft = subLR;if (subLR)subLR->_pParent = pParent;if (grandParent == nullptr){_pRoot = subL;_pRoot->_pParent = nullptr;}else{if (pParent == grandParent->_pLeft){grandParent->_pLeft = subL;}else{grandParent->_pRight = subL;}subL->_pParent = grandParent;}//修改平衡因子subL->_bf = pParent->_bf = 0;}

3.右左双旋

右左双旋和左右双旋都是对左单旋和右单旋的复用,这里就不在赘述了
直接上动图演示

1.先右旋

旋转之前:
在这里插入图片描述
旋转过程:
在这里插入图片描述
旋转之后:
在这里插入图片描述

2.再左旋

旋转之前:
在这里插入图片描述
旋转过程:
在这里插入图片描述
旋转之后:
在这里插入图片描述

3.代码实现

// 右左双旋void RotateRL(Node* pParent){Node* subR = pParent->_pRight;Node* subRL = subR->_pLeft;int bf = subRL->_bf;//对subR进行一次右旋RotateR(subR);//在对pParent进行一次左旋RotateL(pParent);//这两次旋转达到了一个目的:把subRL的左子树给pParent成为pParent的右子树//把subRL的右子树给subR成为subR的左子树//根据旋转前subRL的平衡因子调整平衡后的平衡因子if (bf == 0){subR->_bf = pParent->_bf = subRL->_bf = 0;}//说明subRL的左子树更低else if (bf == 1){pParent->_bf = -1;subR->_bf = subRL->_bf = 0;}else if (bf == -1){subR->_bf = 1;pParent->_bf = subRL->_bf = 0;}else{assert(false);}}

4.左右双旋

1.先左旋

旋转之前:
在这里插入图片描述
旋转过程:
在这里插入图片描述
旋转之后:
在这里插入图片描述

2.再右旋

旋转之前:
在这里插入图片描述
旋转过程:
在这里插入图片描述
旋转之后:
在这里插入图片描述

3.代码实现

// 左右双旋void RotateLR(Node* pParent){Node* subL = pParent->_pLeft;Node* subLR = subL->_pRight;int bf = subLR->_bf;RotateL(subL);RotateR(pParent);//旋转的过程就是把subLR的左子树给subL成为subL的右子树//把subLR的右子树给pParent成为pParent的左子树if (bf == 0){subL->_bf = subLR->_bf = pParent->_bf = 0;}else if (bf == 1){subL->_bf = -1;subLR->_bf = pParent->_bf = 0;}else if (bf == -1){pParent->_bf = 1;subL->_bf = subLR->_bf = 0;}else{assert(false);}}

五.AVL树的验证

为了验证AVL树的正确性
我们添加中序遍历代码,求高度代码,验证左右子树高度差不大于1的代码

// AVL树的验证bool IsAVLTree(){return _IsAVLTree(_pRoot);}void InOrder(){_InOrder(_pRoot);}private:void _InOrder(Node* root){if (root == nullptr) return;_InOrder(root->_pLeft);cout << root->_data.first << " " << root->_data.second << " ";_InOrder(root->_pRight);}// 根据AVL树的概念验证pRoot是否为有效的AVL树bool _IsAVLTree(Node* pRoot){if (pRoot == nullptr) return true;int leftHeight = _Height(pRoot->_pLeft);int rightHeight = _Height(pRoot->_pRight);return abs(leftHeight - rightHeight) < 2 && _IsAVLTree(pRoot->_pLeft) && _IsAVLTree(pRoot->_pRight);}size_t _Height(Node* pRoot){if (pRoot == nullptr){return 0;}int leftHeight = _Height(pRoot->_pLeft);int rightHeight = _Height(pRoot->_pRight);return max(leftHeight, rightHeight) + 1;}

下面是测试代码

#include "AVLTree.h"#include int test1(){//int a[] = { 16, 3, 7, 11, 9, 26, 18, 14, 15 };int a[] = { 4, 2, 6, 1, 3, 5, 15, 7, 16, 14 };AVLTree<int,string> tree;for (auto& e : a){cout << e << " : " << tree.Insert(make_pair(e,"wzs")) << endl;}cout << endl;tree.InOrder();cout << endl;cout << tree.IsAVLTree() << endl;return 0;}int test2(){const int N = 300000;vector<int> v;v.reserve(N);srand(time(0));for (size_t i = 0; i < N; i++){v.push_back(rand() + i);//cout << v.back() << endl;}AVLTree<int,int> t;for (auto e : v){if (e == 14604){int x = 0;}t.Insert(make_pair(e,e));//cout << "Insert:" << e << "->" << t.IsAVLTree()<< endl;}cout << t.IsAVLTree() << endl;return 0;}

在这里插入图片描述
验证成功

六.完整代码

1.AVLTree.h:

#pragma once#include using namespace std;#include template<class K,class V>struct AVLTreeNode{AVLTreeNode(const pair<K,V>& data = pair<K,V>()): _pLeft(nullptr), _pRight(nullptr), _pParent(nullptr), _data(data), _bf(0){}AVLTreeNode<K,V>* _pLeft;AVLTreeNode<K,V>* _pRight;AVLTreeNode<K,V>* _pParent;pair<K,V> _data;int _bf;   // 节点的平衡因子};// AVL: 二叉搜索树 + 平衡因子的限制template<class K,class V>class AVLTree{typedef AVLTreeNode<K,V> Node;public:AVLTree(): _pRoot(nullptr){}// 在AVL树中插入值为data的节点bool Insert(const pair<K,V>& data){if (_pRoot == nullptr){_pRoot = new Node(data);return true;}Node* cur = _pRoot, * parent = nullptr;//1.找插入位置while (cur){if (cur->_data > data){parent = cur;cur = cur->_pLeft;}else if (cur->_data < data){parent = cur;cur = cur->_pRight;}else{return false;}}//2.插入节点调整平衡因子cur = new Node(data);//在parent左边插入节点if (parent->_data > data){parent->_pLeft = cur;parent->_bf--;}//在parent的右边插入节点else{parent->_pRight = cur;parent->_bf++;}cur->_pParent = parent;//3.向上影响平衡因子//影响的结束条件://1.cur到达_pRoot,也就是parent到达nullptr//2.parent的bf调整之后变为0//因为0只能由1或者-1变过来//而且由1或者-1变成0时,parent这个树的高度没有发生变化,因此不会在往上去影响了//3.parent这棵树需要旋转//旋转之后会达到1个目的://降低parent这棵树的高度,降为插入这个结点之前的高度//因此此时就不会在往上去影响了while (parent){//此时无需在往上去影响if (parent->_bf == 0){break;}//此时需要再往上去影响//因为1或者-1只能由0变过来,因此parent这个树的高度变高,需要往上去影响else if (parent->_bf == 1 || parent->_bf == -1){cur = parent;parent = parent->_pParent;if (parent != nullptr){//说明parent是左子树,因此会让祖父的bf--if (parent->_pLeft == cur){parent->_bf--;}//说明parent是右子树,因此会让祖父的bf++else{parent->_bf++;}}}else if (parent->_bf == 2 || parent->_bf == -2){//左单旋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){RotateR(parent);}//左右双旋else if (parent->_bf == -2 && cur->_bf == 1){RotateLR(parent);}break;}else{assert(false);}}return true;}// AVL树的验证bool IsAVLTree(){return _IsAVLTree(_pRoot);}void InOrder(){_InOrder(_pRoot);}private:void _InOrder(Node* root){if (root == nullptr) return;_InOrder(root->_pLeft);cout << root->_data.first << " " << root->_data.second << " ";_InOrder(root->_pRight);}// 根据AVL树的概念验证pRoot是否为有效的AVL树bool _IsAVLTree(Node* pRoot){if (pRoot == nullptr) return true;int leftHeight = _Height(pRoot->_pLeft);int rightHeight = _Height(pRoot->_pRight);return abs(leftHeight - rightHeight) < 2 && _IsAVLTree(pRoot->_pLeft) && _IsAVLTree(pRoot->_pRight);}size_t _Height(Node* pRoot){if (pRoot == nullptr){return 0;}int leftHeight = _Height(pRoot->_pLeft);int rightHeight = _Height(pRoot->_pRight);return max(leftHeight, rightHeight) + 1;}// 右单旋void RotateR(Node* pParent){Node* subL = pParent->_pLeft;Node* subLR = subL->_pRight;Node* grandParent = pParent->_pParent;subL->_pRight = pParent;pParent->_pParent = subL;pParent->_pLeft = subLR;if (subLR)subLR->_pParent = pParent;if (grandParent == nullptr){_pRoot = subL;_pRoot->_pParent = nullptr;}else{if (pParent == grandParent->_pLeft){grandParent->_pLeft = subL;}else{grandParent->_pRight = subL;}subL->_pParent = grandParent;}//修改平衡因子subL->_bf = pParent->_bf = 0;}// 左单旋void RotateL(Node* pParent){Node* subR = pParent->_pRight;Node* subRL = subR->_pLeft;Node* grandParent = pParent->_pParent;pParent->_pParent = subR;subR->_pLeft = pParent;pParent->_pRight = subRL;if (subRL)subRL->_pParent = pParent;//说明此时pParent是_pRootif (grandParent == nullptr){_pRoot = subR;_pRoot->_pParent = nullptr;}//说明此时pParent所在的树是一颗子树,需要跟父亲链接else{if (pParent == grandParent->_pLeft){grandParent->_pLeft = subR;}else{grandParent->_pRight = subR;}subR->_pParent = grandParent;}//调整平衡因子pParent->_bf = subR->_bf = 0;}// 右左双旋void RotateRL(Node* pParent){Node* subR = pParent->_pRight;Node* subRL = subR->_pLeft;int bf = subRL->_bf;//对subR进行一次右旋RotateR(subR);//在对pParent进行一次左旋RotateL(pParent);//这两次旋转达到了一个目的:把subRL的左子树给pParent成为pParent的右子树//把subRL的右子树给subR成为subR的左子树//根据旋转前subRL的平衡因子调整平衡后的平衡因子if (bf == 0){subR->_bf = pParent->_bf = subRL->_bf = 0;}//说明subRL的左子树更低else if (bf == 1){pParent->_bf = -1;subR->_bf = subRL->_bf = 0;}else if (bf == -1){subR->_bf = 1;pParent->_bf = subRL->_bf = 0;}else{assert(false);}}// 左右双旋void RotateLR(Node* pParent){Node* subL = pParent->_pLeft;Node* subLR = subL->_pRight;int bf = subLR->_bf;RotateL(subL);RotateR(pParent);//旋转的过程就是把subLR的左子树给subL成为subL的右子树//把subLR的右子树给pParent成为pParent的左子树if (bf == 0){subL->_bf = subLR->_bf = pParent->_bf = 0;}else if (bf == 1){subL->_bf = -1;subLR->_bf = pParent->_bf = 0;}else if (bf == -1){pParent->_bf = 1;subL->_bf = subLR->_bf = 0;}else{assert(false);}}private:Node* _pRoot;};

2.test.cpp

#include "AVLTree.h"#include int test1(){//int a[] = { 16, 3, 7, 11, 9, 26, 18, 14, 15 };int a[] = { 4, 2, 6, 1, 3, 5, 15, 7, 16, 14 };AVLTree<int,string> tree;for (auto& e : a){cout << e << " : " << tree.Insert(make_pair(e,"wzs")) << endl;}cout << endl;tree.InOrder();cout << endl;cout << tree.IsAVLTree() << endl;return 0;}int test2(){const int N = 300000;vector<int> v;v.reserve(N);srand(time(0));for (size_t i = 0; i < N; i++){v.push_back(rand() + i);//cout << v.back() << endl;}AVLTree<int,int> t;for (auto e : v){if (e == 14604){int x = 0;}t.Insert(make_pair(e,e));//cout << "Insert:" << e << "->" << t.IsAVLTree()<< endl;}cout << t.IsAVLTree() << endl;return 0;}int main(){test1();cout << "=============    开始test2的验证   =================" << endl;test2();return 0;}

以上就是C++ AVL树(四种旋转,插入)的全部内容,希望能对大家有所帮助!

来源地址:https://blog.csdn.net/Wzs040810/article/details/135099616

--结束END--

本文标题: C++ AVL树(四种旋转,插入)

本文链接: https://lsjlt.com/news/551782.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • C++ AVL树(四种旋转,插入)
    C++ AVL树[四种旋转,插入] 一.AVL树的概念及性质二.我们要实现的大致框架1.AVL树的节点定义2.AVL树的大致框架 三.插入1.插入逻辑跟BST相同的那一部分2.修改平衡因子1.前置说明2.画图演示1.情况1(一直...
    99+
    2023-12-25
    c++ AVL树 高度平衡二叉搜索树
  • C++AVL树4种旋转详讲(左单旋、右单旋、左右双旋、右左双旋)
    目录引子:AVL树是因为什么出现的?1.AVl树的的特性2.AVl树的框架3.AVL树的插入 3.1四种旋转(左单旋、右单旋、左右双旋、右左双旋)3.1.1左单旋3.1.2...
    99+
    2022-11-13
    c++ avl树 AVL树的旋转 c++实现树
  • C++ AVL树插入新节点后的四种调整情况梳理介绍
    AVL树是一个高度平衡的二叉搜索树 满足二叉搜索树的所有特性。左子树和右子树的高度之差的绝对值不大于1。 此处AVL树结点的定义 template<class K, class...
    99+
    2024-04-02
  • C++的四种类型转换
    目录一、隐式类型转换和显示类型转换二、C++的四种类型转换2.1 static_cast 相似转化2.2 reinterpret_cast 不同类型转化2.3 const_cast ...
    99+
    2023-05-14
    C++实现类型转换方法 C++实现类型转换
  • C#四舍五入的4种方法
    在C#中,有多种方法可以实现四舍五入。以下是四种常用的方法:1. Math.Round() 函数:Math.Round() 函数可以...
    99+
    2023-09-28
    C#
  • 详解C++中四种类型的转换
    目录const_caststatic_castreinterpret_castdynamic_cast代码C语言中我们使用 int a=(int) b;的方式强制转换 C++提供了四...
    99+
    2022-12-08
    C++类型转换 C++ 类型转换符 C++ 转换符
  • C++强制类型转换的四种方式
    目录1 C++类型转换本质1.1 自动类型转换(隐式)1.2 强制类型转换(显式)1.3 类型转换的本质1.4 类型转换的安全性2 四种类型转换运算符2.1 C语言的强制类型转换与C...
    99+
    2024-04-02
  • C++四种强制转换原理与价值
    目录四种强制转换的原理价值意义static_cast(最常用) 与隐式转换的区别新型强制转换降低转换开销,举例demo四种强制转换的原理 static_cast:static_cas...
    99+
    2023-05-16
    C++强制转换 C++强制转换原理 C++强制转换价值
  • C++中的四种强制类型转换介绍
    这篇文章主要讲解了“C++中的四种强制类型转换介绍”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C++中的四种强制类型转换介绍”吧!在了解c++的强制类形转换的时候,先看看在c语言中是怎么进...
    99+
    2023-06-20
  • C++中的四种类型转换符是什么
    本篇内容介绍了“C++中的四种类型转换符是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一:背景在玩 C 的时候,经常会用 v...
    99+
    2023-07-02
  • C++实例讲解四种类型转换的使用
    目录C++类型转换C语言风格的转换C++风格的类型转换static_castreinterpret_castconst_castdynamic_cast小结C++类型转换 C语言风格...
    99+
    2024-04-02
  • 一起聊聊C++中的四种类型转换符
    目录一:背景二:理解四大运算符1. const_cast2. reinterpret_cast3. dynamic_cast3. static_cast一:背景 在玩 C 的时候,经...
    99+
    2024-04-02
  • C++中四种类型转换的方法是什么
    这篇文章主要介绍了C++中四种类型转换的方法是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C++中四种类型转换的方法是什么文章都会有所收获,下面我们一起来看看吧。一、隐式类型转换和显示类型转换当等号两边的...
    99+
    2023-07-05
  • C语言字符串旋转问题的深入讲解
    目录字符串的旋转:实现旋转字符串:1、暴力求解法: 2、三步翻转法判断一个字符串是否由另一个字符串旋转而来1、暴力求解法:2、优化算法:关于用到的函数扩展:1、assert...
    99+
    2024-04-02
  • Yolov5旋转框(斜框)检测tensorrt部署(C++)从入门到入坟
             本博客将从标图到最终采用tensorrt部署yolov5_obb(用于斜框目标检测的yolov5框架),一步一步手把手教你如何成为一个合格的算法搬运工。yolov5_obb是一款用于斜框目标检测的神经网络,具体原理我就不说...
    99+
    2023-09-28
    深度学习 计算机视觉 python
  • 一文搞懂C++中的四种强制类型转换
    在了解c++的强制类形转换的时候,先看看在c语言中是怎么进行强制类形转换的。 C语言中的强制类形转换分为两种  隐式类型转换 显示类型转换 int ...
    99+
    2024-04-02
  • C++11新特性之四种类型转换cast说明
    目录引言1、static_cast1.1 基本类型转换1.2 类的上行转换(安全)1.3 类的下行转换(不安全)2、const_cast2.1 改变常量属性3、dynamic_cas...
    99+
    2023-02-09
    c++11类型转换 c++11类型转换cast C++类型转换
  • C语言二叉排序树的创建,插入和删除
    目录一、二叉排序树(二叉查找树)的概念二、二叉排序树的判别三、二叉排序树的创建(creat、insert)四、二叉排序树的插入五、二插排序树的删除六、完整代码(可以运行)总结一、二叉...
    99+
    2024-04-02
  • C++类型转换引入了那几种方式
    这期内容当中小编将会给大家带来有关C++类型转换引入了那几种方式,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。众所周知C++关于类型转换引入了四种方式:static_castconst_castdynam...
    99+
    2023-06-15
  • C语言数据结构二叉树先序、中序、后序及层次四种遍历
    目录一、图示展示(1)先序遍历(2)中序遍历(3)后序遍历(4)层次遍历(5)口诀二、代码展示一、图示展示 (1)先序遍历 先序遍历可以想象为,一个小人从一棵二叉树根节点为起点,沿着...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作