本文参考:
- SQL存储过程
- CS-Notes
- 一千行MySQL命令
本文已汇总于个人的Wiki库,欢迎前来交流学习 CS-Wiki
CREATE DATABASE test;
USE test;
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`));
# 添加列
ALTER TABLE mytable
ADD col CHAR(20);
# 删除列
ALTER TABLE mytable
DROP COLUMN col;
# 删除表
DROP TABLE mytable;
# 普通插入
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;
UPDATE mytable
SET col = val
WHERE id = 1;
DELETE FROM mytable
WHERE id = 1;
# TRUNCATE TABLE 可以清空表,也就是删除所有行。
TRUNCATE TABLE mytable;
相同值只会出现一次。它作用于所有列,也就是说所有列的值都相同才算相同。
SELECT DISTINCT col1, col2
FROM mytable;
限制返回的行数。可以有两个参数,第一个参数为起始行,从 0 开始
;第二个参数为返回的总行数。
# 返回前 5 行:
SELECT *
FROM mytable
LIMIT 5;
SELECT *
FROM mytable
LIMIT 0, 5;
# 返回第 3 ~ 5 行:
SELECT *
FROM mytable
LIMIT 2, 3;
ASC
:升序(默认)DESC
:降序可以按多个列进行排序,并且为每个列指定不同的排序方式:
SELECT *
FROM mytable
ORDER BY col1 DESC, col2 ASC;
通配符也是用在过滤语句中,但它只能用于文本字段。
% 匹配 >=0 个任意字符
;_ 匹配 ==1 个任意字符
;[ ] 可以匹配集合内的字符
,例如 [ab] 将匹配字符 a 或者 b。用脱字符 ^ 可以对其进行否定
,也就是不匹配集合内的字符。使用 Like
来进行通配符匹配。
SELECT *
FROM mytable
WHERE col LIKE '[^AB]%'; -- 不以 A 和 B 开头的任意文本
Copy to clipboardErrorCopied
不要滥用通配符,通配符位于开头处匹配会非常慢。
在数据库服务器上完成数据的转换和格式化的工作往往比客户端上快得多,并且转换和格式化后的数据量更少的话可以减少网络通信量。
计算字段通常需要使用 AS
来取别名,否则输出的时候字段名为计算表达式。
SELECT col1 * col2 AS alias
FROM mytable;
CONCAT()
用于连接两个字段。
许多数据库会使用空格把一个值填充为列宽,因此连接的结果会出现一些不必要的空格,
使用 TRIM()
可以去除首尾空格。
SELECT CONCAT(TRIM(col1), '(', TRIM(col2), ')') AS concat_col
FROM mytable;
函数
各个 DBMS 的函数都是不相同的,因此不可移植,以下主要是 MySQL 的函数。
AVG()
返回某列的平均值(会忽略 NULL 行)COUNT()
返回某列的行数MAX()
返回某列的最大值MIN()
返回某列的最小值SUM()
返回某列值之和使用 DISTINCT 可以汇总不同的值。
SELECT AVG(DISTINCT col1) AS avg_col
FROM mytable;
分组:把具有相同的数据值的行放在同一组中。
可以对同一分组数据使用汇总函数进行处理,例如求分组数据的平均值等。
指定的分组字段除了能按该字段进行分组,也会自动按该字段进行排序。
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;
子查询需用括号包裹。
下面的语句可以检索出客户的订单数量,子查询语句会对第一个查询检索出的每个客户执行一次:
SELECT cust_name, (SELECT COUNT(*)
FROM Orders
WHERE Orders.cust_id = Customers.cust_id)
AS orders_num
FROM Customers
ORDER BY cust_name;
from子查询返回一个表,表型子查询
。select * from (select * from tb where id>0) as subfrom where id>1;
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);
select * from t1 where (id, gender) in (select id, gender from t2);
连接用于连接多个表,使用 JOIN
关键字,并且条件语句使用 ON
而不是 WHERE。
连接可以替换子查询,并且比子查询的效率一般会更快。
可以用 AS 给列名、计算字段和表名取别名,给表名取别名是为了简化 SQL 语句以及连接相同表。
内连接又称等值连接,使用 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;
自连接可以看成内连接的一种,只是连接的表是自身而已。
一张员工表,包含员工姓名和员工所属部门,要找出与 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";
外连接保留了没有关联的那些行。
检索所有顾客的订单信息,包括还没有订单信息的顾客。
SELECT Customers.cust_id, Orders.order_num
FROM Customers LEFT OUTER JOIN Orders
ON Customers.cust_id = Orders.cust_id;
使用 UNION 来组合两个查询,如果第一个查询返回 M 行,第二个查询返回 N 行,那么组合查询的结果一般为 M+N 行。
每个查询必须包含相同的列、表达式和聚集函数。
默认会去除相同行,如果需要保留相同行,使用 UNION ALL。
只能包含一个 ORDER BY 子句,并且必须位于语句的最后。
SELECT col FROM mytable WHERE col = 1
UNION
SELECT col FROM mytable WHERE col =2;
视图是虚拟的表,本身不包含数据,也就不能对其进行索引操作。
对视图的操作和对普通表的操作一样。
视图具有如下好处:
CREATE VIEW myview AS
SELECT Concat(col1, col2) AS concat_col, col3*col4 AS compute_col
FROM mytable
WHERE col5 = val;
存储过程可以看成是对一系列 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;
触发器会在某个表执行以下语句时而自动执行: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
参数:
CREATE TRIGGER mytrigger AFTER INSERT ON mytable FOR EACH ROW SELECT NEW.col into @result;
DROP TRIGGER trigger_name