红黑树介绍与模拟实现(insert+颜色调整精美图示超详解哦)

红黑树

  • 引言
  • 红黑树的介绍
  • 实现
    • 结点类
    • insert
      • 搜索插入位置
      • 插入
      • 调整
        • 当parent为gparent的左子结点
        • 当parent为gparent的右子结点
  • 参考源码
  • 测试红黑树是否合格
  • 总结

引言

在上一篇文章中我们认识了高度平衡的平衡二叉树AVL树:戳我看AVL树详解哦
(关于旋转调整的部分,在AVL树的时候已经详细介绍过了,如果大家对旋转调平衡的部分有疑惑的话,请移步至AVL树的详解)

由于AVL树的高度平衡,其平均搜索时间复杂度几乎可以达到严格的O(logN)。同样也因为平衡的程度很高,在维护平衡上时间的花费对于搜索上时间的提升是得不偿失的。
大多数情况下,我们并不需要很高的平衡程度,只需要达成一种接近平衡的状态,搜索的平均时间复杂度基本达到O(logN)即可。 红黑树就是这样的一种结构,它的最高子树的高度小于等于最低子树的二倍。通过减少调平衡时的时间成本来提高效率

红黑树的介绍

红黑树是平衡二叉树的一种,他在满足二叉搜索树特性的基础上,给每个结点增加了一个颜色属性,包括RedBlack;并要求从根节点到一个叶子结点形成的任意一条路径中,通过对结点颜色的限制规则,没有任何一条路径回比其他的路径长一倍
在这里插入图片描述
(红黑圣诞树)

对于红黑树结点颜色的限制规则如下:

  • 每个结点的颜色只有红色或黑色两种
  • 根结点的颜色一定是黑色的
  • 如果某一个结点的颜色是红色的,它的两个孩子结点的颜色一定是黑色(即不存在两个连续的红色结点);
  • 对于任一结点,到叶子结点的任一路径上,包含的黑色结点的数量一定相等
  • 叶子结点一定是黑色的(这里的叶子结点直最后的nullptr

当满足上面的所有规则时,根节点到叶子结点的任一路径就都不可能比其他路径长一倍。由于不存在连续的红色结点,所以当黑色结点的数量 n 一定时,最长路径的长度为2 * n,最短路径的长度为 n ,所以不可能相差一倍以上。这样就达成了一种相对平衡的状态,并不需要经常去旋转调平了。

实现

红黑树的实现,在之前的二叉搜索树上增加了结点的颜色,以及对于结点的颜色调整的部分;
在本篇文章中依旧实现 K-V的模式 的树, 并且以非递归实现insert
为防止命名冲突,将实现放在我们的命名空间qqq中:

结点的颜色我们使用 枚举常量enum Color 来表示;

RBTree是一个类模板,有两个模板参数,即KV,表示其中存储的索引类型与值类型;
成员变量类型为Node*(由结点类RBTreeNode重命名),表示根结点的指针:

 //基本的代码结构
namespace qqq
{enum Color //枚举常量表示颜色{RED,BLUCK};template<class K, class V>struct RBTreeNode  //结点类{};template<class K, class V>class RBTree  //红黑树{typedef RBTreeNode<K, V> Node;public:bool insert(const pair<K, V>& kv){}protected:Node* _root = nullptr;};
}

结点类

首先,对红黑树的结点进行实现:

RBTreeNode是也一个类模板, 两个模板参数同样为KV,表示索引与值的类型;
在结点中储存数据的结构为pair,其中firstK类型,secondV类型;

成员变量包括结点中的数据 _kv
指向父亲结点的指针 _parent
指向左右孩子结点的指针 _left_right
表示结点颜色的枚举常量 _col

template<class K, class V>
struct RBTreeNode  //结点类
{pair<K, V> _kv;RBTreeNode<K, V>* _parent;RBTreeNode<K, V>* _left;RBTreeNode<K, V>* _right;Color _col;
};

结点类的构造函数
我们需要实现一个默认构造函数:
参数类型为const pair<K, V>,缺省值为一个pair<K, V>的匿名对象;
对于父子结点的指针,在初始化列表中初始化为nullptr即可;
对于结点的颜色,在初始化列表中初始化为RED(要保证每条路上黑色结点的个数相等,就必须初始化为红色):

	RBTreeNode(const pair<K, V> kv = pair<K, V>()): _kv(kv), _left(nullptr), _right(nullptr), _parent(nullptr), _col(RED){}

insert

红黑树的insert分为两个部分,即搜索并插入以及调整使其满足红黑树的性质:

搜索插入位置

  • 与二叉搜索树类似,搜索并插入时,首先用要插入的pair对象创建一个新结点newnode
  • 与此同时,使用结点指针 cur来记录当前位置parent来记录cur的父结点,便于后面插入;
  • 然后 while循环向下查找 插入的位置:当newnode小于cur的元素时,向左查找,当newnode大于cur的元素时,向右查找,相等时即该元素已经存在,返回false
  • curnullptr时,表示找到了插入的位置,循环终止:
// 部分代码:搜索插入位置 //
Node* newnode = new Node(kv);
Node* parent = nullptr;
Node* cur = _root;while (cur != nullptr) //搜索
{if (newnode->_kv.first > cur->_kv.first){parent = cur;cur = parent->_right;}else if (newnode->_kv.first < cur->_kv.first){parent = cur;cur = parent->_left;}else  //相等即插入失败{return false;}
}

插入

  • newnode插入红黑树即将newnodeparent链接,这时就需要判断parent是否为空
  • parent为空时,即cur就是根节点_root,即newnode是这棵红黑树中的第一个结点,将 其赋值给_root 即可;
  • parent不为空时, 还需要判断cur位于parent的左边还是右边,然后再插入:

在这里插入图片描述

//  局部代码:插入newnode //
if (parent == nullptr) //插入
{_root = newnode;
}
else if (newnode->_kv.first < parent->_kv.first)
{parent->_left = newnode;newnode->_parent = parent;cur = newnode;
}
else
{parent->_right = newnode;newnode->_parent = parent;cur = newnode;
}

调整

在完成插入后,首先需要判断的是parent指针的状态:

  • parent为空时,表明当前结点为根结点,将其颜色改为BLACK即可调整完毕;
  • parent指向结点的颜色为BLACK 时,如果是在刚插入时,新插入的结点颜色为RED,不会影响该路径的黑色节点个数,所以不再需要调整。如果是在向上调整的过程中parent指向的结点为BLACK,也意味着整棵树调整结束了(这一点在后面的调整部分会详细介绍);
  • 除了上面不用调整的两种情况外,其余的情况,即 parent指向的结点存在且为RED 的情况就需要进行调整了。调整是自下而上的,循环向上调整,直到 parent为空或parent指向的结点为BLACK时循环结束,调整完成
// 部分代码:自下而上调整的循环框架 //
while (parent != nullptr && parent->_col != BLUCK) 
{}

开始调整时首先对parentgparentcur的祖父结点)的左子结点还是右子结点做一分类讨论(当parent不为黑时,由于根结点必须为黑,所以parent不是根结点,所以gparent一定存在):

