300字范文,内容丰富有趣,生活中的好帮手!
300字范文 > MySQL基础——数据库和SQL概述\MySQL基本使用\DQL语言学习\条件查询\排序查询\常

MySQL基础——数据库和SQL概述\MySQL基本使用\DQL语言学习\条件查询\排序查询\常

时间:2024-01-16 17:42:14

相关推荐

MySQL基础——数据库和SQL概述\MySQL基本使用\DQL语言学习\条件查询\排序查询\常

本文详细讲解了MySQL中DQL语言,也就是数据查询语句的使用。全文3w余字,是对学习MySQL知识的整理总结,因为篇幅较长,MySQL基础知识余下部分发表在余下博客中

DML语言学习\插入数据\删除数据\更新数据DDL语言学习\数据库的建立\MySQL数据类型\MySQL常见约束\表的增删改如有侵权,联系删除;如有问题,欢迎指出

1. 数据库和SQL概述

1.1 DB

数据库(database):存储数据的“仓库”

指长期储存在计算机内的、有组织的、可共享的数据集合

1.2 DBMS

数据库管理系统(Database Management System)

是一种操纵和管理数据库的大型软件,用于建立、使用和维护数据库

数据库是通过DBMS创建和操作的容器

用户通过DBMS访问数据库中的数据,数据库管理员也通过DBMS进行数据库的维护工作

常见的数据库管理系统:MySQL、oracle、DB2、SQL server

1.3 SQL

结构化查询语言(Structured Query Language)

是一种特殊目的的编程语言,是一种数据库查询和程序设计语言,用于存取数据以及查询、更新和管理关系数据库系统

专门用来和数据库通信的语言

SQL的优点

不是某个特定数据库供应商专有的语言,几乎所有DBMS都支持SQLSQL语言简单易学虽然简单,但实际上是一种强有力的语言,灵活使用其语言元素,可以进行非常复杂和高级的数据库操作

1.4 数据库的特点

将数据放入表中,表在放入库中

一个数据库可以有多个表,每一个表都有一个名字,用来标识自己。同时,表名具有唯一性

表具有一些特性,这些特性定义了数据在表中如何存储,类似java中 " 类 "

表由列组成,我们也称为字段。所有的表都是由一个或多个列组成,每一列类似于java中的 " 属性 "

表中的数据是按行存储的,每一行类似于java中的 " 对象 "

1.5 SQL语言分类

结构化查询语言(SQL)包含6个部分:

DML(Data Manipulation Language):数据操纵语句。用于添加、删除、修改、查询数据库记录,并检查数据完整性。

DDL(Data Definition Language):数据定义语句。用于库和表的创建、修改、删除。

DCL(Data Control Language):数据控制语句。用于定义用户的访问权限和安全级别。

DQL(Data Query Language):数据查询语句。其语句,也称为“数据检索语句”,用以从表中获得数据,确定数据怎样在应用程序给出。

TCL(Transaction Control Language): 事务控制语句。它的语句能确保被DML语句影响的表的所有行及时得以更新。

CCL:指针控制语句。用于对一个或多个表单独行的操作。

2. MySQL基本使用

2.1 MySQL服务的启动和停止

1、方式一:打开计算机管理中的服务

2、方式二:DOS命令启动(以管理员身份运行cmd)

启动 -net start MySQL

停止 -net stop MySQL

2.2 MySQL数据库的登录和退出

1、方式一:使用MySQL Command Line Client登录

输入密码后,既可以登录MySQL数据库

2、方式二:DOS窗口中操作

命令格式为mysql -h hostname -P port -u username -p

-h 后面跟服务器主机地址,这里客户端和服务器在一台机器上,所以是 “localhost” 或 127.0.0.1-P 后面跟端口,默认3306-u 后面跟的是登录数据库的用户名称,这里为 “root”-p 后面是用户登录密码

这里输入DOS命令mysql -h localhost -P 3306 -u root -p; 本机操作时,直接输入mysql -u root -p也可以

3、退出数据库

exit+ 回车即可退出

2.2 MySQL常见命令介绍

查看当前所有的数据库 :SHOW DATABASES;

打开指定的库 :USE 库名;

查看当前库的所有表:SHOW TABLES;

查看当前库的其他表:SHOW TABLES FROM 库名;

创建一个表

