MySQL 基础知识(六)之数据查询(二)

目录

6 数值型函数

7 字符串函数

8 流程控制函数

9 聚合函数

10 分组查询 (group by)

11 分组过滤 (having)

12 限定查询 (limit)

13 多表查询

13.1 连接条件关键词 (on、using)

13.2 连接算法

13.3 交叉连接 (cross join)

13.4 内连接 (inner join)

13.5 外连接 (left join、right join)

14 子查询

14.1 select 子查询 (只需了解)

14.2 from 子查询

14.3 where 子查询

15 查询顺序总结


6 数值型函数

数值型函数描述
abs()求绝对值
sqrt()求二次方根
mod(x,y)求 x 除以 y 的余数
pi()返回圆周率
ceil() 或 ceiling()两个函数功能相同,都是向上取整
floor()向下取整,返回值为 bigint 类型
rand()生成一个 0 ~ 1 的随机数,传入相同的参数,得到的随机数也相同
round(x,y)对 x 进行四舍五入,y 表示保留几位小数
truncate(x,y)返回 x 截断后有 y 位小数后的结果
pow(x,y) 或power(x,y)求 x 的 y 次方
# 求 -3 的绝对值、4 的二次方根、11 / 3 的余数、圆周率
select abs(-3), sqrt(4), mod(11,3), pi();# 对 3.49 向上取整、3.49 向下取整、生成一个随机数
select ceil(3.49), floor(3.49), rand();# 对 3.415 保留两位小数、返回 3.415 截断后有两位小数的结果、求 2 的三次方
select round(3.415,2), truncate(3.415,2), pow(2,3);

7 字符串函数

字符串函数描述
length(s)返回字符串 s 的长度
concat(s1,s2,...,sn)拼接字符串
insert(s,idx,len,replacestr)替换字符串,将字符串 s 从第 idx 位置开始,len 个字符长的子串替换为字符串 replacestr,字符串下标从 1 开始的
lower(s) 或 lcase(s)将字符串 s 中的字母都转换为小写
upper(s) 或 ucase(s)将字符串 s 中的字母都转换为大写
left(s,n)从字符串 s 左侧开始截取,截取 n 位字符
right(s,n)从字符串 s 右侧开始截取,截取 n 位 字符
lpad(s,len,pad)从字符串 s 左侧开始填充字符串 pad,直到字符串 s 的长度为 len
rpad(s,len,pad)从字符串 s 右侧开始填充字符串 pad,直到字符串 s 的长度为 len
trim(s)去掉字符串 s 左右两侧的空格
ltrim(s)去掉字符串 s 左侧的空格
rtrim(s)去掉字符串 s 右侧的空格
repeat(s,n)返回字符串 s 重复 n 次后的结果
space(n)返回 n 个空格
strcmp(s1,s2)

比较字符串 s1 和 s2 的 ASCII 值大小

replace(s,a,b)用字符串 b 替换 字符串 s 中的所有子串 a
substr(s,idx,len) 或 substring(s,idx,len)截取字符串,从字符串的索引 idx 开始,截取 len 位字符
reverse(s)字符串反转
NULLIF(s1,s2)比较字符串 s1 和 s2,如果两个字符串相等,则返回 NULL,否则返回 s1
field(s,s1,s2,...,sn)返回字符串 s 在字符串列表中第一次出现的位置,下标从 1 开始
find_in_set(s1,s2)

返回字符串 s1 在 字符串 s2 中第一次出现的位置。

其中,字符串 s2是一个以逗号分隔的字符串 

