深入浅出MySQL(二) 查询技巧

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,高可用系列 2核4GB
简介: 深入浅出MySQL(二) 查询技巧

今天这篇文章主要是来讲解一些mysql高级查询技巧方面的内容:时间宝贵,现在我们直接进入主题:


技巧1:union和union all联表查询


Mysql的联合查询命令UNION和UNION ALL,总结了使用语法和注意事项,以及学习例子和项目例子,需要的朋友可以参考下


一、UNION和UNION ALL的作用和语法


UNION 用于合并两个或多个 SELECT 语句的结果集,并消去表中任何重复行。UNION 内部的 SELECT 语句必须拥有相同数量的列,列也必须拥有相似的数据类型。同时,每条 SELECT 语句中的列的顺序必须相同.


SQL UNION 语法:


sql脚本代码如下:


SELECT column_name FROM table1
UNION
SELECT column_name FROM table2
复制代码


注释:默认地,UNION 操作符选取不同的值。如果允许重复的值,请使用 UNION ALL。当 ALL 随 UNION 一起使用时(即 UNION ALL),不消除重复行。


SQL UNION ALL 语法


sql脚本代码如下:


SELECT column_name FROM table1
UNION ALL
SELECT column_name FROM table2
复制代码


举个例子来说明


网络异常,图片无法展示
|


如果要用union来进行相应的查询的话:


SELECT country,'female', sum(population) from country GROUP 
BY country
UNION
SELECT country,'male',sum(population) from country GROUP 
BY country
复制代码


乍眼一看,没什么意义,这个案例只是用于进行讲解


网络异常,图片无法展示
|


嗯嗯,确实查询出来也没什么意义,但是如果我们把性别字段统一之后看会有什么效果:


SELECT country,'sex', sum(population) from country GROUP 
BY country
UNION
SELECT country,'sex',sum(population) from country GROUP 
BY country
复制代码


网络异常,图片无法展示
|


果然字段信息里面重复的内容会有消失,这就是union链表查询的特点,去重

如果我们将关键字换成了union all的话,就会变成了以下内容:


SELECT country,'sex', sum(population) from country GROUP
 BY country
UNION ALL
SELECT country,'sex',sum(population) from country GROUP 
BY country
复制代码


网络异常,图片无法展示
|


这个案例告诉了我们联合查询里面的union和union all的区别了。


那么这种用法又有什么应用场景呢?


不急,现在就来一个案例:


还是之前country这张表:


如果想要按照国家和性别进行分组,得出结果如下 :


国家 男 女

中国 340 260

美国 45 55

加拿大 51 49

英国 40 60


那么sql该怎么写?


这个时候可以借鉴上述的union关键字来进行联表查询了!!


SELECT country,'男', sum(population) from country WHERE 
sex=1 GROUP BY country 
UNION 
SELECT country,'女',sum(population) from country  WHERE 
sex=2 GROUP BY country
复制代码


网络异常,图片无法展示
|


技巧2:case语句


按照上述的那个案例来说:


还是之前country这张表:


如果想要按照国家和性别进行分组,得出结果如下 :


国家 男 女

中国 340 260

美国 45 55

加拿大 51 49

英国 40 60


那么sql该怎么写?


原先的写法是:


SELECT country,'男', sum(population) from country WHERE 
sex=1 GROUP BY country 
UNION 
SELECT country,'女',sum(population) from country  WHERE 
sex=2 GROUP BY country
复制代码


嗯嗯,这样写是没有错,但是你是否有考虑过性能优化方面的问题呢?


假射现在我往这个数据表里面插入了100万条数据之后了?sql查询两次,是否会造成性能方面的耽误?为何不试试用case语句来进行优化呢?


以下是相关解决方案:


SELECT country, 
       SUM( CASE WHEN sex = 1 THEN 
                      population ELSE 0 END) as 'male',   
       SUM( CASE WHEN sex = 2 THEN 
                      population ELSE 0 END)  as 'female'
FROM  country 
GROUP BY country; 
复制代码


这样一来,sql的查询次数就只需要一遍了


网络异常,图片无法展示
|