CREATE TABLE 表名(列名 列类型,列名 列类型,...// 最后一列末尾不用加 " , ");

查看表结构:DESC 表名;

查看服务器的版本

方式一:登录到mysql服务端select version方式二:没有登录到mysql服务端mysql --versionmysql --V

2.3 MySQL语法规范

不区分大小写, 但是建议关键字大写, 表名、列名小写

每句话用 " ; " 或 " \g " 结尾

各子句一般分行写,建议关键字占一行

注释:

单行注释:#注释文字-- 注释文字多行注释:/* 注释文字 */

3. DQL语言学习

3.1 基础查询

语法:SELECT 查询列表 FROM 表名;

SELECT查询列表FROM表名;

特点:

查询列表可以是: 表中的字段、常量值、表达式、函数SELECT * FROM 表名:查询表中所有记录查询的结果是一个虚拟的表格

1、 查询表中的单个字段

SELECT last_name FROM employees;

2、查询表中的多个字段

SELECT last_name,salary,email FROM employees;

3、查询表中所有字段

SELECT * FROM 表名;

4、查询常量值

注意: 字符型和日期型的常量值必须用单引号引起来

SELECT 常量值;SELECT 100;SELECT 'john';

5、查询表达式

SELECT 表达式;SELECT 100*98;SELECT 100%98;

6、查询函数:得到其返回值并显示

SELECT 函数名(实参列表);SELECT VERSION();

3.1.1 转义字符 - ``
这里需要注意,如果查询的列名或表名和 mysql 本身的关键字冲突,可以用转义字符``

3.1.2 起别名 - AS 或 空格
提高可读性,便于理解区分字段重名情况

# 方式一:使用ASSELECT last_name AS 姓, first_name AS 名 FROM employees;# 方式二:使用空格SELECT last_name 姓, first_name 名 FROM employees;

3.1.3 去重 - DISTINCT

# 案例:查询员工表中涉及到的所有部门编号SELECT DISTINCT department_id FROM employees;

3.1.4 +号 的作用

MySQL中+号只能作运算符,具体用法举例说明

两个操作数都为数值型,则做加法运算,如SELECT 100+90;

只要其中一个为字符型,试图即将字符型数值转换成数值型,转换成功则进行加法运算,如SELECT '123'+90;

如果转换失败,则将字符型数值转换为0 ,如'john'+90;

只要其中一方为null,结果肯定为null,如SELECT null+10;

3.1.5 补充:CONCAT函数

在MySQL中,若想进行字符拼接,不能用+号,这里可以用CONCAT()函数

SELECTCONCAT( last_name, first_name ) AS 姓名FROMemployees;

3.1.6 补充:IFNULL函数

功能:判断某字段或表达式是否为NULL。若为NULL,返回指定值。否则,返回本来的值

SELECTIFNULL(commission_pct, 0)FROMemployees;

3.2 条件查询

语法:

SELECT查询列表FROM表名WHERE筛选条件;

执行顺序: FROM => WHERE => SELECT通过WHERE子句,可以对数据进行过滤

3.2.1 按条件表达式筛选,比较运算符

MySQL中的比较运算符有

案例 1:查询工资大于12000的员工信息

SELECT*FROMemployeesWHEREsalary>12000;

案例 2:查询部门编号不等于90号的员工名和部门编号

SELECTlast_name,department_idFROMemployeesWHEREdepartment_id<>90;

3.2.1 按逻辑表达式筛选,逻辑运算符

作用:连接条件表达式

案例 1:查询工资在10000到20000之间的员工名、工资以及奖金

SELECTlast_name,salary,commission_pctFROMemployeesWHEREsalary >= 10000ANDsalary <= 20000;

3.2.1 模糊查询,LIKE、BETWEEN AND、IN、IS NULL

1、 LIKE

LIKE在进行匹配时,可以使用以下两种通配符

%:匹配任何数目的字符,甚至包括零字符

_:只能匹配一个字符

案例 1:查询员工名中包含字符a的员工信息

SELECT*FROMemployeesWHEREfirst_name LIKE '%a%';

特殊地,如果匹配内容中含有字符 ’ _ ',可以使用 ’ \ ’ 转义,或者是ESCAPE转义

ESCAPE关键字的主要作用就是指定一个字符替代“\”的作用

SELECTlast_nameFROMemployeesWHERElast_name LIKE '_$_%' ESCAPE '$';

2、 BETWEEN AND

语法格式:expr BETWEEN min AND max

包含临界值,等价于expr >= min AND expr <= max

案例 1:查询员工编号在100到120之间的员工信息

SELECT*FROMemployeesWHEREemployee_id BETWEEN 100 AND 120;

3、 IN

作用:判断某字段的值是否属于IN列表中的某一项

注意:

IN列表中的值类型必须一致或兼容(‘123’,123)IN列表中不支持通配符

案例 1:查询员工的工种编号是 IT_PROG、AD_VP、AD_PRES中的一个 的员工名和工种编号

SELECTlast_name,job_idFROMemployeesWHEREjob_id IN ('IT_PROT','AD_VP','AD_PRES');

4、 IS NULL

IS NULL,用于空值判断" = " 不能用于空值(NULL)的判断,但是 " <=> " 可以用于判断NULL值

案例 1:查询没有奖金的员工名和奖金率

SELECTlast_name,commission_pctFROMemployeesWHEREcommission_pct IS NULL;

案例 2:查询有奖金的员工名和奖金率

SELECTlast_name,commission_pctFROMemployeesWHEREcommission_pct IS NOT NULL;

3.3 排序查询

语法:

SELECT查询列表FROM表名WHERE筛选条件ORDER BY 排序列表 ASC 或 DESC;

注:

ASC是升序排序,ASC是作为默认的排序方式,所以可以省略

DESC是降序排序

执行顺序:FROM => WHERE => SELECT => ORDER BY

ORDER BY子句中可以支持单个字段、多个字段、表达式、函数、别名

ORDER BY子句一般放在查询的最后面,LIMIT子句除外

案例 1:查询员工信息,要求从高到低排序

SELECT * FROM employees ORDER BY salary DESC;

案例 2:查询部门编号>=90的员工信息,按入职时间的先后进行排序(筛选排序)

SELECT*FROMemployeesWHEREdepartment_id >= 90ORDER BY hiredate;

案例 3:按年薪的高低显示员工的信息和年薪(按别名排序)

SELECT*,salary*12*(1+IFNULL(commission_pct,0)) 年薪FROMemployeesORDER BY 年薪 DESC;

案例 4:按姓名的长度显示员工的姓名和工资(按函数排序)

SELECTLENGTH(last_name) 字节长度,last_name,salaryFROMemployeesORDER BY LENGTH(last_name) DESC;

案例 5:查询员工信息,要求先按工资升序,再按员工编号降序(按多个字段排序)

先按前面的要求排序,如果相同再按后面的要求排序。前面为主要关键字,后面为次要关键字

SELECT*FROMemployeesORDER BY salary, employee_id DESC;

3.4 常见函数——单行函数

概念:类似于java的方法,将一组逻辑语句封装在方法体中,对外暴露方法名

调用:SELECT 函数名(实参列表)

分类:

单行函数:对每一条记录输入值进行计算,得到相应的计算结果,返回给用户,也就是说,每条记录作为一个输入参数,经过函数计算得到每条记录的计算结果。如CONCAT、LENGTH、IFNULL等

分组函数:对多条记录输入值进行计算,得到多条记录对应的单个结果。做统计使用,又称为统计函数、聚合函数、组函数。

单行函数又可以分为以下几种:

字符函数数学函数日期函数流程控制函数其他函数

下面将依次介绍这几种函数的使用方法

3.4.1 字符函数
1. LENGTH: 获取参数值的字节个数

作用:获得参数值的字节个数

SELECT LENGTH('john'),# 4LENGTH("张三丰yyds");# 13

这里Navicat客户端所用编码是UTF-8,字母占一个字节,汉字占三个字节

2. CONCAT:拼接字符串

语法:CONCAT(str1,str2,...)

SELECT CONCAT(last_name,'_',first_name) 姓名 FROM employees;

3. UPPER、LOWER:大小写

案例:将姓大写,名小写,后拼接

SELECT CONCAT(UPPER(last_name),LOWER(first_name)) 姓名 FROM employees;

4. SUBSTR、SUBSTRING:截取字符
有四种函数重载形式

SELECT SUBSTR('努力百日我要上哈佛大学',8) 学校,# 哈佛大学SUBSTR('努力百日我要上哈佛大学',1,4) 方法;# 努力百日

这里要注意,SQL语言中,字符的索引从1开始

5. INSTR:返回子串第一次出现的起始索引

语法:INSTR(str,substr)

SELECT INSTR('努力百日我要上哈佛大学','哈佛大学');# 8

6. TRIM:首尾移除

TRIM ( [ [位置] [要移除的字串] FROM ] 字串)

[位置] 的可能值为 LEADING (起头),TRAILING (结尾), or BOTH (起头及结尾)这个函数将把 [要移除的字串] 从字串的起头、结尾,或是起头及结尾移除如果我们没有列出 [要移除的字串] 是什么的话,那空白就会被移除LTRIM(字串): 将所有字串起头的空白移除RTRIM(字串): 将所有字串结尾的空白移除

SELECT TRIM(' 张无忌 '),#张无忌TRIM('a' FROM 'aaaa张无忌aaaa');#张无忌

7. LPAD、RPAD: 用指定字符 左\右填充 指定长度的字符

语法:LPAD(str,len,padstr)

LPAD(要查询的字段,长度,用来填充的字段)LPAD是在左边填充,RPAD是在右边填充

SELECTLPAD('金毛狮王谢逊',10,'-'); #----金毛狮王谢逊RPAD('金毛狮王谢逊',10,'-'); #金毛狮王谢逊----

8. REPLACE:替换

语法:REPLACA(str,from_str,to_str)

SELECT REPLACE('勇敢牛牛,不怕困难','牛牛','狗狗');#勇敢狗狗,不怕困难

3.4.2 数学函数
1. ROUND:四舍五入

语法:ROUND(X)ROUND(X,D)

ROUND(X):对X四舍五入,如果是负数则取绝对值四舍五入再加负号ROUND(X,D):对X四舍五入,保留D位小数

SELECTROUND(-1.55),# -2ROUND(1.678, 2) # 1.68

2. CEIL:向上取整、FLOOR:向下取整

语法:CEIL(X)FLOOR(X)

SELECT CEIL(1.002),# 2FLOOR(1.002);# 1

3. TRUNCATE:截断

语法:TRUNCATE(X,D)

X是数值,D是保留小数的位数按照小数位数,进行数值截取(此处的截取是按保留位数直接进行截取,没有四舍五入)

SELECT TRUNCATE(1.699999,1) # 1.6

4. MOD:取余

语法:MOD(N,M)

返回N除以M余值,N - N / M * M

SELECT MOD(-10,-3),# -1MOD(10,-3);# 1

3.4.3 日期函数

MySQL 使用下列数据类型在数据库中存储日期或日期/时间值

这里注意两位的年份(YY).

若以数字表示,范围(1 - 99):1 - 69 和 70 - 99 范围的值分别转换为 2001 - 2069,1970 - 1999。超过范围的值被转为0000若以字符表示,范围(‘00’ - ‘99’):‘00’ 表示 2000,超过范围的值被转为2000

MySQL 中最重要的内建日期函数如下表

1. NOW:返回当前系统日期 + 时间

SELECT NOW();# -07-05 09:16:52

2. CURDATE:返回当前日期

SELECT CURDATE();# -07-05

3. CURTIME:返回当前时间

SELECT CURTIME();# 09:19:15

4. 获得时间指定部分

可以获得指定的部分,如年、月、日、小时、分钟、秒

对应YEAR、MONTH、MONTHNAME(返回英文)、HOUR、MINUTE、SECOND

对于-07-05 09:25:55这一时间SELECT YEAR(NOW());# SELECT MONTH(NOW());# 7SELECT MONTHNAME(NOW());# JulySELECT HOUR(NOW());# 9SELECT MINUTE(NOW());# 25SELECT SECOND(NOW());# 55

5. str_to_data:日期格式的字符转为指定格式日期
在web页面中,用户输入的日期格式不一定相同,所以可以用此方法将用户输入的字符串转换为指定的格式

SELECT STR_TO_DATE('5-7-','%m-%d-%Y');# -05-07

具体的日期格式符如下

6. DATE_FORMAT:将日期转成字符

语法:DATE_FORMAT(date,format)

date 参数是合法的日期format 规定日期/时间的输出格式

SELECT DATE_FORMAT(NOW(),"%y年%m月%d日"); # 07月05日

3.4.4 流程控制函数

1、 IF函数

语法:IF(condition, value_if_true, value_if_false)

condition:判断条件value_if_true:条件正确输出value_if_false:条件错误输出

SELECT IF(10<5,'大','小'); # 小

2、CASE函数

语法一(简单函数):CASE [col_name] WHEN [value1] THEN [result1] . . . ELSE [default] END

相当于java中switch case枚举这个字段所有可能的值

语法二(搜索函数):CASE WHEN [expr] THEN [result1]…ELSE [default] END

相当于java中多重if,实现区间判断搜索函数可以写判断,并且搜索函数只会返回第一个符合条件的值

案例1:查询员工的工资,要求:

部门号为30,工资为原工资的1.1倍部门号为40,工资为原工资的1.2倍部门号为50,工资为原工资的1.3倍其他部门,保持原工资

SELECTsalary 原始工资,department_id,CASE department_id WHEN 30 THEN salary * 1.1 WHEN 40 THEN salary * 1.2 WHEN 50 THEN salary * 1.3 ELSE salary END AS 新工资FROM employees;

案例2:查询员工的工资,要求:

工资>20000,显示A级别15000<工资<20000,显示B级别10000<工资<15000,显示C级别否则,显示D级别

SELECT salary 薪资,CASE WHEN salary>20000 THEN 'A'WHEN salary>15000 THEN'B'WHEN salary>10000 THEN'C'ELSE'D'END AS 级别FROMemployees;

3.4.5 其他函数

简单介绍前期学习过程中遇到的其他几种函数

3.5 常见函数——分组函数

用作统计使用,又称聚合函数、统计函数、组函数

这些函数的名称和作用如下表

简单使用

SELECTSUM(salary)和,ROUND(AVG(salary),2) 平均,MAX(salary)最高,MIN(salary)最低,COUNT(salary) 个数FROMemployees;

注意事项

1、COUNT函数,使用方法有两种

COUNT(*):计算表中总的行数,无论列中有数值或者为空值COUNT(字段名):计算指定列下总的行数,计算时将忽略空值的行

2、MAX函数和MIN函数,不仅适用于查找数值类型,也可应用于字符类型、日期型

字符型按字典序,‘Z’ > ‘A’;日期型, > 1990

3、SUM函数和AVG函数在计算时,忽略列值为NULL的行

4、和分组函数一起查询的字段要求是GROUP BY后的字段

3.6 分组查询

分组数据:GROUP BY 子句语法

可以使用GROUP BY子句将表中的数据分成若干组

语法:

SELECT column_name, aggregate_function(column_name)FROM table_nameWHERE column_name operator valueGROUP BY column_name;

3.6.1 简单的使用

案例 1:查询每个工种的最高工资

SELECT job_id, MAX(salary)FROM employeesGROUP BY job_id;

案例 2:查询每个位置上的部门个数

SELECT COUNT(*), location_idFROM departmentsGROUP BY location_id;

3.6.2 分组前的筛选 —— WHERE

案例 1:查询邮箱中包含a字符的,每个部门的平均工资

SELECT AVG(salary), department_idFROM employeesWHERE email LIKE '%a%'GROUP BY department_id

3.6.3 分组后的筛选 —— HAVING

以上是添加分组前的筛选,若想实现添加分组后的筛选,可以利用HAVING关键字

分组前筛选的数据源 —— 原始表分组后筛选的数据源 —— 分组后的结果集MySQL中,GROUP BYHAVING子句中支持别名

HAVING

语法:

SELECT column_name, aggregate_function(column_name)FROM table_nameWHERE column_name operator valueGROUP BY column_nameHAVING aggregate_function(column_name) operator value;

在 SQL 中增加 HAVING 子句原因是,WHERE 关键字无法与聚合函数一起使用HAVING 子句可以让我们筛选分组后的各组数据

案例 1:查询哪个部门的员工个数大于2

SELECT COUNT(*), department_idFROM employeesGROUP BY department_idHAVING COUNT(*)>2;

案例 2:查询每个工种有奖金的员工和最高工资

SELECT job_id, salaryFROM employeesWHERE commission_pct IS NOT NULLGROUP BY job_idHAVING MAX(salary) > 12000;

3.6.4 按多个字段分组
多个字段间逗号隔开,没有顺序要求

案例 :查询每个部门每个工种员工的平均工资

SELECT AVG(salary), department_id, job_idFROM employeesGROUP BY department_id, job_id;

3.6.5 添加排序
排序放在整个分组查询的最后

案例 :查询每个部门每个工种员工的平均工资,按工资从高到低排序

SELECT AVG(salary), department_id, job_idFROM employeesGROUP BY job_id, department_idORDER BY salary DESC;

3.7 连接查询

含义:又称多表查询,当查询的字段来自多个表时,就会用到连接查询

3.7.1 笛卡尔乘积现象

若使用如SELECT NAME, boyName FROM boys, beauty;从beauty表和boys表查询,会出现笛卡尔集的错误情况

即: 表1有m行,表2有n行,查询结果为m * n行

若想避免以上现象,需要添加有效的连接条件

SELECT NAME, boyName FROM boys, beautyWHERE beauty.boyfriend_id = boys.id;

3.7.2 连接查询的分类

1、按年代分类

sql-92标准: 仅仅支持内连接sql-99标准【推荐】:支持内连接+外连接(左外和右外)+交叉连接

2、按功能分类

内连接:等值连接、非等值连接、自连接外连接:左外连接、右外连接、全外连接交叉连接

3.7.3 sql92——内连接

下面介绍sql92标准下的内连接,依次为等值连接、非等值连接、自连接

1、等值连接

案例 1:查询女神名和对应的男神名

SELECT NAME, boyName FROM boys, beautyWHERE beauty.boyfriend_id = boys.id;

案例 2:查询员工名和对应的部门名

SELECT last_name, department_nameFROM employees, departmentsWHERE employees.department_id = departments.department_id;

起别名

对案例2,为了减少复杂量,区分重名字段,可以为表起别名

SELECT e.last_name, d.department_nameFROM employees e, departments dWHERE e.department_id = d.department_id;

不过需要注意,如果已经起了别名,则查询字段的限定不能再使用原表名,因为执行顺序为FROM=>SELECT

因此,如下代码会产生错误

添加筛选

案例 1:查询有奖金的员工名、部门名

SELECT last_name, department_name, commission_pctFROM employees e, departments dWHERE e.department_id = d.department_idAND mission_pct IS NOT NULL;

案例 2:查询城市名中第二个字符为‘o’的部门名和城市名

SELECT department_name, cityFROM departments d, locations lWHERE d.location_id = l.location_idAND city LIKE '_o%';

添加分组

案例 1:查询每个城市的部门个数

SELECT COUNT(*) 个数, cityFROM departments d, locations lWHERE d.location_id = l.location_idGROUP BY city;

案例 2:查询有奖金的每个部门的部门名和部门的领导编号以及该部门的最低工资

SELECT department_name, d.manager_id, MIN(salary)FROM departments d, employees eWHERE d.department_id = e.department_idAND commission_pct IS NOT NULLGROUP BY department_name, d.manager_id;

添加排序

案例 1: 查询每个工种的工种名和员工的个数,并按员工个数降序

SELECT job_title, COUNT(*)FROM employees e, jobs jWHERE e.job_id = j.job_idGROUP BY job_titleORDER BY COUNT(*) DESC;

三表连接,利用多个AND添加筛选条件

案例 1: 查询员工名、部门名和所在城市

SELECT last_name, department_name, cityFROM employees e, departments d, locations lWHERE e.department_id = d.department_idAND d.location_id = l.location_id;

总结

多表等值连接的结果为多表的交集部分

n个表连接,至少需要n-1个连接条件

多表的顺序没有要求

一般需要为表起别名

可以搭配前面介绍的所有子句使用, 比如排序、分组、筛选

2、非等值连接

利用BETWEEN AND

案例:查询员工的工资和工资级别

SELECT salary, grade_levelFROM employees e, job_grades gWHERE salary BETWEEN g.lowest_sal AND g.highest_sal;

3、自连接

自连接查询其实等同于连接查询,需要两张表,只不过它的左表(父表)和右表(子表)都是自己。

做自连接查询的时候,是自己和自己连接,分别给父表和子表取两个不同的别名,然后附上连接条件

案例: 查询员工名及其上级的名称

这里员工和经理的信息都在同一张表中

SELECT e.employee_id 员工编号, e.last_name 员工名, m.employee_id 经理编号, m.last_name 经理名 FROM employees e, employees mWHERE e.manager_id = m.employee_id;

sql99标准

语法

SELECT 查询列表

FROM 表1 别名

【连接类型】JOIN 表2 别名

ON 连接条件

【WHERE 筛选条件】

【GROUP BY 分组】

【HAVING 筛选条件】

【ORDER BY 排序列表】

分类及关键字

1、内连接:INNER

2、外连接:左外:LEFT 【OUTER】; 右外:RIGHT【OUTER】;全外:FULL【OUTER】

3、交叉连接:CROSS

注意:MySQL不支持全外连接

用图解释四种连接

3.7.4 sql99——内连接

1、等值连接

案例 1:查询员工名、部门名

SELECT last_name, department_nameFROM employees eINNER JOIN departments dON e.department_id = d.department_id;

案例 2:查询名字中包含e的员工名和工种名(添加筛选)

SELECT last_name, job_titleFROM employees eINNER JOIN jobs jON e.job_id = j.job_idWHERE e.last_name LIKE '%e%';

案例 3: 查询部门个数>3的城市名和部门个数(添加分组+筛选)

SELECT city, COUNT(*) 部门个数FROM departments dINNER JOIN locations lON d.location_id = l.location_idGROUP BY cityHAVING COUNT(*) > 3;

案例 4:查询人数大于3的部门名和员工个数,按降序(添加排序)

SELECT department_name 部门名, COUNT(*) 员工个数FROM departments dINNER JOIN employees eON d.department_id = e.department_idGROUP BY department_nameHAVING COUNT(*) > 3ORDER BY COUNT(*) DESC;

案例 5: 查询员工名、部门名、工种名,并按部门名降序(多表连接)

只要能找到,那么连接顺序就无所谓

SELECT last_name 员工名, department_name 部门名, job_title 工种名FROM employees eINNER JOIN departments d ON e.department_id = d.department_idINNER JOIN jobs j ON e.job_id = j.job_idORDER BY department_name DESC;

特点:

可以添加排序、分组、筛选

INNER可以省略

筛选条件放在WHERE后面,连接条件放在ON后面,提高分离性,便于阅读

INNER JOIN连接和sql——92语法中的等值连接效果一样,都是查询多表的交集

2、非等值连接

案例 1: 查询员工的工资级别

利用 BETWEEN AND

SELECT salary, grade_levelFROM employees eJOIN job_grades gON e.salary BETWEEN g.lowest_sal AND g.highest_salORDER BY grade_level

3、自连接

案例 1:查询员工名及其领导的名字

SELECT e.last_name, m.last_nameFROM employees eJOIN employees mON e.manager_id = m.employee_id;

3.7.5 sql99——外连接

应用场景:用于查询一个表中没有另一个表中有的记录

特点:

外连接的查询结果为主表的所有记录,和从表中匹配的值,如果从表中没有匹配,则显示null

外连接查询结果 = 内连接结果 + 主表中有而从表中没有的记录

左外连接:left join 左边的是主表; 右外连接:right join 右边的是主表

左外和右外交换两个表的顺序,可以实现相同的效果

1、左(右)外连接

案例 1:查询美女名(beauty表)及其对应的伴侣名(boys表)

SELECT b.name, bo.boyNameFROM beauty b# beauty是主表, boys是从表LEFT OUTER JOIN boys boON b.boyfriend_id = bo.id

案例 2:查询帅哥名(boys表)及其对应的伴侣名(beauty表)

左外

SELECT b.name, bo.boyNameFROM boys bo# boys是主表, beauty是从表LEFT OUTER JOIN beauty bON b.boyfriend_id = bo.id

等同于右外

SELECT b.name, bo.boyNameFROM beauty b# beauty是主表, boys是从表RIGHT OUTER JOIN boys boON b.boyfriend_id = bo.id

2、全外连接

左表和右表都不做限制,所有的记录都显示,两表不足的地方用null填充注意:MySQL 不支持全外连接,但如Oracle是支持全外连接的,而且用到的还比较多,因此在此简单介绍一下

3、交叉连接

使用sql99语法标准实现笛卡尔乘积和前文sql92语法中笛卡尔乘积现象一样,前者使用 逗号, 这里使用CROSS JOIN交叉联接返回左表中的所有行,左表中的每一行与右表中的所有行组合

SELECT b.`name`, bo.boyNameFROM beauty bCROSS JOIN boys bo;

3.8 子查询

3.8.1 相关概念

1、出现在其他语句内部的select语句,称为子查询或内查询。外部的查询语句,称为主查询或外查询

2、子查询可以添加到SELECT、UPDATE、DELETE语句中,而且可以进行多层嵌套

3、子查询中常用的操作符有ANY(SOME)、ALL、IN、EXISTS

4、在SELECT子句中先计算子查询,子查询结果作为另一个查询的过滤条件,查询可以基于一个表或者多个表

5、子查询也可以使用比较运算符,如"<" “<=” “>” “>=” 和 “!=” 等

按子查询出现的位置分类

SELECT后面:子查询的结果必须为单行单列(标量子查询)FROM后面:子查询的结果可以是多行多列WHERE或HAVING后面:要求查询结果必须为单列,单行子查询多行子查询EXISTS后面:子查询结果必须为单列(相关子查询)

3.8.2 WHERE或HAVING后面

特点

子查询放在小括号内子查询放在条件中,要求必须放在条件的右侧单行子查询,搭配单行操作符使用:> < <= >= = <>多行子查询,搭配多行操作符使用:in、any/some、all子查询的执行优先于主查询执行

1、单行子查询

案例 1:查询谁的工资比 Abel 高

SELECT *FROM employeesWHERE salary > (SELECT salaryFROM employeesWHERE last_name = 'Abel')

案例 2:查询员工信息,满足其 job_id 与141号员工相同,salary 比143号员工多

SELECT last_name, job_id, salaryFROM employeesWHERE job_id = (SELECT job_idFROM employeesWHERE employee_id = 141)AND salary > (SELECT salaryFROM employeesWHERE employee_id = 143)

案例 3:返回公司工资最少的员工的 last_name, job_id 和 salary

SELECT last_name, job_id, salaryFROM employeesWHERE salary = (SELECT MIN(salary)FROM employees)

案例 4:查询指定部门的部门编号和最低工资,满足其最低工资大于50号部门的最低工资

SELECT department_id, MIN(salary)FROM employeesGROUP BY department_idHAVING MIN(salary) > (SELECT MIN(salary)FROM employeesWHERE department_id = 50)

注意

这里 “>”、“<”(单行操作符)后面只能跟单行子查询,也就是说查询结果只有单行单列的查询,否则会出错

2、多行子查询

IN:判断某字段是否在指定列表内 x in(10,20,30) ANY/SOME:判断某字段的值是否满足其中任意一个 x > ANY(10,20,30) 等价于 x > MIN(10,20,30)x = ANY(10,20,30) 等价于 x IN(10,20,30) ALL:判断某字段的值是否满足其中所有条件 x >ALL(10,20,30) 等价于 x > MAX(10,20,30)

案例 1: 返回location_id是1400或1700的部门中的所有员工姓名

SELECT last_nameFROM employeesWHERE department_id IN (SELECT department_idFROM departmentsWHERE location_id IN(1400,1700))

这里子查询的结果是多行,因此限定条件时不能用=而是用IN, 即使是子查询的结果是单行,用IN也是没问题的

3.8.3 放在其他位置的子查询

1、放在SELECT后面

案例 : 查询部门编号是50的员工个数

SELECT (SELECT COUNT(*)FROM employeesWHERE department_id = 50) 个数;

2、放在FROM后面(表子查询)

案例 :查询每个部门平均工资的工资级别(和job_grades表连接查询)

SELECT dep_ag.department_id, dep_ag.ag, g.grade_levelFROM job_grades gJOIN (SELECT department_id, AVG(salary) agFROM employeesGROUP BY department_id) dep_ag ON dep_ag.ag BETWEEN g.lowest_sal AND g.highest_sal;

3、放在EXISTS后面

EXISTS:此关键字后面的参数是一个任意的子查询,系统对子查询运算判断是否返回行,如果至少返回一行,EXISTS结果为TRUE,否则为FALSE

案例 :查询有无名字为 “张三丰” 的员工

对比以下两种方式

SELECT *FROM employeesWHERE last_name = '张三丰'

SELECT EXISTS (SELECT *FROM employeesWHERE last_name = '张三丰') 是否存在;

3.9 分页查询

1、应用场景:当页面上的数据,一页显示不全,需要分页显示时

2、一般步骤:分页查询的sql命令请求数据库服务器 => 服务器响应查询到的多条数据 => 前台页面

3、语法:

SELECT 查询列表FROM 表1 别名JOIN 表2 别名ON 连接条件WHERE 筛选条件GROUP BY 分组HAVING 分组后筛选ORDER BY 排序列表LIMIT 起始条目索引,显示的条目数

LIMIT, 放在整个查询语句最后执行顺序:FROM => JOIN => ON => WHERE => GROUP BY => HAVING => SELECT => ORDER BY => LIMIT起始条目索引默认从 0 显示

4、公式:假如要显示的页数是page, 每页显示的条目数是size

LIMIT (page-1)*size, size;

5、案例

案例1: 查询员工信息表的前五条

SELECT * FROM employees LIMIT 5;

案例 2: 查询员工信息表,从第二条开始,显示五条

SELECT * FROM employees LIMIT 2, 5;

3.10 联合查询

1、概述

当查询结果来自于多张表,但多张表之间没有关联,这个时候往往使用联合查询,也称为UNION查询。利用UNION关键字,可以给出多条SELECT语句,并将它们的结果合成单个结果集。合并时,两个表对应的列数和数据类型要相同

2、语法

SELECT 查询列表 FROM 表1UNION [ALL]SELECT 查询列表 FROM 表2

各个SELECT语句之间使用UNION或UNION ALL关键字分隔UNION不使用关键字ALL,执行的时候删除重复的记录,所有返回的行都是唯一的

3、案例

SELECT 'MySQL', 666 UNIONSELECT 'MySQL', 666UNIONSELECT 'MySQL', 666UNIONSELECT 'MySQL', 666

SELECT 'MySQL', 666 UNIONALLSELECT 'MySQL', 666UNIONALLSELECT 'MySQL', 666UNION ALLSELECT 'MySQL', 666

MySQL基础——数据库和SQL概述\MySQL基本使用\DQL语言学习\条件查询\排序查询\常见函数\分组查询\连接查询\子查询\分页查询\联合查询

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。