# 求字符串 'MySQL' 的长度,拼接 'My' 、'SQL'和'数据库',替换 'Oracle数据库' 为 'MySQL数据库'
select length('MySQL'), concat('My','SQL','数据库'), insert('Oracle数据库',1,5,'MySQL');# 将字符串 'MySQL' 转为小写、大写,从 'MySQL数据库' 左侧截取 5 位字符、右侧截取 3 位字符
select lower('MySQL'), ucase('MySQL'), left('MySQL数据库',5), right('MySQL数据库',3);# 从字符串 '情人节' 左侧开始填充字符串 '财神节',直到字符串 '情人节' 的长度为 11;
# 从字符串 '情人节' 右侧开始填充字符串 '财神节',直到字符串 '情人节' 的长度为 11;
select lpad('情人节',11,'财神节'), rpad('情人节',11,'财神节');# 去掉 '   MySQL数据库   ' 两侧、左侧、右侧的空格
select trim('   MySQL数据库   '), ltrim('   MySQL数据库   '), rtrim('   MySQL数据库   ');# 返回 '财神节' 重复 8 次的结果,返回 3 个空格,比较  '情人节' 和 '财神节' ASCII 值大小
select repeat('财神节',8), space(3), strcmp('情人节','财神节');# 用 '财神节' 替换 '情人节情人节情人节' 中的子串 '情人节'
# 从 'MySQL数据库' 截取出 'MySQL'
select replace('情人节情人节情人节','情人节','财神节'), substr('MySQL数据库',1,5);# 反转字符串 'MySQL数据库',判断 c 和 'MySQL数据库' 是否相等
select reverse('MySQL数据库'), nullif('MySQL数据库','Oracle数据库');# 返回字符串 'ab' 在字符串列表 'abc','a','b','ab' 中第一次出现的位置
# 返回字符串 'ab' 在字符串 'abc,a,b,ab' 中第一次出现的位置
select field('ab','abc','a','b','ab'), find_in_set('ab','abc,a,b,ab');

8 流程控制函数

流程控制函数描述
if(condition,value1,value2)如果 condition 条件为true,返回 value1,否则返回 value2
IFNULL(value1,value2)如果 value1 不为 NULL,返回 value1,否则返回 value2
select if(1>2,'真','假'),if(1,'真','假'),if('1','真','假'),if(1<2,'真','假');
select ifnull(null,'为NULL'),ifnull('不为NULL',NULL);

9 聚合函数

  • 聚合函数会自动忽略 NULL,若该行有一个列值不为 NULL,则这行数据有效
  • 在 where 子句中不能使用聚合函数
聚合函数说明

count()

求总行数
sum()求单列中所有行的总和
avg()求单列中所有行的平均值
max()求单列中所有行的最大值
min()求单列中所有行的最小值
#计算表 goods 总行数,weight 总和、平均 netprice,最大 saleprice,最小 saleprice
select count(*), sum(weight), avg(netprice), max(saleprice), min(saleprice) from goods;

10 分组查询 (group by)

在一条 select 语句中,如果有 group by 语句,select 后面只能是:参加分组的字段、聚合函数

# 按照商品名 name 进行分组,计算每种商品的重量 weight 
select name, sum(weight) from goods group by name;

11 分组过滤 (having)

  • 使用 having 可以对分组后的数据进行过滤
  • having 不能单独使用,必须和 group by 联合使用
  • having 不能代替 where,优先使用 where,where 实在完成不了,再选择 having
# 按照商品名 name 进行分组,计算每种商品的重量 weight,选择其中总重量 > 40 的
select name, sum(weight) from goods group by name having sum(weight) > 40;

12 限定查询 (limit)

  • limit offset_start,row_count,限定查询的起始行 (offset_start) 和总行数 (row_count),起始下标从 0 开始
  • offset_start 默认是 0,可以省略,表示从第一行开始查询

关键字顺序

select

        ...

from

        ...

where

        ...

group by

        ...

having

        ...

order by

        ...

limit

        ...

以上语句执行顺序:

  1. from
  2. where
  3. group by
  4. having
  5. select
  6. order by
  7. limit
# 查询表 goods 第 3 行到第 5 行的数据
select * from goods limit 2,3;# 查询表 goods 前 3 行的数据
select * from goods limit 3;

13 多表查询

sales 表

drop table if exists sales;
create table sales (
id int primary key auto_increment,
name varchar(20),
saleprice float(7,2)
)charset=utf8;insert into sales(id, name, saleprice) values
(1, '香蕉', 3.8),
(3, '苹果', 7.5),
(4, '橘子', 4.5),
(7, '葡萄', 4.7);
  • 交叉连接:cross join (,)
  • 内连接:(inner) join
  • 外连接:left (outer) join、right (outer) join、union

MySQL join 语法官方文档: 