当parent为gparent的左子结点

parentgparent的左子结点时,我们首先要考察cur叔叔结点的情况:

  • cur的叔叔结点为RED 时,不需要进行旋转调整,只需要将gparent指向结点设置为RED,将parentcur的叔叔结点全部设置为BLACK即可:
    由于在调整完颜色后,gparent指向结点颜色就为RED了,这时如果gparent父结点的颜色正好为红,就出现了连续的两个红色结点。所以需要cur向上移动两个结点,再对其parent指向的结点继续进行判断:
    在这里插入图片描述

  • cur的叔叔结点为BLACK或不存在时,就需要进行旋转调整了:
    旋转的逻辑与之前AVL树类似,当cur位于parent的左边时,即左左——单次右旋
    在这里插入图片描述
    cur位于parent的右边时,即左右——左右双旋
    在这里插入图片描述
    在这里插入图片描述
    旋转调整后将gparent指向结点设置为RED,将子树顶部的结点设置为BLACK 即可(parent指向的结点或cur指向的结点):
    在这里插入图片描述

经过旋转后的红黑树,子树顶部的颜色一定为黑色,再向上也就不会存在两个连续的红色结点的问题了,所以旋转之后直接终止循环即可。

需要注意的是叔叔是黑色结点的情况一定是出现在调整过程中发生的,当叔叔结点为黑色时,cur下的路径中一定存在着与gparent右子树中黑色结点相同个数的黑色结点,parent下的路径中同样也存在着相同数目的黑色结点,这样在旋转调平衡后,这棵子树中的所有路径中的黑色结点数目与之前是不变的

