【数据库学习总结】 —— 基础sql语句
???? 本文参考:
文章目录
1. 数据库创建与使用 CREATE / USE
CREATE DATABASE test;
USE test;
2. 创建表 CREATE
CREATE TABLE mytable (
# int 类型,不为空,自增
id INT NOT NULL AUTO_INCREMENT,
# int 类型,不可为空,默认值为 1,不为空
col1 INT NOT NULL DEFAULT 1,
# 变长字符串类型,最长为 45 个字符,可以为空
col2 VARCHAR(45) NULL,
# 日期类型,可为空
col3 DATE NULL,
# 设置主键为 id
PRIMARY KEY (`id`));
3. 修改表 ALTER
# 添加列
ALTER TABLE mytable
ADD col CHAR(20);
# 删除列
ALTER TABLE mytable
DROP COLUMN col;
# 删除表
DROP TABLE mytable;
4. 插入 INSERT INTO
# 普通插入
INSERT INTO mytable(col1, col2)
VALUES(val1, val2);
# 插入检索出来的数据
INSERT INTO mytable1(col1, col2)
SELECT col1, col2
FROM mytable2;
# 将一个表的内容插入到一个新表
CREATE TABLE newtable AS
SELECT * FROM mytable;
5. 更新 UPDATE
UPDATE mytable
SET col = val
WHERE id = 1;
6. 删除 DELETE
DELETE FROM mytable
WHERE id = 1;
# TRUNCATE TABLE 可以清空表,也就是删除所有行。
TRUNCATE TABLE mytable;
7. 查询 SELECT
DISTINCT 去除重复记录
相同值只会出现一次。它作用于所有列,也就是说所有列的值都相同才算相同。
SELECT DISTINCT col1, col2
FROM mytable;
LIMIT 限制结果数量子句
限制返回的行数。可以有两个参数,第一个参数为起始行,从 0 开始
;第二个参数为返回的总行数。
# 返回前 5 行:
SELECT *
FROM mytable
LIMIT 5;
SELECT *
FROM mytable
LIMIT 0, 5;
# 返回第 3 ~ 5 行:
SELECT *
FROM mytable
LIMIT 2, 3;
8. 排序 ORDER BY
-
ASC
:升序(默认) -
DESC
:降序
可以按多个列进行排序,并且为每个列指定不同的排序方式:
SELECT *
FROM mytable
ORDER BY col1 DESC, col2 ASC;
9. 通配符 Like
通配符也是用在过滤语句中,但它只能用于文本字段。
-
% 匹配 >=0 个任意字符
; -
_ 匹配 ==1 个任意字符
; -
[ ] 可以匹配集合内的字符
,例如 [ab] 将匹配字符 a 或者 b。 -
用脱字符 ^ 可以对其进行否定
,也就是不匹配集合内的字符。
使用 Like
来进行通配符匹配。
SELECT *
FROM mytable
WHERE col LIKE '[^AB]%'; -- 不以 A 和 B 开头的任意文本
Copy to clipboardErrorCopied
不要滥用通配符,通配符位于开头处匹配会非常慢。
10. 计算字段
在数据库服务器上完成数据的转换和格式化的工作往往比客户端上快得多,并且转换和格式化后的数据量更少的话可以减少网络通信量。
计算字段通常需要使用 AS
来取别名,否则输出的时候字段名为计算表达式。
SELECT col1 * col2 AS alias
FROM mytable;
CONCAT()
用于连接两个字段。
许多数据库会使用空格把一个值填充为列宽,因此连接的结果会出现一些不必要的空格,
使用 TRIM()
可以去除首尾空格。
SELECT CONCAT(TRIM(col1), '(', TRIM(col2), ')') AS concat_col
FROM mytable;
11. 函数
函数
各个 DBMS 的函数都是不相同的,因此不可移植,以下主要是 MySQL 的函数。
-
AVG()
返回某列的平均值(会忽略 NULL 行) -
COUNT()
返回某列的行数 -
MAX()
返回某列的最大值 -
MIN()
返回某列的最小值 -
SUM()
返回某列值之和
使用 DISTINCT 可以汇总不同的值。
SELECT AVG(DISTINCT col1) AS avg_col
FROM mytable;
12. 分组 GROUP BY,HAVING
分组:把具有相同的数据值的行放在同一组中。
可以对同一分组数据使用汇总函数进行处理,例如求分组数据的平均值等。
指定的分组字段除了能按该字段进行分组,也会自动按该字段进行排序。
SELECT col, COUNT(*) AS num
FROM mytable
GROUP BY col;
GROUP BY
自动按分组字段进行排序,ORDER BY
也可以按汇总字段来进行排序。
- GROUP BY 子句出现在 WHERE 子句之后,ORDER BY 子句之前;
- 除了汇总字段外,
SELECT 语句中的每一字段都必须在 GROUP BY 子句中给出
;- NULL 的行会单独分为一组;
SELECT col, COUNT(*) AS num
FROM mytable
where col > 2
GROUP BY col
ORDER BY num;
WHERE
过滤行,HAVING
过滤分组,行过滤应当先于分组过滤。
having 与 where 功能、用法相同,执行时机不同。
where 在开始时执行检测数据,对原数据进行过滤。
having 对筛选出的结果再次进行过滤。
having 字段必须是查询出来的,where 字段必须是数据表存在的。
where 不可以使用字段的别名,having 可以。因为执行WHERE代码时,可能尚未确定列值。
where 不可以使用合计函数。一般需用合计函数才会用 having
SQL标准要求HAVING必须引用GROUP BY子句中的列或用于合计函数中的列
。
SELECT col, COUNT(*) AS num
FROM mytable
WHERE col > 2
GROUP BY col
HAVING num >= 2;
13. 子查询
子查询需用括号包裹。
下面的语句可以检索出客户的订单数量,子查询语句会对第一个查询检索出的每个客户执行一次:
SELECT cust_name, (SELECT COUNT(*)
FROM Orders
WHERE Orders.cust_id = Customers.cust_id)
AS orders_num
FROM Customers
ORDER BY cust_name;
FROM 型
-
from子查询返回一个表,表型子查询
。 - from后要求是一个表,必须给子查询结果取个别名。
- 简化每个查询内的条件。
- from型需将结果生成一个临时表格,可用以原表的锁定的释放。
select * from (select * from tb where id>0) as subfrom where id>1;
WHERE 型
-
where子查询返回一个值,标量子查询
。 - 不需要给子查询取别名。
- where子查询内的表,不能直接用以更新。
select * from tb where money = (select max(money) from tb);
-
列子查询
- 如果子查询结果返回的是一列,使用
in 或 not in
将子查询的结果作为 WHRER 语句的过滤条件 -
exists 和 not exists
条件,如果子查询返回数据,则返回1或0。常用于判断条件。
- 如果子查询结果返回的是一列,使用
SELECT * FROM mytable1
WHERE col1 IN (SELECT col2 FROM mytable2);
select column1 from t1 where exists (select * from t2);
-
行子查询
查询条件是一个行。
行构造符:(col1, col2, …) 或 ROW(col1, col2, …)
行构造符通常用于与对能返回两个或两个以上列的子查询进行比较。
select * from t1 where (id, gender) in (select id, gender from t2);
14. 连接 JOIN ON
连接用于连接多个表,使用 JOIN
关键字,并且条件语句使用 ON
而不是 WHERE。
连接可以替换子查询,并且比子查询的效率一般会更快。
可以用 AS 给列名、计算字段和表名取别名,给表名取别名是为了简化 SQL 语句以及连接相同表。
内连接 INNER JOIN
内连接又称等值连接,使用 INNER JOIN 关键字。
SELECT A.value, B.value
FROM tablea AS A INNER JOIN tableb AS B
ON A.key = B.key;
等同于
SELECT A.value, B.value
FROM tablea AS A, tableb AS B
WHERE A.key = B.key;
自连接 INNER JOIN
自连接可以看成内连接的一种,只是连接的表是自身而已。
一张员工表,包含员工姓名和员工所属部门,要找出与 Jim 处在同一部门的所有员工姓名。
- 子查询版本
SELECT name FROM employee
WHERE department = (
SELECT department
FROM employee
WHERE name = "Jim");
- 自连接版本
SELECT e1.name
FROM employee AS e1 INNER JOIN employee AS e2
ON e1.department = e2.department
AND e2.name = "Jim";
外连接 LEFT/RIGHT OUTER JOIN
外连接保留了没有关联的那些行。
- 左外连接(保留左表所有行)
- 右外连接
- 全外连接
检索所有顾客的订单信息,包括还没有订单信息的顾客。
SELECT Customers.cust_id, Orders.order_num
FROM Customers LEFT OUTER JOIN Orders
ON Customers.cust_id = Orders.cust_id;
15. 组合查询 UNION
-
使用 UNION 来组合两个查询,如果第一个查询返回 M 行,第二个查询返回 N 行,那么组合查询的结果一般为 M+N 行。
-
每个查询必须包含相同的列、表达式和聚集函数。
-
默认会去除相同行,如果需要保留相同行,使用 UNION ALL。
-
只能包含一个 ORDER BY 子句,并且必须位于语句的最后。
SELECT col FROM mytable WHERE col = 1
UNION
SELECT col FROM mytable WHERE col =2;
16. 视图 VIEW
视图是虚拟的表,本身不包含数据,也就不能对其进行索引操作。
对视图的操作和对普通表的操作一样。
视图具有如下好处:
- 简化复杂的 SQL 操作,比如复杂的连接;
- 只使用实际表的一部分数据;
- 通过只给用户访问视图的权限,保证数据的安全性;
- 更改数据格式和表示。
CREATE VIEW myview AS
SELECT Concat(col1, col2) AS concat_col, col3*col4 AS compute_col
FROM mytable
WHERE col5 = val;
17. 存储过程 PROCEDURE
存储过程可以看成是对一系列 SQL 操作的批处理。
使用存储过程的好处:
- 代码封装,保证了一定的安全性;
- 代码复用;
- 由于是预先编译,因此具有很高的性能。
不带参数的存储过程
create procedure GetUsers()
begin
select * from users;
end;
调用:
call GetUsers();
删除:
drop procedure if exists GetUsers;
带参数的存储过程
命令行中创建存储过程需要使用 delimeter
自定义分隔符,因为命令行是以 ; 为结束符,而存储过程中也包含了分号,因此会错误把这部分分号当成是结束符,造成语法错误。
包含 in、out 和 inout 三种参数。
-
IN
输入:在调用过程中,将数据输入到过程体内部的参数 -
OUT
输出:在调用过程中,将过程体处理完的结果返回到客户端 -
INOUT
输入输出:既可输入,也可输出
给变量赋值都需要用 select into
语句。
每次只能给一个变量赋值,不支持集合的操作。
示例:输入一个用户ID,返回该用户的名字
# 自定义结束符号
delimiter //
create procedure GetNameByID(
in userID int
out userName varchar(200))
begin
select name from user
where id = userID
into userName;
end //
# 修改结束符号为原来的分号
delimiter ;
调用:MySQL 中变量都必须以 @ 开始
call GetNameByID(1,@userName)
select @userName;
18. 触发器 TRIGGER
触发器会在某个表执行以下语句时而自动执行:DELETE、INSERT、UPDATE。
触发器必须指定在语句执行之前还是之后自动执行,之前执行使用 BEFORE 关键字,之后执行使用 AFTER 关键字。BEFORE
用于数据验证和净化,AFTER
用于审计跟踪,将修改记录到另外一张表中。
-
INSERT
触发器包含一个名为 NEW 的虚拟表。 -
DELETE
触发器包含一个名为 OLD 的虚拟表,并且是只读的。 -
UPDATE
触发器包含一个名为 NEW 和一个名为 OLD 的虚拟表,其中 NEW 是可以被修改的,而 OLD 是只读的。
创建触发器
CREATE TRIGGER trigger_name trigger_time trigger_event ON tbl_name FOR EACH ROW trigger_stmt
参数:
- trigger_time 是触发程序的动作时间。它可以是 before 或 after,以指明触发程序是在**它的语句之前或之后触发。
- trigger_event 指明了**触发程序的语句的类型
- INSERT:将新行插入表时**触发程序
- UPDATE:更改某一行时**触发程序
- DELETE:从表中删除某一行时**触发程序
- tbl_name:监听的表,必须是永久性的表,不能将触发程序与TEMPORARY表或视图关联起来。
- trigger_stmt:当触发程序**时执行的语句。执行多个语句,可使用 BEGIN…END 复合语句结构
CREATE TRIGGER mytrigger AFTER INSERT ON mytable FOR EACH ROW SELECT NEW.col into @result;
删除触发器
DROP TRIGGER trigger_name
上一篇: mysql 基础查询、条件查询练习1
下一篇: 【文件系统】VFS虚拟文件系统