MySQL 5.7 Reference Manual / ... / JOIN Clauseicon-default.png?t=N7T8https://dev.mysql.com/doc/refman/5.7/en/join.html

13.1 连接条件关键词 (on、using)

执行顺序:

通过 on | using 连接多张表,再通过 where 进行筛选

select * from A join B on A.id=B.id;
# 等价于
select * from A join B using(id);# 执行顺序
from -> on|using -> where -> group by -> having -> select -> order by -> limit 

13.2 连接算法

参考文档:

MySQL 连接查询超全详解icon-default.png?t=N7T8https://learnku.com/articles/46944

  • Simple Nested Loop Join (SNLJ)

两张表做笛卡尔积进行扫描,比较费时,MySQL 不会选用该连接算法,时间复杂度 O(n * n)

  • Block Nested Loop Join (BNLJ)

没有索引,MySQL使用该算法。对外层循环的结果集进行分块,减少内层循环的次数,时间复杂度 O(n / m * n) ,其中 m 为分片数。

  • Index Nested Loop Join (INLJ)

有索引,MySQL使用该算法。用小结果集驱动大结果集,将筛选结果小的表首先连接,再去连接结果集比较大的表,即用小的表的取连接大的表

补充 left join | right join 连接算法,瞎编的伪代码,具体可以看官方文档:

MySQL 5.7 Reference Manual / ... / Nested Join Optimizationicon-default.png?t=N7T8https://dev.mysql.com/doc/refman/5.7/en/nested-join-optimization.html

注:MySQL 8.0 版本提供了 hash join 连接算法,其他版本仍是 nested loop join 连接算法,尽量少用 join,不同时连接三张表

13.3 交叉连接 (cross join)

  • A cross join B 把表 A 和表 B 的数据进行一个 N * M 的组合,即笛卡尔积。
  • 逗号 (,) 相当于 cross join,但逗号 (,) 的优先级低于 xxx join,不建议使用逗号进行多表连接(可读性差,易出错)

MySQL 中逗号 (,) 的优先级低于 xxx joinicon-default.png?t=N7T8https://blog.csdn.net/zjs246813/article/details/135706189

# 查询 表 goods 和 sales 通过 cross join 连接后的结果
select * from goods cross join sales;# 查询 表 goods 和 sales 通过 逗号 (,) 连接后的结果
select * from goods, sales;

上述代码中,表 goods 有 7 行数据,sales 有 4 行数据,通过 cross join 连接的后的表 有 7 * 4 行数据

 

13.4 内连接 (inner join)

  • A (inner) join B on A.id = B.id,内连接产生的结果集是表 A 和 B 的交集,即符合连接条件的结果集,其中 inner 可省略,在 MySQL 中 join 默认是 inner join

  • 如果不加连接条件,在 MySQL 中,join、inner join 等同于 cross join,根据本文 11.2 连接算法,没有连接条件则结果集是笛卡尔积
# 通过 goos.name 和 sales.name 内连接两张表,可以对表名使用别名
select * from goods g join sales s on g.name = s.name;

13.5 外连接 (left join、right join)

  • A left (outer) join B:以左表 A 为主表,outer 可以省略,大致连接步骤看 本文 11.2 连接算法中的 left join | right join 连接算法

  • A right (outer) join B:以右表 B 为主表,outer 可以省略

  • union:拼接两张表的查询结果,两张表的查询结果的列数需要相同并且每列的数据类型也要相同,列名可以不同 。union 默认去除重复的行,union all 不去除重复行。
# 左连接,例子不是很好
select * from goods g left join sales s on g.name = s.name;# 右连接
select * from goods g right join sales s on g.name = s.name;# 如果不去除重复行,将 union 改为 union all
select id, name,netprice from goods
union
select id,name,saleprice from sales;

14 子查询

select 语句中嵌套 select 语句,嵌套的 select 语句被称为子查询,不建议使用子查询(需要创建临时表,用完后还要删除临时表)

子查询可以用在哪里:

select

        ... (select)

from

        ... (select)

where

        ... (select)

14.1 select 子查询 (只需了解)

# 查询表 goods 和 sales 相同商品名的数据,并用 goods 中的 id、name 和 sales 中的 name 显示
select g.id, g.name goodname, (select g.name from sales s where g.name = s.name)  as  salename
from goods g;