//部分代码:当parent为gparent左子结点的情况
if (parent == gparent->_left)   
{if (gparent->_right == nullptr || gparent->_right->_col == BLUCK)  //1.叔叔结点不存在或为黑,需要旋转并调色{if (cur == parent->_left)//左左->单次右旋{RotateR(gparent);parent->_col = BLUCK;gparent->_col = RED;}else//左右->左旋+右旋{RotateL(parent);RotateR(gparent);cur->_col = BLUCK;gparent->_col = RED;}break; //通过旋转调整后,该子树的根结点一定是黑,所以可以直接结束循环}else if (gparent->_right->_col == RED)  //2.叔叔结点为红,通过调色即可实现红黑树{//调色parent->_col = BLUCK;gparent->_right->_col = BLUCK;gparent->_col = RED;//继续向上cur = gparent;parent = cur->_parent;}else{assert(0);}
}
当parent为gparent的右子结点

parentgparent的右子结点时,与上面的情况一致,只是左右对调了,所以这里只给出图示与代码(如果在这种情况下遇到了问题,希望你在上面的情况中能够找到答案):

  • 叔叔结点为RED,仅调整颜色
    在这里插入图片描述

  • 叔叔结点为BLACK或不存在,左单旋或右左双旋:
    左单旋
    在这里插入图片描述
    右左双旋
    在这里插入图片描述在这里插入图片描述
    旋转后调整颜色
    在这里插入图片描述

旋转后子树顶部的结点一定为BLACK,所以直接break即可。

//部分代码:当parent为gparent右子结点的情况 //
else //parent == gparent->_right
{if (gparent->_left == nullptr || gparent->_left->_col == BLUCK)  //1.叔叔结点不存在或为黑,需要旋转并调色{if (cur == parent->_right)//右右->单次左旋{RotateL(gparent);parent->_col = BLUCK;gparent->_col = RED;}else//右左->右旋+左旋{RotateR(parent);RotateL(gparent);cur->_col = BLUCK;gparent->_col = RED;}break; //通过旋转调整后,该子树的根结点一定是黑,所以可以直接结束循环}else if (gparent->_left->_col == RED)  //2.叔叔结点为红,通过调色即可实现红黑树{//调色parent->_col = BLUCK;gparent->_left->_col = BLUCK;gparent->_col = RED;//继续向上cur = gparent;parent = cur->_parent;}else{assert(0);}
}

while循环调整结束之后,再将根结点_root的颜色改为BLACK,统一做处理(insert的整体代码在这里就不做展示了,大家跳转至参考源码部分查看即可)。

参考源码