查询结果如上图所示,有没有感觉清晰了很多,哈哈哈。这是一种sql优化的技巧。


技巧3:case语句技巧深入


Case具有两种格式。简单Case函数和Case搜索函数。


–简单Case函数


CASE sex

WHEN ‘1’ THEN ‘男’

WHEN ‘2’ THEN ‘女’

ELSE ‘其他’ END


–Case搜索函数


CASE WHEN sex = ‘1’ THEN ‘男’

WHEN sex = ‘2’ THEN ‘女’

ELSE ‘其他’ END


简单case函数的案例:


有如下数据:(为了看得更清楚,我并没有使用国家代码,而是直接用国家名作为Primary Key)


国家(country) 人口(population)

中国 600

美国 100

加拿大 100

英国 200

法国 300

日本 250

德国 200

墨西哥 50

印度 250


根据这个国家人口数据,统计亚洲和北美洲的人口数量。应该得到下面这个结果。


洲 人口

亚洲 1100

北美洲 250

其他 700


想要解决这个问题,你会怎么做?生成一个带有洲Code的View,是一个解决方法,但是这样很难动态的改变统计的方式。


如果使用Case函数,SQL代码如下:


SELECT  SUM(population), 
        CASE country 
                WHEN '中国'     THEN '亚洲' 
                WHEN '印度'     THEN '亚洲' 
                WHEN '日本'     THEN '亚洲' 
                WHEN '美国'     THEN '北美洲' 
                WHEN '加拿大'  THEN '北美洲' 
                WHEN '墨西哥'  THEN '北美洲' 
                  WHEN '德国'  THEN '欧洲'
             WHEN '法国'  THEN '欧洲'
             WHEN '英国'  THEN '欧洲'
        ELSE '其他' END 
FROM  country 
GROUP BY CASE country 
                WHEN '中国'     THEN '亚洲' 
                WHEN '印度'     THEN '亚洲' 
                WHEN '日本'     THEN '亚洲' 
                WHEN '美国'     THEN '北美洲' 
                WHEN '加拿大'  THEN '北美洲' 
                WHEN '墨西哥'  THEN '北美洲' 
                  WHEN '德国'  THEN '欧洲'
                WHEN '法国'  THEN '欧洲'
                WHEN '英国'  THEN '欧洲' 
        ELSE '其他' END; 
复制代码


同样的,我们也可以用这个方法来判断工资的等级,由于之前的emp表里面的数据已经过了一百万,操作起来比较慢,因此先copy其中的部分数据到emp2表里面进行操作:


INSERT INTO emp2

SELECT * from emp limit 10


ok,这下新的表格创建好了。


现在需要对数据库表里面的数据进行分类管理:


SELECT  FIRST_NAME,SALARY,
 CASE 
  WHEN emp2.SALARY <=1000 THEN '穷苦'
  WHEN emp2.SALARY >1000 AND SALARY<=5000  THEN '普通'
  WHEN emp2.SALARY >5000 THEN '富裕'
END
FROM emp2
复制代码


经过查询最后得出相应的结果图如下:


网络异常,图片无法展示
|


技巧4:update函数里面使用case


还是刚才的那个emp2表格里面:


例,有如下更新条件


工资5000以上的职员,工资减少10%


工资在2000到4600之间的职员,工资增加15%


很容易考虑的是选择执行两次UPDATE语句,如下所示


–条件1


UPDATE emp2
SET salary = salary * 0.9 
WHERE salary >= 5000; 
复制代码


–条件2

UPDATE emp2
SET salary = salary * 1.15 
WHERE salary >= 2000 AND salary < 4600; 
复制代码


但是事情没有想象得那么简单,假设有个人工资5000块。首先,按照条件1,工资减少10%,变成工资4500。接下来运行第二个SQL时候,因为这个人的工资是4500在2000到4600的范围之内, 需增加15%,最后这个人的工资结果是5175,不但没有减少,反而增加了。如果要是反过来执行,那么工资4600的人相反会变成减少工资。暂且不管这个规章是多么荒诞,如果想要一个SQL 语句实现这个功能的话,我们需要用到Case函数。代码如下:


UPDATE emp2 SET emp2.SALARY=
CASE
 WHEN salary>5000 THEN emp2.SALARY*1.15
 WHEN salary>2000 AND salary<4000 THEN SALARY*0.4
ELSE salary END;
复制代码


嗯嗯,更新成功了


网络异常,图片无法展示
|

相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
目录
相关文章
|
2月前
|
缓存 关系型数据库 MySQL
MySQL索引策略与查询性能调优实战
在实际应用中,需要根据具体的业务需求和查询模式,综合运用索引策略和查询性能调优方法,不断地测试和优化,以提高MySQL数据库的查询性能。
196 66
|
13天前
|
存储 Oracle 关系型数据库
索引在手,查询无忧:MySQL索引简介
MySQL 是一款广泛使用的关系型数据库管理系统,在2024年5月的DB-Engines排名中得分1084,仅次于Oracle。本文介绍MySQL索引的工作原理和类型,包括B+Tree、Hash、Full-text索引,以及主键、唯一、普通索引等,帮助开发者优化查询性能。索引类似于图书馆的分类系统,能快速定位数据行,极大提高检索效率。
48 8
|
16天前
|
SQL 关系型数据库 MySQL
MySQL 窗口函数详解:分析性查询的强大工具
MySQL 窗口函数从 8.0 版本开始支持,提供了一种灵活的方式处理 SQL 查询中的数据。无需分组即可对行集进行分析,常用于计算排名、累计和、移动平均值等。基本语法包括 `function_name([arguments]) OVER ([PARTITION BY columns] [ORDER BY columns] [frame_clause])`,常见函数有 `ROW_NUMBER()`, `RANK()`, `DENSE_RANK()`, `SUM()`, `AVG()` 等。窗口框架定义了计算聚合值时应包含的行。适用于复杂数据操作和分析报告。
57 11
|
20天前
|
存储 关系型数据库 MySQL
mysql怎么查询longblob类型数据的大小
通过本文的介绍,希望您能深入理解如何查询MySQL中 `LONG BLOB`类型数据的大小,并结合优化技术提升查询性能,以满足实际业务需求。
79 6
|
2月前
|
SQL 前端开发 关系型数据库
SpringBoot使用mysql查询昨天、今天、过去一周、过去半年、过去一年数据
SpringBoot使用mysql查询昨天、今天、过去一周、过去半年、过去一年数据
68 9
|
2月前
|
缓存 监控 关系型数据库
如何优化MySQL查询速度?
如何优化MySQL查询速度?【10月更文挑战第31天】
128 3
|
2月前
|
SQL NoSQL 关系型数据库
2024Mysql And Redis基础与进阶操作系列(5)作者——LJS[含MySQL DQL基本查询:select;简单、排序、分组、聚合、分组、分页等详解步骤及常见报错问题所对应的解决方法]
MySQL DQL基本查询:select;简单、排序、分组、聚合、分组、分页、INSERT INTO SELECT / FROM查询结合精例等详解步骤及常见报错问题所对应的解决方法
|
2月前
|
监控 关系型数据库 MySQL
数据库优化:MySQL索引策略与查询性能调优实战
【10月更文挑战第27天】本文深入探讨了MySQL的索引策略和查询性能调优技巧。通过介绍B-Tree索引、哈希索引和全文索引等不同类型,以及如何创建和维护索引,结合实战案例分析查询执行计划,帮助读者掌握提升查询性能的方法。定期优化索引和调整查询语句是提高数据库性能的关键。
343 1
|
3月前
|
SQL Java 关系型数据库
java连接mysql查询数据(基础版,无框架)
【10月更文挑战第12天】该示例展示了如何使用Java通过JDBC连接MySQL数据库并查询数据。首先在项目中引入`mysql-connector-java`依赖,然后通过`JdbcUtil`类中的`main`方法实现数据库连接、执行SQL查询及结果处理,最后关闭相关资源。
233 6
|
2月前
|
SQL 关系型数据库 MySQL
定时任务频繁插入数据导致锁表问题 -> 查询mysql进程
定时任务频繁插入数据导致锁表问题 -> 查询mysql进程
59 1