14.2 from 子查询

把内层的查询结果当成临时表,供外层 SQL 再次查询

# 查询 表 goods 和 sales 相同商品 id 的数据
# 并用 goods 中的 id、name 和 临时表中 saleprice 显示(实际 SQL 不是这样写的)
select g.id, g.name, s.saleprice
from goods g, (select id, saleprice from sales) s
where g.id = s.id;

14.3 where 子查询

where 子查询把 select 查询结果当作一个列表项,结合 in、exists、any、all 等关键字使用。

# 查询 goods表的 id 出现在 sales 表中的数据
select * from goods where id in (select id from sales);

all  表示所有,需要结合 =、>、<、>=、<=、!=、>< 来使用

条件描述
c > all(集合)筛选 c 列中大于集合中的最大值的值
c >= all(集合)筛选 c 列中大于或等于集合中的最大值的值
c < all(集合)筛选 c 列中小于集合中的最小值的值
c <= all(集合)筛选 c 列中小于或等于集合中的最小值的值
c <> all(集合)筛选 c 列中不和集合中的任何值相等的值,可用于字符串
c != all(集合)筛选 c 列中不和集合中的任何值相等的值,可用于字符串
c = all(集合)筛选 c 列中和集合所有值都相等的值,可用于字符串
select * from goods where name != all(select name from goods where name = '苹果');

any 表示任意一个, 需要结合 =、>、<、>=、<=、!=、>< 来使用

条件描述
c > any(集合)筛选 c 列中大于集合中的最小值的值
c >= any(集合)筛选 c 列中大于或等于集合中的最小值的值
c < any(集合)筛选 c 列中小于集合中的最大值的值
c <= any(集合)筛选 c 列中小于或等于集合中的最大值的值
c <> any(集合)

筛选 c 列中不等于集合中的所有值的值,可用于字符串

c = any(集合) 表示 c 列中的值等于集合中的一个或多个值就满足筛选条件,即 c = a1 | c = a2

c <> any(集合) 是 c = any(集合) 的逆否命题,即 c != a1 & c != a2

c != any(集合)筛选 c 列中不等于集合中的所有值的值,可用于字符串
c = any(集合)筛选 c 列中只要和集合中一个值相等的值,可用于字符串
select * from goods where name != any(select name from goods where name = '苹果');

exists(子查询) 用来判断子查询是否为空,不为空,则返回 true;为空,则返回 false

select id, name, netprice from goods where exists(select 1);

15 查询顺序总结

关键字顺序:

select

        ...

distinct

        ...

from

        ...

join

        ...

on

        ...

where

        ...

group by

        ...

having

        ...

order by

        ...

limit

        ...

以上语句执行顺序:

  1. from
  2. join
  3. on
  4. where
  5. group by
  6. having
  7. select
  8. distinct
  9. order by
  10. limit

注:在MySQL中,group by、order by、having 中可以使用别名,Oracle 中不可以。

An alias can be used in a query select list to give a column a different name. You can use the alias in GROUP BY, ORDER BY, or HAVING clauses 

来自官方文档:

MySQL 8.0 Reference Manual / ... / Problems with Column Aliasesicon-default.png?t=N7T8https://dev.mysql.com/doc/refman/8.0/en/problems-with-alias.html

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

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

相关文章

装饰工程|装饰工程管理系统-项目立项子系统的设计与实现|基于Springboot的装饰工程管理系统设计与实现(源码+数据库+文档)

装饰工程管理系统-项目立项子系统目录 目录 基于Springboot的装饰工程管理系统设计与实现 一、前言 二、系统功能设计 三、系统实现 1、管理员功能实现 &#xff08;2&#xff09;合同报价管理 &#xff08;3&#xff09;装饰材料总计划管理 &#xff08;4&#xff0…

边缘计算:重塑数字世界的未来

引言 随着物联网&#xff08;IoT&#xff09;设备的激增和5G网络的普及&#xff0c;我们正站在一个计算模式的新纪元门槛上——边缘计算。这一技术范式将数据处理和分析推向网络的边缘&#xff0c;即设备或终端&#xff0c;为实时性要求较高的应用提供了前所未有的可能性。 目…