namespace qqq
{enum Color //枚举常量表示颜色{RED,BLUCK};template<class K, class V>struct RBTreeNode  //结点类{pair<K, V> _kv;RBTreeNode<K, V>* _parent;RBTreeNode<K, V>* _left;RBTreeNode<K, V>* _right;Color _col;RBTreeNode(const pair<K, V> kv = pair<K, V>()): _kv(kv), _left(nullptr), _right(nullptr), _parent(nullptr), _col(RED){}};template<class K, class V>class RBTree  //红黑树{typedef RBTreeNode<K, V> Node;public:bool insert(const pair<K, V>& kv){ //先插入Node* newnode = new Node(kv);Node* parent = nullptr;Node* cur = _root;while (cur != nullptr) //搜索{if (newnode->_kv.first > cur->_kv.first){parent = cur;cur = parent->_right;}else if (newnode->_kv.first < cur->_kv.first){parent = cur;cur = parent->_left;}else  //相等即插入失败{return false;}}if (parent == nullptr) //插入{_root = newnode;}else if (newnode->_kv.first < parent->_kv.first){parent->_left = newnode;newnode->_parent = parent;cur = newnode;}else{parent->_right = newnode;newnode->_parent = parent;cur = newnode;}//调整颜色以及旋转使满足红黑树while (parent != nullptr && parent->_col != BLUCK) //当parent不为黑时,由于根结点必须为黑,所以parent不是根结点,所以gparent一定存在{Node* gparent = parent->_parent;if (parent == gparent->_left)   {if (gparent->_right == nullptr || gparent->_right->_col == BLUCK)  //1.叔叔结点不存在或为黑,需要旋转并调色{if (cur == parent->_left)//左左->单次右旋{RotateR(gparent);parent->_col = BLUCK;gparent->_col = RED;}else//左右->左旋+右旋{RotateL(parent);RotateR(gparent);cur->_col = BLUCK;gparent->_col = RED;}break; //通过旋转调整后,该子树的根结点一定是黑,所以可以直接结束循环}else if (gparent->_right->_col == RED)  //2.叔叔结点为红,通过调色即可实现红黑树{//调色parent->_col = BLUCK;gparent->_right->_col = BLUCK;gparent->_col = RED;//继续向上cur = gparent;parent = cur->_parent;}else{assert(0);}}else{if (gparent->_left == nullptr || gparent->_left->_col == BLUCK)  //1.叔叔结点不存在或为黑,需要旋转并调色{if (cur == parent->_right)//右右->单次左旋{RotateL(gparent);parent->_col = BLUCK;gparent->_col = RED;}else//右左->右旋+左旋{RotateR(parent);RotateL(gparent);cur->_col = BLUCK;gparent->_col = RED;}break; //通过旋转调整后,该子树的根结点一定是黑,所以可以直接结束循环}else if (gparent->_left->_col == RED)  //2.叔叔结点为红,通过调色即可实现红黑树{//调色parent->_col = BLUCK;gparent->_left->_col = BLUCK;gparent->_col = RED;//继续向上cur = gparent;parent = cur->_parent;}else{assert(0);}}}_root->_col = BLUCK;return true;}void RotateL(Node* parent){Node* gparent = parent->_parent;Node* cur = parent->_right;Node* curleft = cur->_left;parent->_right = curleft;if (curleft != nullptr){curleft->_parent = parent;}cur->_left = parent;parent->_parent = cur;cur->_parent = gparent;if (gparent == nullptr){_root = cur;}else{if (cur->_kv.first < gparent->_kv.first){gparent->_left = cur;}else{gparent->_right = cur;}}}void RotateR(Node* parent){Node* gparent = parent->_parent;Node* cur = parent->_left;Node* curright = cur->_right;parent->_left = curright;if (curright != nullptr){curright->_parent = parent;}cur->_right = parent;parent->_parent = cur;cur->_parent = gparent;if (gparent == nullptr){_root = cur;}else{if (cur->_kv.first < gparent->_kv.first){gparent->_left = cur;}else{gparent->_right = cur;}}}bool isRBTree(){return isRBTree(_root);}protected:Node* _root = nullptr;bool checkColor(Node* root, int countBluck, int& baseBluck) //计算黑结点数量与红结点连续是否满足条件{if (root == nullptr){if (baseBluck == countBluck){return true;}return false;}if (root->_col == RED && root->_parent != nullptr && root->_parent->_col == RED){return false;}if (root->_col == BLUCK){countBluck++;}return checkColor(root->_left, countBluck, baseBluck) && checkColor(root->_right, countBluck, baseBluck);}bool isRBTree(Node* root){if (root == nullptr)return true;if (root->_col == RED)return false;//先计算一条路径中的黑色结点数量int baseBluck = 0;Node* cur = root;while (cur != nullptr){if (cur->_col == BLUCK){++baseBluck;}cur = cur->_right;}return checkColor(root, 0, baseBluck);}};
}

测试红黑树是否合格

在写完红黑树的insert之后,我们可以再编写一个测试模块来测试一棵树是否满足红黑树的特性

我们其实只需要判断两点即可:

  • 任一路径上的黑色结点个数是否相等
  • 是否存在连续的两个红色结点

我们使用递归的方式来判断:

判断任一路径上的黑色结点个数是否相等时,必须要先计算某条路径上的黑色结点个数,可以通过 while循环计算最右侧一条路径上的黑色结点个数。将cur_root开始一直向右子节点遍历即可,当cur为空时即该路径结束,终止循环。

然后使用checkColor函数(参数为当前结点指针Node*,计算中的当前路径黑色结点个数int以及先前计算的最外层黑色结点个数int )递归计算每条路径上的黑色结点个数,顺便判断是否存在连续的红色结点。
在递归过程中:
_root为空时,即当前路径已经结束,判断countBluckbaseBlack的值是否相等,若相等返回true,否则返回false
若当前结点为红色,则判断其父结点是否为红色,是就返回false
若当前结点为黑色,countBluck加1,并继续向左右子结点递归,返回左子结点的结果&&右子结点的结果:
在这里插入图片描述

测试红黑树是否合格的代码这里就不赘述了,大家可以在参考源码部分查找。这里来展示一下测试结果