Unity如何修改预制体(预制件)?

文章目录 19 复制复制复制&#xff0c;预制体与变体 19 复制复制复制&#xff0c;预制体与变体 【预制件】 预制件作用&#xff1a;方便复用 【预制件】的制作 直接拖拽&#xff0c;从层级面板 -> 项目面板。层级面板中当前图标会变蓝&#xff0c;子物体名字变蓝色。预制件…

【JavaEE】_文件与IO

目录 1.文件概述 1.1 文件的概念 1.2 文件的存储 1.3 文件的分类 1.4 目录结构 1.5 文件操作 1.5.1 文件系统操作 1.5.2 文件内容操作 2. Java文件系统操作 2.1 File类所处的包 2.2 构造方法 2.3 方法 2.3.1 与文件路径、文件名有关的方法 2.3.2 文件是否存在与普…

【Visual Studio】使用空格替换制表符

环境 VS版本&#xff1a;VS2013 问题 如何生成空格替换制表符&#xff1f; 步骤 1、菜单 工具->选项&#xff0c;文本编辑器->C/C->制表符&#xff0c;选择【插入空格】。

课时30:内容格式化_输出格式化_字体颜色

1.1.2 字体颜色 学习目标 这一节&#xff0c;我们从 基础知识、简单实践、小结 三个方面来学习。 基础知识 场景需求 echo本质上是将信息内容输出到当前的屏幕终端&#xff0c;如果只是一种颜色的话&#xff0c;可能导致视觉疲劳。所以&#xff0c;一般情况下&#xff0c;…

使用纯 CSS 实现元素高度的过渡(不采用 max-height)

1. 前言 不知大家是否接触过元素高度的过渡&#xff0c;之前呢我是使用 CSS 加 JS 的方式来解决的&#xff0c;就是通过 JS 计算一下要过渡的元素的高度&#xff0c;然后自己给它加上 height 为某一数值。不知大家是如何解决的&#xff1f; 相信大家在做高度过渡时&#xff0…

基于FPGA的ECG信号滤波与心率计算verilog实现,包含testbench

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 4.1 ECG信号的特点与噪声 4.2 FPGA在ECG信号处理中的应用 4.3 ECG信号滤波原理 4.4 心率计算原理 4.5 FPGA在ECG信号处理中的优势 5.算法完整程序工程 1.算法运行效果图预览 其RTL结构如…

每日五道java面试题之java基础篇(九)

目录&#xff1a; 第一题 你们项⽬如何排查JVM问题第二题 ⼀个对象从加载到JVM&#xff0c;再到被GC清除&#xff0c;都经历了什么过程&#xff1f;第三题 怎么确定⼀个对象到底是不是垃圾&#xff1f;第四题 JVM有哪些垃圾回收算法&#xff1f;第五题 什么是STW&#xff1f; 第…

行测考试怎么搜题找答案?用这5款神器就够了!!! #职场发展#媒体

以下软件拥有强大的搜索功能&#xff0c;能够快速找到与题目相关的资料和答案&#xff0c;让大学生们更容易理解和掌握知识点。 1.GT4T 可在14万个语言对间进行翻译。GT4T不仅能在任何窗口获得所选内容的翻译建议&#xff0c;还也可批量翻译Office、PDF、CAT、Markdown、Asci…

《合成孔径雷达成像算法与实现》FIgure6.20

% rho_r c/(2*Fr)而不是rho_r c/(2*Bw) % Hsrcf exp函数里忘记乘pi了 clc clear close all参数设置 距离向参数设置 R_eta_c 20e3; % 景中心斜距 Tr 2.5e-6; % 发射脉冲时宽 Kr 20e12; % 距离向调频率 alpha_os_r 1.2; …

免费听歌软件,音乐搜索APP:掌中的音乐宝库,为您的音乐生活增添色彩

引言 在数字音乐的浪潮中&#xff0c;我们通常会想到QQ音乐、虾米、网易云音乐等主流平台APP。然而&#xff0c;这些商业软件在为用户提供服务的同时&#xff0c;也不可避免地伴随着一些限制和不尽如人意的地方&#xff0c;如曲库有限、音质不尽如人意或广告干扰或会员才能听歌…