namespace qqq
{void testfunc(){const int N = 10000000;vector<int> v;v.reserve(N);srand(time(0));for (size_t i = 0; i < N; i++){v.push_back(i);}RBTree<int, int> rbt;for (auto e : v){rbt.insert(make_pair(e, e));cout << "insert:" << e << "->" << rbt.isRBTree() << endl;}cout << rbt.isRBTree() << endl;}
}int main()
{qqq::testfunc();return 0;
}

在这里插入图片描述
因为这段测试代码中,存在大量I/O,所以运行速度很慢,大家可以将cout注释掉,只打印最后的结果。

总结

到此,关于红黑树的知识就介绍完了
在接下来的文章中,将会对红黑树进行封装,即mapset,尽情期待哦

如果大家认为我对某一部分没有介绍清楚或者某一部分出了问题,欢迎大家在评论区提出

如果本文对你有帮助,希望一键三连哦

希望与大家共同进步哦

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

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

相关文章

vscode安装通义灵码

作为vscode的插件&#xff0c;直接使用 通义灵码-灵动指间&#xff0c;快码加编&#xff0c;你的智能编码助手 通义灵码&#xff0c;是一款基于通义大模型的智能编码辅助工具&#xff0c;提供行级/函数级实时续写、自然语言生成代码、单元测试生成、代码注释生成、代码解释、研…

蓝桥杯单片机速成8-NE555频率测量

一、原理图 NOTE&#xff1a;使用NE555测量频率之前&#xff0c;需要将J3-15(SIGNAL)与J3-16(P34短接) 在使用矩阵键盘的时候也记得把跳冒拔下&#xff0c;因为有公共引脚P34 又是因为他的输出引脚是P34&#xff0c;所以只能用定时器0来作为计数器进行频率测量了 二、代码实现 …

基于java的电影院售票网站

开发语言&#xff1a;Java 框架&#xff1a;ssm 技术&#xff1a;JSP JDK版本&#xff1a;JDK1.8 服务器&#xff1a;tomcat7 数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09; 数据库工具&#xff1a;Navicat11 开发软件&#xff1a;eclipse/myeclip…

Rust所有权和Move关键字使用和含义讲解,以及Arc和Mutex使用

Rust 所有权规则 一个值只能被一个变量所拥有&#xff0c;这个变量被称为所有者。 一个值同一时刻只能有一个所有者&#xff0c;也就是说不能有两个变量拥有相同的值。所以对应变量赋值、参数传递、函数返回等行为&#xff0c;旧的所有者会把值的所有权转移给新的所有者&#…

云原生技术精选:探索腾讯云容器与函数计算的最佳实践

文章目录 写在前面《2023腾讯云容器和函数计算技术实践精选集》深度解读案例集特色&#xff1a;腾讯云的创新实践与技术突破精选案例分析——Stable Diffusion云原生部署的最佳实践精选集实用建议分享总结 写在前面 在数字化转型的浪潮下&#xff0c;云计算技术已成为企业运营…

spring security

权限代码业务逻辑代码不分离会造成形式结构的阅读冗余混乱。spring security是spring的安全框架。实现不影响原有业务逻辑代码的前提下&#xff0c;使用filter(拦截web请求)保护资源层级。实现使用spring aop权限控制方法层级。 spring security是基于spring、springMvc的申明…

C语言——指针

地址是由物理的电线上产生的&#xff0c;能够标识唯一一个内存单元。在C语言中&#xff0c;地址也叫做指针。 在32位机器中&#xff0c;有32根地址线。地址是由32个0/1组成的二进制序列&#xff0c;也就是用4个字节来存储地址。 在64位机器中&#xff0c;有64根地址线。地址是…

代码随想录训练营总结篇

哈哈哈 今天不玩段子 今天也没有LeetCode 今天总结一下训练营。 第一次参加这种形式的训练营&#xff0c;其实刚开始的时候很多顾虑&#xff0c;觉得时间不够&#xff0c;怕自己太菜&#xff0c;菜的丢人&#xff08;虽然确实是哈哈哈哈哈哈&#xff09;。 但是不得不说&am…

【解决】Unity Profile | FindMainCamera

开发平台&#xff1a;Unity 2020.3.7f1c1 关键词&#xff1a;FindMainCamera   问题背景 ModelViewer 是开发者基于 UnityEngine 编写的相机控制组件。ModelView.Update 中调度52次并触发3次GC.Collect。显然并不期望并尽可能避免 Update 造成的GC 问题。事实上 FindMainCame…

【Linux学习】Linux 的虚拟化和容器化技术

˃͈꒵˂͈꒱ write in front ꒰˃͈꒵˂͈꒱ ʕ̯•͡˔•̯᷅ʔ大家好&#xff0c;我是xiaoxie.希望你看完之后,有不足之处请多多谅解&#xff0c;让我们一起共同进步૮₍❀ᴗ͈ . ᴗ͈ აxiaoxieʕ̯•͡˔•̯᷅ʔ—CSDN博客 本文由xiaoxieʕ̯•͡˔•̯᷅ʔ 原创 CSDN 如…

前端-css-01

1.CSS 长度单位和颜色设置 1.1CSS 中的长度单位 px 像素 em 字体大小的倍数&#xff08;字体默认是16px&#xff09; % 百分比 1.2CSS 中的颜色设置方式 1.2.1使用颜色名表示颜色 red、orange、yellow、green、cyan、blue、purple、pink、deeppink、skyblue、greenyellow .…

SpringMvc工作流程

用户通过浏览器发送请求到前端控制器DispatcherServlet。前端控制器直接将请求转给处理器映射器HandlerMapping。处理器映射器HandlerMapping会根据请求&#xff0c;找到负责处理该请求的处理器&#xff0c;并将其封装为处理器执行链HandlerExecutionChina后返回给前端控制器Di…

DFS:深搜+回溯+剪枝解决组合问题

创作不易&#xff0c;感谢支持!!! 一、电话号码的组合 . - 力扣&#xff08;LeetCode&#xff09; class Solution { public:string hash[10]{"","","abc","def","ghi","jkl","mno","pqrs"…

MT3017 上色

思路&#xff1a;使用分治&#xff0c;在每个连续区域递归调用heng()和shu() #include <bits/stdc.h> using namespace std; int n, m; int h[5005];int shu(int l, int r) {return r - l 1; } int heng(int l, int r) {int hmin 0x3f3f3f3f;for (int i l; i < r;…

unity工程输出的log在哪里?

在编辑器里进行活动输出的log位置&#xff1a; C:\Users\username\AppData\Local\Unity\Editor\Editor.log ------------------------------------ 已经打包完成&#xff0c;形成的exe运行后的log位置&#xff1a; C:\Users\xxx用户\AppData\LocalLow\xx公司\xx项目

python相对路径导包与绝对路径导包的正确方式

【python相对路径导包与绝对路径导包的正确方式】 python相对路径导包与绝对路径导包的正确方式_哔哩哔哩_bilibilipython导包的难题&#xff0c;今天解决了&#xff0c;相对路径导包和绝对路径导包&#xff0c;均可以&#xff01;&#xff01;&#xff01;, 视频播放量 5、弹…

使用Coze工作流(二)

文章目录 使用Coze工作流 使用Coze工作流 通过本文你可以了解如何创建、发布、复制工作流&#xff0c;以及如何在 Bot 中添加工作流。 使用工作流的顺序如下&#xff1a; 创建工作流。 配置工作流。添加工作流节点并按照要处理的用户任务顺序连接工作流。 测试并发布工作流…

Java 面试宝典:请说下你对 Netty 中Reactor 模式的理解

大家好&#xff0c;我是大明哥&#xff0c;一个专注「死磕 Java」系列创作的硬核程序员。 本文已收录到我的技术网站&#xff1a;https://skjava.com。有全网最优质的系列文章、Java 全栈技术文档以及大厂完整面经 回答 Reactor 模式是一种高效处理并发网络事件的设计模式&…

大商创多用户商城系统 多处SQL注入漏洞复现

0x01 产品简介 大商创多用户商城系统是一个功能强大、灵活多变的新零售电商系统服务商。该系统支持平台自营和商家入驻,实现多元化经营模式,能够全面整合供应商、生产商、经销商和消费者等产业链资源,提高产品多样性,加快资金流动速度,并有助于减少不必要的成本输出。 0…

Python学习笔记-Flask接收post请求数据并存储数据库

1.引包 from flask import Flask, request, jsonify from flask_sqlalchemy import SQLAlchemy 2.配置连接,替换为自己的MySQL 数据库的实际用户名、密码和数据库名 app Flask(__name__) #创建应用实列 app.config[SQLALCHEMY_DATABASE_URI] mysqlpymysql://ro…