数据表
,这些表的组成都是结构化的(structured)。ORM思想(Object Relational Mapping):对象关系映射,即表与面向对象语言的一一对应关系。 数据库中一个表 <-----> 一个类 表中一条数据 <-----> 一个对象(或实体) 表中的每一列 <-----> 一个属性、字段(filed)
联接表
也称中间表
,它将多对多的关系分为两个一对多的关系。将这两个表的主键都插入到第三个表中。用于访问和操作数据库
CREATE
:创建DROP
:删除ALTER
:修改RENAME
:重命名TRUNCATE
:清空INSERT
:新增DELETE
:删除UPDATE
:修改SELECT
:查询用于操作数据
GRANT
:赋予权限REVOKE
:删除权限COMMIT
:提交ROLLBACK
:回滚SAVEPOINT
:恢复用于在DML操作完数据后,对数据进行提交到数据库服务器,或回滚,赋予权限等操作
因为查询语句SELECT
的使用非常频繁,所有有很多人将查询语句抽取出来作为一类:
还有人将COMMIT
、ROLLBACK
取出来称为:
复合词用_分割,不能用驼峰命名
soucer + 绝对路径
(注意:路径不需要引号)
SELECT 字段 From 表 WHERE 条件 其他子句;
查询表
;过滤字段
;取出
需要的字段
;按条件返回数据
。规则与规范区别:规则是必须遵守的,规范可以不用遵守。
;
或\g
或\G
结束。''
表示。""
表示,不建议省略as
。SELECT 字段1,字段2... FROM 表名
:查询指定字段。SELECT * FROM 表名
:查询所有字段,*
代表所有字段(所有列)。SELECT 字段 FROM DUAL
:从MySQL内置的公共表中输出结果集
DUAL
关键字用于访问sys用户下的一张公共虚拟表(非实际存在的)。FROM DUAL也是可以省的,可省为SELECT 字段
SELECT 字段 别名 FROM 表名
以空格
分隔,即给字段重命名展示在结果集中。SELECT 字段 as 别名 FROM 表名
以as
分隔,即给字段重命名展示在结果集中。SELECT 字段 "别名" FROM 表名
以""
双引号包裹别名。双引号中可以为中文。别名只能在WHERE语句执行之前使用。
详见: SQL语句执行顺序
SELECT DISTINCT 字段1 FROM 表名
IFNULL(字段,0)
意为若字段存在则使用字段,若不存在则用0,0也可以时其他自定义数据。
也就是用自定义数据替换null。作用:
SELECT '常数',字段 FROM 表名
常数可以是字符串,可以是数字(数字不需要单引号) SELECT '尚硅谷',123,employee_id,last_name FROM employess;
SELECT 字段 FROM 表 WHERE 字段 LIKE '模糊匹配字符'
通配符:
%
:代表任何长度的字符串,字符串的长度可以为 0。
LIKE 'T%'
:查询以T开头的数据。
LIKE '%T'
:查询以T结尾的数据_
:只能代表单个字符,字符的长度不能为 0。
LIKE '_T'
:查询第2个字符为T的数据。
LIKE '__T'
:查询第3个字符为T的数据。DESCRIBE 表名
或 DESC 表名
显示表中所有字段的详细信息,字段名、字段类型、是否允许为null、默认值…SELECT 字段 FROM表名 WHERE 查询条件
SELECT 字段1,字段2 FROM 表 ORDER BY 排序字段 ASC/DESC;
ORDER BY
后边未显示的指定排序方式,默认是升序。ASC
(ascend):升序DESC
(descend):降序
SELECT 字段1,字段2 FROM 表 ORDER BY 排序字段1 ASC/DESC,排序字段2 ASC/DESC;
SELECT 字段 FROM 表名 LIMIT 位置偏移量,条目数量;
SELECT 字段 FROM 表名 LIMIT 条目数量 OFFSEF 位置偏移量;
(MySQL新特性8.0)
SELECT 字段 FROM 表名 LIMIT 0,20;
意为:从第1条数据开始,返回20条数据(第一条数据索引为0)。偏移量为0时可不写偏移量。SELECT 字段 FROM 表名 LIMIT 20,20;
意为:从第20条数据开始,返回20条数据。LIMIT (pageNum-1) * pageSize, pageSize
LIMIT pageSize OFFSET (pageNum-1) * pageSize,
意为:每页显示pageSize条数据,当前显示第pageNum页。SELECT ... FROM ... WHERE ...ORDER BY ... LIMIT
一对一
、一对多
的对应关系。SELECT 字段1,字段2 FROM 表1,表2 WHERE 表1.id=表2.id
查询多表中同一主键id下的数据。SELECT 表1.字段1,表2.字段2 FROM 表1,表2,表3 WHERE 表1.id=表2.id AND 表1.id=表3.id
从SQL优化的角度来看,建议多表查询时,所有查询的字段都指定其所在表。表的别名
在SELECT和WHERE语句中使用表的别名。=
号连接的WHERE条件语句。=号以外的
其他运算符连接的WHERE条件语句。自我引用
。
如:WHERE 表1.id1 = 表1.id2
,从同一个表中查询id1和id2相同的数据。结果集中除了有满足条件语句的数据,还查询到了
左表中
(运算符左边的表达式)不满足条件语句的数据,即为左外连接。
结果集中除了有满足条件语句的数据,还查询到了
右表中
(运算符右边的表达式)不满足条件语句的数据,即为右外连接。
结果集中除了有满足条件语句的数据,还查询到了
左、右两表中
(运算符两边的表达式)不满足条件语句的数据,即为满外连接。
注意:
1. MySQL不支持SQL92标准语法,但其他数据库管理系统支持。
2. 内连接的实现方式和之前一样。
(+)
。
示例:e.department_id = d.department_id(+);
(+)
。
示例:e.department_id(+) = d.department_id;
注意:SQL99语法是MySQL支持的。
SELECT 字段1,字段2 FROM 表1 [INNER] JOIN 表2 ON 查询条件;
不再使用WHERE来控制查询条件, 取而代之的是JOIN…ONJOIN 表1 ON 查询条件 JOIN 表2 ON 查询条件 JOIN 表3 ON 查询条件;
JOIN...ON...
即可,INNER
关键字表示内连接,可省。SELECT last_name,department_name,city FROM employees e JOIN departments d ON e.department_id=d.department_id JOIN locations l ON d.location_id=l.location_id;
SELECT 字段 FROM 表1 [OUTER] JOIN 表2 ON 查询条件
OUTER
关键字可省SELECT 字段 FROM 表1 LEFT JOIN 表2 ON 查询条件
SELECT 字段 FROM 表1 RIGHT JOIN 表2 ON 查询条件
SELECE 字段 FROM 表1 FULL OUTER JOIN 表2 ON 查询条件;
但MySQL不支持改语法SELECT 字段... FROM 表1 NATURAL JOIN 表2;
SELECT 字段 FROM 表1 JOIN 表2 USING(表1和表2中相同的字段)
自动查询两表中的USING()中的参数字段。两个
关联字段在关联表中名称一致,而且只能表示关联字段值相等。超过3个表就不要使用join连接了。
错误语句示例
:
SELECT 字段1,字段2 FROM 表1,表2;
SELECT 字段1,字段2 FROM 表1 CROSS JOIN 表2;
合并多条SELECT查询语句
,并将所有SELECT查询语句的结果集合并为一个结果集。SELECT 字段1,字段2 FROM 表1 UNION [ALL] SELECT 字段3,字段4 FROM 表2;
UNION
操作符:返回两个查询的结果集的并集
,去除重复的记录
。UNION ALL
操作符:返回两个查询的结果集的并集
,不去重
。如中图
SELECT 字段1,字段2 FROM 表1 JOIN 表2 ON 查询条件;
如左上图
SELECT 字段... FROM 表1 LEFT JOIN 表2 ON 查询条件;
如右上图
SELECT 字段... FROM 表1 RIGHT JOIN 表2 ON 查询条件;
如左下图
SELECT 字段... FROM 表1 LEFT JOIN 表2 ON 查询条件 WHERE 右表.字段 IS NULL;
如右下图
SELECT 字段... FROM 表1 LEFT JOIN 表2 ON 查询条件 WHERE 左表.字段 IS NULL;
如下左图
通过UNION ALL 关键字连合并查询条件即可。
SELECT 字段... FROM 表1 LEFT JOIN 表2 ON 查询条件 *UNION ALL* SELECT 字段... FROM 表1 LEFT JOIN 表2 ON 查询条件 WHERE 左表.字段 IS NULL;
SELECT 字段... FROM 表1 RIGHT JOIN 表2 ON 查询条件 **UNION ALL** SELECT 字段... FROM 表1 LEFT JOIN 表2 ON 查询条件 WHERE 右表.字段 IS NULL;
如下右图
通过UNION ALL 关键字连合并查询条件即可。
SELECT 字段... FROM 表1 LEFT JOIN 表2 ON 查询条件 WHERE 右表.字段 IS NULL **UNION ALL** SELECT 字段... FROM 表1 LEFT JOIN 表2 ON 查询条件 WHERE 左表.字段 IS NULL;
提高代码效率
,又提高了可维护性
,提高用户对数据库的管理效率
。MySQL的内置函数从
实现功能角度
可分为:
这些内置函数又可以分为:
什么是单行函数?
函数 | 用法 |
---|---|
ABS(x) | 返回x的绝对值 |
SIGN(X) | 返回X的符号。正数返回1,负数返回-1,0返回0 |
PI() | 返回圆周率的值 |
CEIL(x),CEILING(x) | 返回大于或等于某个值的最小整数 |
FLOOR(x) | 返回小于或等于某个值的最大整数 |
LEAST(e1,e2,e3…) | 返回列表中的最小值 |
GREATEST(e1,e2,e3…) | 返回列表中的最大值 |
MOD(x,y) | 返回X除以Y后的余数 |
RAND() | 返回0~1的随机值 |
RAND(x) | 返回0~1的随机值,其中x的值用作种子值,相同的X值会产生相同的随机数 |
ROUND(x) | 返回一个对x的值进行四舍五入后, 接近于X的整数 |
ROUND(x,y) | 返回一个对x的值进行四舍五入后 接近X的值,并保留到小数点后面Y位 |
TRUNCATE(x,y) | 返回数字x截断为y位小数的结果 |
SQRT(x) | 返回x的平方根。当X的值为负数时,返回NULL |
示例:
函数 | 用法 |
---|---|
SIN(x) | 返回x的正弦值,其中,参数x为弧度值 |
ASIN(x) | 返回x的反正弦值,即获取正弦为x的值。如果x的值不在-1到1之间,则返回NULL |
COS(x) | 返回x的余弦值,其中,参数x为弧度值 |
ACOS(x) | 返回x的反余弦值,即获取余弦为x的值。如果x的值不在-1到1之间,则返回NULL |
TAN(x) | 返回x的正切值,其中,参数x为弧度值 |
ATAN(x) | 返回x的反正切值,即返回正切值为x的值 |
ATAN2(m,n) | 返回两个参数的反正切值 |
COT(x) | 返回x的余切值,其中,X为弧度值 |
举例:
ATAN2(M,N)函数的使用示例如下:
函数 | 用法 |
---|---|
RADIANS(x) | 将角度转化为弧度,其中,参数x为角度值 |
DEGREES(x) | 将弧度转化为角度,其中,参数x为弧度值 |
示例:
SELECT RADIANS(30),RADIANS(60),RADIANS(90),DEGREES(2*PI()),DEGREES(RADIANS(90)) FROM DUAL;
函数 | 用法 |
---|---|
POW(x,y),POWER(X,Y) | 返回x的y次方 |
EXP(X) | 返回e的X次方,其中e是一个常数,2.718281828459045 |
LN(X),LOG(X) | 返回以e为底的X的对数,当X <= 0 时,返回的结果为NULL |
LOG10(X) | 返回以10为底的X的对数,当X <= 0 时,返回的结果为NULL |
LOG2(X) | 返回以2为底的X的对数,当X <= 0 时,返回NULL |
函数 | 用法 |
---|---|
BIN(x) | 返回x的二进制编码 |
HEX(x) | 返回x的十六进制编码 |
OCT(x) | 返回x的八进制编码 |
CONV(x,f1,f2) | 返回f1进制数变成f2进制数 |
函数 | 用法 |
---|---|
ASCII(S) | 返回字符串S中的第一个字符的ASCII码值 |
CHAR_LENGTH(s) | 返回字符串s的字符数。作用与CHARACTER_LENGTH(s)相同 |
LENGTH(s) | 返回字符串s的字节数,和字符集有关 |
CONCAT(s1,s2,…,sn) | 连接s1,s2,…,sn为一个字符串 |
CONCAT_WS(x,s1,s2,…,sn) | 同CONCAT(s1,s2,…)函数,但是每个字符串之间要加上x |
INSERT(str, idx, len, replacestr) | 将字符串str从第idx位置开始,len个字符长的子串替换为字符串replacestr |
REPLACE(str, a, b) | 用字符串b替换字符串str中所有出现的字符串a |
UPPER(s) 或 UCASE(s) | 将字符串s的所有字母转成大写字母 |
LOWER(s) 或LCASE(s) | 将字符串s的所有字母转成小写字母 |
LEFT(str,n) | 返回字符串str左边的n个字符 |
RIGHT(str,n) | 返回字符串str右边的n个字符 |
LPAD(str, len, pad) | 用字符串pad对str左边进行填充,直到str的长度为len个字符 |
RPAD(str ,len, pad) | 用字符串pad对str右边进行填充,直到str的长度为len个字符 |
LTRIM(s) | 去掉字符串s左侧的空格 |
RTRIM(s) | 去掉字符串s右侧的空格 |
TRIM(s) | 去掉字符串s开始与结尾的空格 |
TRIM(s1 FROM s) | 去掉字符串s开始与结尾的s1 |
TRIM(LEADING s1 FROM s) | 去掉字符串s开始处的s1 |
TRIM(TRAILING s1 FROM s) | 去掉字符串s结尾处的s1 |
REPEAT(str, n) | 返回str重复n次的结果 |
SPACE(n) | 返回n个空格 |
STRCMP(s1,s2) | 比较字符串s1,s2的ASCII码值的大小 |
SUBSTR(s,index,len) | 返回从字符串s的index位置其len个字符,作用与SUBSTRING(s,n,len)、MID(s,n,len)相同 |
LOCATE(substr,str) | 返回字符串substr在字符串str中首次出现的位置,作用于POSITION(substr IN str)、INSTR(str,substr) |
ELT(m,s1,s2,…,sn) | 返回指定位置的字符串,如果m=1,则返回s1,如果m=2,则返回s2,如果m=n,则返回sn |
FIELD(s,s1,s2,…,sn) | 返回字符串s在字符串列表中第一次出现的位置 |
FIND_IN_SET(s1,s2) | 返回字符串s1在字符串s2中出现的位置。其中,字符串s2是一个以逗号分隔的字符串 |
REVERSE(s) | 返回s反转后的字符串 |
NULLIF(value1,value2) | 比较两个字符串,如果value1与value2相等,则返回NULL,否则返回value1 |
注意:MySQL中,字符串的位置是从1开始的。
函数 | 用法 |
---|---|
CURDATE() ,CURRENT_DATE() | 返回当前日期,只包含年、月、日 |
CURTIME() , CURRENT_TIME() | 返回当前时间,只包含时、分、秒 |
NOW() / SYSDATE() / CURRENT_TIMESTAMP() / LOCALTIME() /LOCALTIMESTAMP() | 返回当前系统日期和时间 |
UTC_DATE() | 返回UTC(世界标准时间)日期 |
UTC_TIME() | 返回UTC(世界标准时间)时间 |
函数 | 用法 |
---|---|
UNIX_TIMESTAMP() | 以UNIX时间戳的形式返回当前时间。 |
UNIX_TIMESTAMP(date) | 将时间date以UNIX时间戳的形式返回。 |
FROM_UNIXTIME(timestamp) | 将UNIX时间戳的时间转换为普通格式的时间 |
函数 | 用法 |
---|---|
YEAR(date) / MONTH(date) / DAY(date) | 返回具体的日期值 |
HOUR(time) / MINUTE(time) /SECOND(time) | 返回具体的时间值 |
MONTHNAME(date) | 返回月份:January,… |
DAYNAME(date) | 返回星期几:MONDAY,TUESDAY…SUNDAY |
WEEKDAY(date) | 返回周几,注意,周1是0,周2是1,。。。周日是6 |
QUARTER(date) | 返回日期对应的季度,范围为1~4 |
WEEK(date) , WEEKOFYEAR(date) | 返回一年中的第几周 |
DAYOFYEAR(date) | 返回日期是一年中的第几天 |
DAYOFMONTH(date) | 返回日期位于所在月份的第几天 |
DAYOFWEEK(date) | 返回周几,注意:周日是1,周一是2,。。。周六是7 |
函数 | 用法 |
---|---|
EXTRACT(type FROM date) | 返回指定日期中特定的部分,type指定返回的值 |
EXTRACT(type FROM date)函数中type的取值与含义:
函数 | 用法 |
---|---|
TIME_TO_SEC(time) | 将 time 转化为秒并返回结果值。转化的公式为: 小时3600+分钟60+秒 |
SEC_TO_TIME(seconds) | 将 seconds 描述转化为包含小时、分钟和秒的时间 |
第一组:
函数 | 用法 |
---|---|
DATE_ADD(datetime, INTERVAL expr type),ADDDATE(date,INTERVAL expr type) | 返回与给定日期时间相差INTERVAL时间段的日期时间 |
DATE_SUB(date,INTERVAL expr type),SUBDATE(date,INTERVAL expr type) | 返回与date相差INTERVAL时间间隔的日期 |
第二组:
函数 | 用法 |
---|---|
ADDTIME(time1,time2) | 返回time1加上time2的时间。当time2为一个数字时,代表的是秒 ,可以为负数 |
SUBTIME(time1,time2) | 返回time1减去time2后的时间。当time2为一个数字时,代表的是 秒 ,可以为负数 |
DATEDIFF(date1,date2) | 返回date1 - date2的日期间隔天数 |
TIMEDIFF(time1, time2) | 返回time1 - time2的时间间隔 |
FROM_DAYS(N) | 返回从0000年1月1日起,N天以后的日期 |
TO_DAYS(date) | 返回日期date距离0000年1月1日的天数 |
LAST_DAY(date) | 返回date所在月份的最后一天的日期 |
MAKEDATE(year,n) | 针对给定年份与所在年份中的天数返回一个日期 |
MAKETIME(hour,minute,second) | 将给定的小时、分钟和秒组合成时间并返回 |
PERIOD_ADD(time,n) | 返回time加上n后的时间 |
函数 | 用法 |
---|---|
DATE_FORMAT(date,fmt) | 按照字符串fmt格式化日期date值 |
TIME_FORMAT(time,fmt) | 按照字符串fmt格式化时间time值 |
GET_FORMAT(date_type,format_type) | 返回日期字符串的显示格式 |
STR_TO_DATE(str, fmt) | 按照字符串fmt对str进行解析,解析为一个日期 |
上述 非GET_FORMAT 函数中fmt参数常用的格式符:
格式符 | 说明 | 格式符 | 说明 |
---|---|---|---|
%Y | 4位数字表示年份 | %y | 表示两位数字表示年份 |
%M | 月名表示月份(January,…) | %m | 两位数字表示月份(01,02,03。。。) |
%b | 缩写的月名(Jan.,Feb.,…) | %c | 数字表示月份(1,2,3,…) |
%D | 英文后缀表示月中的天数(1st,2nd,3rd,…) | %d | 两位数字表示月中的天数(01,02…) |
%e | 数字形式表示月中的天数(1,2,3,4,5…) | %H | 两位数字表示小数,24小时制(01,02…) |
%h和%I | 两位数字表示小时,12小时制(01,02…) | %k | 数字形式的小时,24小时制(1,2,3) |
%l | 数字形式表示小时,12小时制(1,2,3,4…) | %i | 两位数字表示分钟(00,01,02) |
%S和%s | 两位数字表示秒(00,01,02…) | %W | 一周中的星期名称(Sunday…) |
%a | 一周中的星期缩写(Sun.,Mon.,Tues.,…) | %w | 以数字表示周中的天数(0=Sunday,1=Monday…) |
%j | 以3位数字表示年中的天数(001,002…) | %U | 以数字表示年中的第几周,(1,2,3。。)其中Sunday为周中第一天 |
%u | 以数字表示年中的第几周,(1,2,3。。)其中Monday为周中第一天 | %% | 表示% |
%T | 24小时制 | %r | 12小时制 |
%p | AM或PM |
GET_FORMAT函数中date_type和format_type参数取值如下:
函数 | 作用 |
---|---|
IF(value,value1,value2) | 如果value的值为TRUE,返回value1,否则返回value2 |
IFNULL(value1, value2) | 如果value1不为NULL,返回value1,否则返回value2 |
CASE WHEN 条件1 THEN 结果1 WHEN 条件2 THEN 结果2… [ELSE resultn] END | 相当于Java的if…else if…else… |
CASE expr WHEN 常量值1 THEN 值1 WHEN 常量值1 THEN值1 … [ELSE 值n] END | 相当于Java的switch…case… |
示例:
函数 | 作用 |
---|---|
PASSWORD(str) | 返回字符串str的加密版本,41位长的字符串。加密结果 不可逆 ,常用于用户的密码加密 |
MD5(str) | 返回字符串str的md5加密后的值,也是一种加密方式。若参数为NULL,则会返回NULL |
SHA(str) | 从原明文密码str计算并返回加密后的密码字符串,当参数为NULL时,返回NULL。 SHA加密算法比MD5更加安全 。 |
ENCODE(value,password_seed) | 返回使用password_seed作为加密密码加密value |
DECODE(value,password_seed) | 返回使用password_seed作为加密密码解密valu |
函数 | 作用 |
---|---|
VERSION() | 返回当前MySQL的版本号 |
CONNECTION_ID() | 返回当前MySQL服务器的连接数 |
DATABASE(),SCHEMA() | 返回MySQL命令行当前所在的数据库 |
USER(),CURRENT_USER()、SYSTEM_USER(),SESSION_USER() | 返回当前连接MySQL的用户名,返回结果格式为“主机名@用户名” |
CHARSET(value) | 返回字符串value自变量的字符集 |
COLLATION(value) | 返回字符串value的比较规则 |
函数 | 作用 |
---|---|
FORMAT(value,n) | 返回对数字value进行格式化后的结果数据。n表示 四舍五入 后保留到小数点后n位 |
CONV(value,from,to) | 将value的值进行不同进制之间的转换 |
INET_ATON(ipvalue) | 将以点分隔的IP地址转化为一个数字 |
INET_NTOA(value) | 将数字形式的IP地址转化为以点分隔的IP地址 |
BENCHMARK(n,expr) | 将表达式expr重复执行n次。用于测试MySQL处理expr表达式所耗费的时间 |
CONVERT(value USINGchar_code) | 将value所使用的字符编码修改为char_code |
数值类型
平均值数值类型
总和
任意类型
最小值任意类型
最大值
不会计算值为NULL的数据
。SELECT 字段1,字段2 FROM 表 GROUP BY 分组字段1,分组字段2;
WITH ROLLUP
关键字:用于在结果集末尾追加一条记录,显示查询的记录总数,即统计记录数量SELECT 字段 FROM 表 GROUP BY 分组字段 WITH ROLLUP;
注意:ROLLUP和ORDER BY是互相排斥的,二者不能同时使用。SELECT 字段 FROM 表 GROUP BY 分组字段 HAVING 有聚合函数的过滤条件;
总结:
总结:
虚拟表
,然后将虚拟表传入下一个步骤作为输入。SELECT 字段 FROM 表 WHERE 字段 运算符 ( SELECT 字段 FROM 表 )
通常是在一个查询语句的条件语句中嵌套另一个查询语句,用括号包裹
表示一个整体。
子查询比主查询先执行
。自连接效率高
。 按子查询返回数据条目数分类:
按子查询执行次数分类:
操作符 | 含义 |
---|---|
= | equal to |
> | greater than |
>= | greater than or equal to |
< | less than |
<= | less than or equal to |
<> | not equal to |
操作符 | 含义 |
---|---|
IN | 等于列表中的任意一个 |
ANY | 需要和单行比较操作符一起使用,和子查询返回的某一个值比较 |
ALL | 需要和单行比较操作符一起使用,和子查询返回的所有值比较 |
SOME | 实际上是ANY的别名,作用相同,一般常使用ANY |
EXISTS
操作符一起来使用,用来检查在子查询中是否存在满足条件的行
。NOT EXISTS
关键字表示如果不存在
某种条件,则返回TRUE,否则返回FALSE。CREWTE DATABASE 数据库名
CREATE DATABASE 数据库名 CHARACTER SET '字符集'
CRESTE DATABASE IF NOT EXISTS 数据库名
SHOW DATABASES;
,注意有sSELECE DATABASE();
,DATABASE()是MySQL中的全局函数。SHOW TABLES FROM 数据库名;
SHOW CREATE DATABASE 数据库名;
或者
SHOW CREATE DATABASE 数据库名\G;
切换/使用数据库
SUE 数据库名;
注意:在操作表和数据之前必须先声明是对哪个数据库进行操作,否则就要对所有对象加上数据库名。ALTER DATABASE 数据库名 CHARACTER SET '字符集';
DROP DATABASE 数据库名;
DROP DATABASE IF EXISTS 数据库名;
示例:
CREATE TEMPORARY TABLE 表 子语句
SHOW CREATE TABLE 表名\G;
DESCRIBE 表名;
DESC 表明;
使用
ALTER TABLE
语句可以实现:
ALTER TABLE 表名 ADD 字段名 字段类型 [FIRST / AFTER 字段名];
ALTER TABLE 表名 MODIFY 字段名1 字段类型 [DEFAULT 默认值] [FIRST / AFTER 字段名2];
ALTER TABLE 表名 DROP COLUMN 字段名;
ALTER TABLE 表名 CHANGE 旧字段名 新字段名 新数据类型;
RENAME TABLE 旧表名 TO 新表名;
ALTER TABLE 旧表名 RENAME TO 新表名;
DROP TABLE [IF EXISTS] 数据表1 [, 数据表2, …, 数据表n];
没有与其他任何数据表形成关联关系
时,可以将当前数据表直接删除。 TRUNCATE TABLE 表t;
或
DELETE FROM 表;
(推荐使用,安全性高)阿里开发规范:
【参考】TRUNCATE TABLE 比 DELETE FROM 速度快,且使用的系统和事务日志资源少,
但 TRUNCATE 无事务且不触发 TRIGGER,有可能造成事故,故不建议在开发代码中使用此语句。
说明:TRUNCATE TABLE 在功能上与不带 WHERE 子句的 DELETE 语句相同。
SET autocommit = FALSE;
,则可以回滚。DDL操作要么成功要么回滚
。DROP TABLE book1,book2;
删除book1和book2两张表,会报错book2不存在,但是book1会被删除。DROP TABLE book1,book2;
,也会报错,但是book1不会被删除。
先成功的删除book1,但在删除book2时报错了,删除book1的操作会被回滚,因为删除book1和book2是同一个事务。INSERT INTO 表 VALUES(值1,值2...)
,若值为字符串需用''
单引号包裹。 INSERT INTO 表(字段1,字段2...) VALUES(值1,值2...)
,没有赋值的字段,值为null。INSERT INTO 表(字段1,字段2...) VALUES(值1,值2...),(值1,值2...)
INSERT INTO 表(字段1,字段2...)
SELECT 字段1,字段2... FROM 表;
注意: UPDATE 表 SET 字段1 = 值1,字段2 = 值2...;
UPDATE 表 SET 字段1 = 值1,字段2 = 值2... WHERE 过滤条件;
DELETE FROM 表;
DELETE FROM 表 WHERE 字段 = 值
GENERATED ALWAYS AS (计算表达式) VIRTUAL
声明计算列CREATE TABLE 表 (a INT,b INT ,c INT GENERATED ALWAYS AS ( a + b ) VIRTUAL);
其中,字段c即为计算列。当b或c的值发生变化时,c的值也会随之改变。运算符 | 名称 | 示例 | 注意 |
---|---|---|---|
| 加 | SELECT A + B | +号只表示数值相加。 |
- | 减 | SELECT A - B | |
* | 乘 | SELECT A * B | 若分母为0,则结果为null |
/ 或 DIV | 除 | SELECT A / B 或SELECT A DIV B | |
% 或 MOD | 取模 | SELECT A % B或 SELECT A MODB |
注意:
+
号只表示数值相加。比较结果为真,返回1;结果为假,返回0;其他情况返回NUll,null参与计算的基本都返回null。
运算符 | 名称 | 作用 | 示例 |
---|---|---|---|
= | 等于 | 数值和字符串比较,存在隐式转换 | SELECT 字段 FROM 表 WHERE A = B |
:= | 赋值运算符 | 将右边的值赋给左边的变量 | name := ‘张三’ |
<=> | 安全等于 | 安全的比较两个操作数 | SELECT 字段 FROM 表 WHERE A <= >B |
<> 或 != | 不等于 | SELECT 字段 FROM 表 WHERE A <> B | |
< | 小于 | SELECT 字段 FROM 表 WHERE A < B | |
<= | 小于等于 | SELECT 字段 FROM 表 WHERE A <= B | |
> | 大于 | SELECT 字段 FROM 表 WHERE A > B | |
>= | 大于等于 | SELECT 字段 FROM 表 WHERE A >= B |
等于运算符:
返回0
。安全等于:
比较结果为真,返回1;结果为假,返回0;其他情况返回NUll,null参与计算的基本都返回null。
运算符 | 名称 | 作用 | 示例 |
---|---|---|---|
IS NULL | 为空运算符 | 判断1个操作数是否为空 | SELECT 字段 FROM 表 WHERE A IS NULL |
ISNULL | 为空运算符 | 判断1个操作数是否为空 | SELECT 字段 FROM 表 WHERE ISNULL(A) |
IS NOTNULL | 不为空运算符 | 判断1个操作数是否不为空 | SELECT 字段 FROM 表 WHERE A IS NOT NULL |
LEAST | 最小值运算符 | 在多个值中返回最小值 | SELECT 字段 FROM 表 WHERE LEAST(A,B,C) |
GREATEST | 最大值运算符 | 在多个值中返回最大值 | SELECT 字段 FROM 表 WHERE GREATEST(A,B,C) |
BETWEEN…AND | 中间值运算符 | 判断一个值是否在两个值之间,双闭区间 | SELECT 字段 FROM 表 WHERE A BETWEEN 下限 AND 上限 |
IN | 属于运算符 | 判断一个值是否在表中 | SELECT 字段 FROM 表 WHERE A IN(B,C) |
NOT IN | 不属于运算符 | 判断一个值是否不在表中 | SELECT 字段 FROM 表 WHERE A NOT IN(B,C) |
LIKE | 模糊匹配运算符 | 判断一个值是否符合模糊匹配规则 | SELECT 字段 FROM 表 WHERE A LIKE B |
RLIKE | 正则表达式运算符 | 判断一个值是否符合正则表达式 | SELECT 字段 FROM 表 WHERE A RLIKE B |
REGEXP | 正则表达式运算符 | 判断一个值是否符合正则表达式规则 | SELECT 字段 FROM 表 WHERE A REGEXP B |
NOT
使用,即:将比较结果取反
。逻辑运算符主要用来判断表达式的真假,在MySQL中,逻辑运算符的返回值为1、0、null。
运算符 | 名称 | 作用 | 示例 |
---|---|---|---|
NOT 或 ! | 逻辑非 | SELECT NOT A | |
AND 或 && | 逻辑与 | SELECT A AND B | |
OR 或 || | 逻辑或 | SELECT A OR B | |
XOR | 逻辑异或 | SELECT A XOR B |
注意:
位运算符会先将操作数转换成二进制数,然后按位进行运算,最后将结果转为十进制返回。
运算符 | 名称 | 作用 | 示例 |
---|---|---|---|
& | 按位与(位AND) | SELECT A B | |
| | 按位或(位OR) | SELECT A | B | |
^ | 按位异或(位XOR) | SELECT A ^ B | |
~ | 按位取反 | SELECT ~ A | |
>> | 按位右移 | SELECT A >> 2 | |
<< | 按位左移 | SELECT << 2 |
类型 | 描述 |
---|---|
整数类型 | TINYINT、SMALLINT、MEDIUMINT、INT(或INTEGER)、BIGINT |
浮点类型 | FLOAT、DOUBLE |
定点数类型 | DECIMAL |
位类型 | BIT |
日期时间类型 | YEAR、TIME、DATE、DATETIME、TIMESTAMP |
文本字符串类型 | CHAR、VARCHAR、TINYTEXT、TEXT、MEDIUMTEXT、LONGTEXT |
枚举类型 | ENUM |
集合类型 | SET |
二进制 | BINARY、VARBINARY、TINYBLOB、BLOB、MEDIUMBLOB、LONGBLOB |
JSON类型 | JSON对象、JSON数组 |
空间数据类型 | 单值:GEOMETRY、POINT、LINESTRING、POLYGON;集合:MULTIPOINT、MULTILINESTRING、MULTIPOLYGON、GEOMETRYCOLLECTION |
常用类型介绍
类型 | 描述 |
---|---|
INT | 从-231到231-1的整型数据。存储大小为 4个字节 |
CHAR(size) | 定长字符数据。若未指定,默认为1个字符,最大长度255 |
VARCHAR(size) | 可变长字符数据,根据字符串实际长度保存,必须指定长度 |
FLOAT(M,D) | 单精度,占用4个字节,M=整数位+小数位,D=小数位。 D<=M<=255,0<=D<=30,默认M+D<=6 |
DOUBLE(M,D) | 双精度,占用8个字节,D<=M<=255,0<=D<=30,默认M+D<=15 |
DECIMAL(M,D) | 高精度小数,占用M+2个字节,D<=M<=65,0<=D<=30,最大取值范围与DOUBLE相同。 |
DATE | 日期型数据,格式’YYYY-MM-DD’ |
BLOB | 二进制形式的长文本数据,最大可达4G |
TEXT | 长文本数据,最大可达4G |
整数类型 | 字节 | 有符号数取值范围 | UNSIGNED无符号数取值范围 |
---|---|---|---|
TINYINT | 1 | -128~127 | 0~255 |
SMALLINT | 2 | -32768~32767 | 0~65535 |
MEDIUMINT | 3 | -8388608~8388607 | 0~16777215 |
INT、INTEGER | 4 | -2147483648~2147483647 | 0~4294967295 |
BIGINT | 8 | -9223372036854775808~9223372036854775807 | 0~18446744073709551615 |
M
: 表示显示宽度(用多少位二进制类显示这个数),M的取值范围是(0, 255)。
例如,int(5):当数据宽度小于5位的时候在数字前面需要用字符填满宽度。CREATE TABLE 表 (字段 类型[(M) ZEROFILL)];
UNSIGNED
: 无符号类型(非负),所有的整数类型都有一个可选的属性UNSIGNED(无符号属性),无
符号整数类型的最小取值为0。 负数是有符号类型,因为有负号,正数是无符号类型。
CREATE TABLE 表(字段 类型 [UNSIGNED]);
TINYINT
:一般用于枚举数据,比如系统设定取值范围很小且固定的场景。SMALLINT
:可以用于较小范围的统计数据,比如统计工厂的固定资产库存数量等。MEDIUMINT
:用于较大整数的计算,比如车站每日的客流量等。INT、INTEGER
:取值范围足够大,一般情况下不用考虑超限问题,用得最多。比如商品编号。 BIGINT
:只有当你处理特别巨大的整数时才会用到。比如双十一的交易量、大型门户网站点击量、证券公司衍生产品持仓等。存储空间
和 可靠性
的平衡问题:一方 面,用占用字节数少的整数类型可以节省存储空间;另一方面,要是为了节省存储空间, 使用的整数类型取值范围太小,一旦遇到超出取值范围的情况,就可能引起 系统错误
,影响可靠性。系统故障产生的成本远远超过增加几个字段存储空间所产生的成本
。因此,我建议你首先确保数据不会超过取值范围,在这个前提之下,再去考虑如何节省存储空间。浮点数会四舍五入,数据不精准。
REAL
默认就是 DOUBLE。如果你把 SQL 模式设定为启用“ REAL_AS_FLOAT ”,那 么,MySQL 就认为REAL 是 FLOAT。SET sql_mode = “REAL_AS_FLOAT”;
因为浮点数是不准确的,所以我们要避免使用“=”来判断两个数是否相等。
定点数类型:DECIMAL
。据类型 | 字节数 | 含义 |
---|---|---|
DECIMAL(M,D),DEC,NUMERIC | M+2字节 | 有效范围由M和D决定 |
DECIMAL(M,D)
的方式表示高精度小数
。其中,M被称为精度(大小),D被称为标度(小数位数)。0<=M<=65,0<=D<=30,D<M。例如,定义DECIMAL(5,2)的类型,表示该列取值范围是-999.99~999.99。定点数
在MySQL内部是以 字符串
的形式进行存储
,这就决定了它一定是精准的。二进制字符串类型 | 长度 | 长度范围 | 占用空间 |
---|---|---|---|
BIT(M) | M | 1 <= M <= 64 | 约为(M + 7)/8个字节 |
类型 | 名称 | 字节 | 日期格式 | 最小值 | 最大值 |
---|---|---|---|---|---|
YEAR | 年 | 1 | YYYY或YY | 1901 | 2155 |
TIME | 时间 | 3 | HH:MM:SS | -838:59:59 | 838:59:59 |
DATE | 日期 | 3 | YYYY-MM-DD | 1000-01-01 | 9999-12-03 |
DATETIME | 日期时间 | 8 | YYYY-MM-DD HH:MM:SS | 1000-01-01 00:00:00 | 9999-12-31 23:59:59 |
TIMESTAMP | 日期时间 | 4 | YYYY-MM-DD HH:MM:SS | 1970-01-01 00:00:00 UTC | 2038-01-1903:14:07UTC |
YEAR
类型通常用来表示年 DATE
类型通常用来表示年、月、日TIME
类型通常用来表示时、分、秒DATETIME
类型通常用来表示年、月、日、时、分、秒TIMESTAMP
类型通常用来表示带时区的年、月、日、时、分、秒1个字节
的存储空间。从MySQL5.5.27开始,2位格式的YEAR已经不推荐使用
。YEAR默认格式就是“YYYY”,没必要写成YEAR(4),DATE类型表示日期,没有时间部分,格式为
YYYY-MM-DD
,其中,YYYY表示年份,MM表示月份,DD表示 日期。需要3个字节
的存储空间。在向DATE类型的字段插入数据时,同样需要满足一定的格式条件。
YYYY-MM-DD
格式或者 YYYYMMDD
格式表示的字符串日期,其最小取值为1000-01-01,最大取值为
9999-12-03。YYYYMMDD格式会被转化为YYYY-MM-DD格式。YY-MM-DD
格式或者 YYMMDD
格式表示的字符串日期,此格式中,年份为两位数值或字符串满足
YEAR类型的格式条件为:当年份取值为00到69时,会被转化为2000到2069;当年份取值为70到99
时,会被转化为1970到1999。CURRENT_DATE()
或者 NOW()
函数,会插入当前系统的日期。 3个字节
的存储空间来存储TIME类型的数
据,可以使用“HH:MM:SS”格式来表示TIME类型,其中,HH表示小时,MM表示分钟,SS表示秒。'D HH:MM:SS'
、'HH:MM:SS'
、'HH:MM'
、'D HH:MM'
、'D HH'
或'SS'
格式,都能被正确地插入TIME类型的字段中。其中D表示天,其最小值为0,最大值为34。如果使用带有D格式的字符串插入TIME类型的字段时,D会被转化为小时,计算格式为D*24+HH。当使用带有冒号并且不带D的字符串表示时间时,表示当天的时间,比如12:10表示12:10:00,而不是00:12:10。'HHMMSS'
或者HHMMSS
。如果插入一个不合法的字符串或者数字,MySQL在存储数据时,会将其自动转化为00:00:00进行存储。比如1210,MySQL会将最右边的两位解析成秒,表示00:12:10,而不是12:10:00。CURRENT_TIME()
或者NOW()
,会插入当前系统的时间。8 个
字节的存储空间。在格式上
为DATE类型和TIME类型的组合,可以表示为 YYYY-MM-DD HH:MM:SS
,其中YYYY表示年份,MM表示月
份,DD表示日期,HH表示小时,MM表示分钟,SS表示秒。YY-MM-DD HH:MM:SS
格式或者 YYMMDDHHMMSS
格式的字符串插入DATETIME类型的字段时,两位数的年份规则符合YEAR类型的规则,00到69表示2000到2069;70到99表示1970到1999。CURRENT_TIMESTAMP()
和NOW()
,可以向DATETIME类型的字段插入系统的当前日期和时间。YYYY-MM-DD
,HH:MM:SS
,需要4个字节的存储空间。但是TIMESTAMP存储的时间范围比DATETIME要小很多,只能存储
“1970-01-01 00:00:01 UTC”到“2038-01-19 03:14:07 UTC”之间的时间。其中,UTC表示世界统一时间,也叫
作世界标准时间。存储数据的时候需要对当前时间所在的时区进行转换,查询数据的时候再将时间转换回当前的时区。因此,使用TIMESTAMP存储的同一个时间值,在不同的时区查询时会显示不同的时间。
TIMESTAMP底层存储的是毫秒值
,距离1970-1-1 0:0:0 0毫秒的毫秒值。TIMESTAMP更方便、更快
。尽量用 DATETIME 类型
。因为这个数据类型包括了完整的日期和时间信息,取值范围也最大,使用起来比较方便。毕竟,如果日期时间信息分散在好几个字段,很不容易记,而且查询的时候,SQL 语句也会更加复杂。不建议使用DATETIME存储,而是使用时间戳
,因为DATETIME虽然直观,但不便于计算。字符串(文本)类型 | 特点 | 长度 | 长度范围 | 占用的存储空间 |
---|---|---|---|---|
CHAR(M) | 固定长度 | M | 0 <= M <= 255 | M个字节 |
VARCHAR(M) | 可变长度 | M | 0 <= M <= 65535 | (实际长度 + 1) 个字节 |
CHAR和VARCHAR类型都可以存储比较短的字符串。
CHAR类型
:
VARCHAR类型
:
类型 | 特点 | 空间上 | 时间上 | 适用场景 |
---|---|---|---|---|
CHAR(M) | 固定长度 | 浪费存储空间 | 效率高 | 存储不大,速度要求高 |
VARCHAR(M) | 可变长度 | 节省存储空间 | 效率低 | 非CHAR的情况 |
MyISAM
数据存储引擎和数据列:MyISAM数据表,最好使用固定长度(CHAR)的数据列代替可变长度(VARCHAR)的数据列。这样使得整个表静态化,从而使 数据检索更快 ,用空间换时间。MEMORY
存储引擎和数据列:MEMORY数据表目前都使用固定长度的数据行存储,因此无论使用CHAR或VARCHAR列都没有关系,两者都是作为CHAR类型处理的。InnoDB
存储引擎,建议使用VARCHAR类型。因为对于InnoDB数据表,内部的行存储格式并没有区分固定长度和可变长度列(所有数据行都使用指向数据列值的头指针),而且主要影响性能的因素是数据行使用的存储总量,由于char平均占用的空间多于varchar,所以除了简短并且固定长度的,其他考虑varchar。这样节省空间,对磁盘I/O和数据存储总量比较好。由于实际存储的长度不确定,MySQL 不允许 TEXT 类型的字段做主键
。遇到这种情况,你只能采用CHAR(M),或者 VARCHAR(M)。文本字符串类型 | 特点 | 长度 | 长度范围 | 占用的存储空间 |
---|---|---|---|---|
TINYTEXT | 小文本、可变长度 | L | 0 <= L <= 255 | L + 2 个字节 |
TEXT | 文本、可变长度 | L | 0 <= L <= 65535 | L + 2 个字节 |
MEDIUMTEXT | 中等文本、可变长度 | L | 0 <= L <= 16777215 | L + 3 个字节 |
LONGTEXT | 大文本、可变长度 | L | 0 <= L<= 4294967295(相当于4GB) | L + 4 个字节 |
赋值时,值只能是ENUM中声明的值之中的一个,值只能是中ENUM声明的。
文本字符串类型 | 长度 | 长度范围 | 占用的存储空间 |
---|---|---|---|
ENUM | L | 1 | <= L <= 65535 |
示例:
SET在赋值时,值可以时SET声明的值中的多个,这一点与ENUM类型不同,值只能是SET中声明的。
成员个数范围(L表示实际成员个数) | 占用的存储空间 |
---|---|
1 <= L <= 8 | 1个字节 |
9 <= L <= 16 | 2个字节 |
17 <= L <= 24 | 3个字节 |
25 <= L <= 32 | 4个字节 |
33 <= L <= 64 | 8个字节 |
存储一些二进制数据,比如可以存储图片、音频和视频等二进制数据。
二进制字符串类型 | 特点 | 值的长度 | 占用空间 |
---|---|---|---|
BINARY(M) | 固定长度 | M | (0 <= M <= 255) |
VARBINARY(M) | 可变长度 | M | (0 <= M <= 65535) |
示例:
二进制大对象
,可以容纳可变数量的数据。图片 、 音频 和 视频
等。 服务器的磁盘上
,并将图片、音频和视频的访问路径存储到MySQL中。二进制字符串类型 | 值的长度 | 长度范围 | 占用空间 |
---|---|---|---|
TINYBLOB | L | 0 <= L <= 255 | L + 1 个字节 |
BLOB L 0 <= | L | <= 65535(相当于64KB) | L + 2 个字节 |
MEDIUMBLOB | L | 0 <= L <= 16777215 (相当于16MB) | L + 3 个字节 |
LONGBLOB | L | 0 <= L <= 4294967295(相当于4GB) | L + 4 个字节 |
示例:
在使用text和blob字段类型时要注意以下几点,以便更好的发挥数据库的性能。
空洞
",以后填入这些"空洞"的记录可能长度不同。为了提高性能,建议定期使用 OPTIMIZE TABLE 功能对这类表进行 碎片整理
。 前缀索引
。但是仍然要在不必要的时候避免检索大型的BLOB或TEXT值。例如,SELECT * 查询就不是很好的想法,除非你能够确定作为约束条件的WHERE子句只会找到所需要的数据行。否则,你可能毫无目的地在网络上传输大量的值。分离到单独的表
中。在某些环境中,如果把这些数据列移动到第二张数据表中,可以让你把原数据表中的数据列转换为固定长度的数据行格式,那么它就是有意义的。这会 减少主表中的碎片
,使你得到固定长度数据行的性能优势。它还使你在主数据表上运行 SELECT * 查询的时候不会通过网络传输大量的BLOB或TEXT值。数据交换格式
。简洁和清晰的层次结构使得 JSON 成为理想的数据交换语言。它易于人阅读和编写,同时也易于机器解析和生成,并有效地提升网络传输效率。JSON 可以将 JavaScript 对象中表示的一组数据转换为字符串,然后就可以在网络或者程序之间轻松地传递这个字符串,并在需要的时候将它还原为各编程语言所支持的数据格式
。 Geometry(几何)
来表示所有地理特征。Geometry指一个点或点的集合,代表世界上任何具有位置的事物。整数
,就用 INT
; 如果是小数
,一定用定点数类型DECIMAL(M,D)
; 如果是日期与时间,就用 DATETIME
。阿里巴巴《Java开发手册》之MySQL数据库:
【 强制 】
小数类型为 DECIMAL,禁止使用 FLOAT 和 DOUBLE。
说明:在存储的时候,FLOAT 和 DOUBLE 都存在精度损失的问题,很可能在比较值的时候,得到不正确的结果。如果存储的数据范围超过 DECIMAL 的范围,建议将数据拆成整数和小数并分开存储。【 强制 】
如果存储的字符串长度几乎相等,使用 CHAR 定长字符串类型。【 强制 】
VARCHAR 是可变长字符串,不预先分配存储空间,长度不要超过 5000。如果存储长度大于此值,定义字段类型为 TEXT,独立出来一张表,用主键来对应,避免影响其它字段索引效率。为了保证数据的完整性。
数据完整性
(Data Integrity)是指数据的精确性(Accuracy)和可靠性(Reliability)。它是防止数据库中存在不符合语义规定的数据和防止因错误信息的输入输出造成无效操作或错误信息而提出的。对表数据进行额外的条件限制
。从以下四个方面考虑: 实体完整性(Entity Integrity)
:例如,同一个表中,不能存在两条完全相同无法区分的记录。域完整性(Domain Integrity
) :例如:年龄范围0-120,性别范围“男/女”。引用完整性(Referential Integrity
) :例如:员工所在部门,在部门表中要能找到这个部门。用户自定义完整性(User-defined Integrity)
:例如:用户名唯一、密码不能为空等,本部门经理的工资不得高于本部门职工的平均工资的5倍。CREATE TABLE 表(字段1 数据类型1 约束1,字段2 数据类型2 约束2...);
ALERT TABLE 表名称 MODIFY 字段名 数据类型 约束;
SELECT * FROM 表 WHERE 字段 = 表;
空字符串不等于NULL,0不等于NULL
。当已存在的字段有NULL值时,不允许修改为NOT NULL
CREATE TABLE 表名称(字段名 数据类型, 字段名 数据类型 NOT NULL, 字段名 数据类型 NOT NULL);
alter table 表名称 modify 字段名 数据类型 not null;
alter table 表名称 modify 字段名 数据类型 NULL;
,
修改为NULL,相当于修改某个非注解字段,该字段允许为空。alter table 表名称 modify 字段名 数据类型;
去掉not null,相当于修改某个非注解字段,该字段允许为空。MySQL会给唯一约束的字段上默认创建一个唯一索引
。ALERT TABLE 表 DROP INDEX 约束名;
可任意通过SHOW INDEX FROM 表;
来查看表的索引。PRIMARY KEY
主键不允许重复,不允许出现空值。
相当于唯一约束+非空约束的组合。一个表中,只能有一个主键约束。
MySQL的主键名固定为PRIMARY
,就算自己命名了主键约束名也没用。主键索引
(能够根据种族间查询的,就根据主键查询,效率更高)。如果删除了主键约束,主键约束对应的索引就自动删除了。列级:
create table 表名称(
字段名 数据类型 primary key, #列级模式
字段名 数据类型,
字段名 数据类型
);
表级:
create table 表名称(
字段名 数据类型,
字段名 数据类型,
字段名 数据类型,
[constraint 约束名] primary key(字段名) #表级模式
);
ALTER TABLE 表名称 ADD PRIMARY KEY(字段列表); #字段列表可以是一个字段,也可以是多个字段,如果是多
个字段的话,是复合主键
create table 表名称(
字段名 数据类型,
字段名 数据类型,
字段名 数据类型,
primary key(字段名1,字段名2) #表示字段1和字段2的组合是唯一的,也可以有更多个字段
);
alter table 表名称 drop primary key;
AUTO_INCREMENT
建表时指定
列级:
create table 表名称(
字段名 数据类型 primary key auto_increment,#该字段为自增长字段,通常为主键id
字段名 数据类型 unique key not null,
字段名 数据类型 unique key,
字段名 数据类型 not null default 默认值,
);
表级:
create table 表名称(
字段名 数据类型 default 默认值 ,
字段名 数据类型 unique key auto_increment,
字段名 数据类型 not null default 默认值,,
primary key(字段名)
);
建表后
alter table 表名称 modify 字段名 数据类型 auto_increment;
alter table 表名称 modify 字段名 数据类型;
去掉auto_increment相当于删除
添加约束语法 alter table 表名称 modify 字段名 数据类型 auto_increment;
给这个字段添加自增约束计数器
来决定的,而该计数器只在内存中维护
,并不会持节话到硬盘中,当重启数据库时,该计数器就会被初始化
。重做日志
中,每次计数器发生改变,都会将其写入重做日志中。如果数据库重启,InnoDB会根据重做日志中的信息来初始化计数器的内存值,因而重启服务器不会出现第二点的情况。FOREIGN KEY
默认名不是列名,而是自动产生一个外键名
。先创建主表,再创建从表
。先删从表(或外键约束),再删主表
。建表时添加
先创建主表:
create table 主表名称(
字段1 数据类型 primary key,
字段2 数据类型
);
再创建从表:
create table 从表名称(
字段1 数据类型 primary key,
字段2 数据类型,
[CONSTRAINT 外键约束名称] FOREIGN KEY(从表的某个字段) references 主表名(被参考字段)
);
#(从表的某个字段)的数据类型必须与主表名(被参考字段)的数据类型一致,逻辑意义也一样
#(从表的某个字段)的字段名可以与主表名(被参考字段)的字段名一样,也可以不一样
-- FOREIGN KEY: 在表级指定子表中的列
-- REFERENCES: 标示在父表中的列
示例:
create table dept( #主表
did int primary key, #部门编号
dname varchar(50) #部门名称
);
create table emp(#从表
eid int primary key, #员工编号
ename varchar(5), #员工姓名
deptid int, #员工所在的部门
foreign key (deptid) references dept(did) #在从表中指定外键约束
#emp表的deptid和和dept表的did的数据类型一致,意义都是表示部门的编号
);
说明:
(1)主表dept必须先创建成功,然后才能创建emp表,指定外键成功。
(2)删除表时,先删除从表emp,再删除主表dept
建表后添加:
一般情况下,表与表的关联都是提前设计好了的,因此,会在创建表的时候就把外键约束定义好。不过,如果需要修改表的设计(比如添加新的字段,增加新的关联关系),但没有预先定义外键约束,那么,就要用修改表的方式来补充定义。
ALTER TABLE 从表名 ADD [CONSTRAINT 约束名] FOREIGN KEY (从表的字段) REFERENCES 主表名(被引用字段) [on update xx][on delete xx];
(1)第一步先查看约束名和删除外键约束
# 查看约束名
SELECT * FROM information_schema.table_constraints WHERE table_name = '表名称';#查看某个表的约束名
# 删除外键约束
ALTER TABLE 从表名 DROP FOREIGN KEY 外键约束名;
(2)第二步查看索引名和删除索引。(注意,只能手动删除)
# 查看索引名
SHOW INDEX FROM 表名称; #查看某个表的索引名
# 删除索引
ALTER TABLE 从表名 DROP INDEX 索引名;
单机低并发
,不适合 分布式
、 高并发集群
;级联更新是强阻塞,存在数据库更新风暴
的风险;外键影响数据库的插入速度
。Cascade方式
:在父表上update/delete记录时,同步update/delete掉子表的匹配记录。Set null方式
:在父表上update/delete记录时,将子表上匹配记录的列设为null,但是要注意子表的外键列不能为not null。No action方式
:如果子表中有匹配的记录,则不允许对父表对应候选键进行update/delete操作。Restrict方式
:同no action, 都是立即检查外键约束。Set default方式
(在可视化工具SQLyog中可能显示空白):父表有变更时,子表将外键列设置成一个默认的值,但Innodb不能识别。如果没有指定等级,就相当于Restrict方式。
ON UPDATE CASCADE ON DELETE RESTRICT
的方式。建主表:
create table dept(
did int primary key, #部门编号
dname varchar(50) #部门名称
);
建子表:
create table emp(
eid int primary key, #员工编号
ename varchar(5), #员工姓名
deptid int, #员工所在的部门
# 约束等级
foreign key (deptid) references dept(did) on update cascade on delete set null
#把修改操作设置为级联修改等级,把删除操作设置为set null等级
);
CHECK
CHECK(字段 运算符 值)
# 示例1
CREATE TABLE temp(
id INT AUTO_INCREMENT,
NAME VARCHAR(20),
age INT CHECK(age > 20),
PRIMARY KEY(id)
);
# 示例2
age tinyint check(age >20) 或 sex char(2) check(sex in(‘男’,’女’))
#示例3
CHECK(height>=0 AND height<3)
DEFAULT
创建时添加:
create table 表名称(
字段名 数据类型 primary key,
字段名 数据类型 unique key not null,
字段名 数据类型 unique key,
# 指定默认值
字段名 数据类型 not null default 默认值,
);
建表后添加:
alter table 表名称 modify 字段名 数据类型 default 默认值;
如果这个字段原来有非空约束,你还保留非空约束,那么在加默认值约束时,还得保留非空约束,否则非空约束就被删除了
同理,在给某个字段加非空约束也一样,如果这个字段原来有默认值约束,
你想保留,也要在modify语句中保留默认值约束,否则就删除了
alter table 表名称 modify 字段名 数据类型 default 默认值 not null;
alter table 表名称 modify 字段名 数据类型 ;
删除默认值约束,也不保留非空约束alter table 表名称 modify 字段名 数据类型 not null;
删除默认值约束,保留非空约束示例:
alter table employee modify gender char;
删除gender字段默认值约束,如果有非空约束,也一并删除alter table employee modify tel char(11) not null;
删除tel字段默认值约束,保留非空约束MySQL虽然提供了外键约束,但我们一般不使用
,通常会在Java中进行约束,因为在sql中约束,当外键关联过多时,会对系统造成阻塞,影响系统性能。 因为外键约束的系统开销而变得非常慢
。所以, MySQL 允许你不使用系统自带的外键约束,在应用层面
完成检查数据一致性的逻辑。也就是说,即使你不用外键约束,也要想办法通过应用层面的附加逻辑,来实现外键约束的功能,确保数据的一致性。对象 | 描述 |
---|---|
表(TABLE) | 表是存储数据的逻辑单元,以行和列的形式存在,列就是字段,行就是记录 |
数据字典 | 就是系统表,存放数据库相关信息的表。系统表的数据通常由数据库系统维护,程序员通常不应该修改,只可查看 |
约束(CONSTRAINT) | 执行数据校验的规则,用于保证数据完整性的规则 |
视图(VIEW) | 一个或者多个数据表里的数据的逻辑显示,视图并不存储数据 |
索引(INDEX) | 用于提高查询性能,相当于书的目录 |
存储过程(PROCEDURE) | 用于完成一次完整的业务处理,没有返回值,但可通过传出参数将多个值传给调用环境 |
存储函数(FUNCTION) | 用于完成一次特定的计算,具有一个返回值 |
触发器(TRIGGER) | 相当于一个事件监听器,当数据库发生特定事件后,触发器被触发,完成相应的处理 |
虚拟
表,本事是不具有数据的,占用空间内存很少,他是SQL中的一个重要概念
。基表
。本质
就是一个存储SELECT语句的虚拟表。其实视图就是一个代理基表的虚拟表,它可以选择展示基表中的某些字段;也可以代理对基表数据的操作,任何会视图的操作都会映射到基表中;视图是由SELECT语句构成的。
CREATE VIEW
语句中,嵌入子查询创建视图精简版:
CREATE VIEW 视图名称
AS 查询语句
完整版:
CREATE [OR REPLACE]
[ALGORITHM = {UNDEFINED | MERGE | TEMPTABLE}]
VIEW 视图名称 [(字段列表)]
AS 查询语句
[WITH [CASCADED|LOCAL] CHECK OPTION]
方式1:字段别名
CREATE VIEW 视图
AS
SELECT 字段1 别名1,字段2 别名2
FROM 表
方式2:视图列表
# 视图列表的字段和SELECT查询的字段数量,顺序必须一致。
CREATE VIEW 视图(视图字段1,视图字段2...)# 视图列表
AS
SELECT 字段1,字段2
FROM 表
CREATE VIEW 视图名
AS
SELECT 字段1,字段2
FROM 表1,表2
CREATE VIEW emp_depart
AS
SELECT CONCAT(last_name,'(',department_name,')') AS emp_dept
FROM employees e JOIN departments d
WHERE e.department_id = d.department_id
SHOW TABLES;
DESC / DESCRIBE 视图名称;
SHOW TABLE STATUS LIKE '视图名称'\G;
某些终端可能无法识别“\G”
结果显示,注释Comment为VIEW,说明该表为视图,其他的信息为NULL,说明这是一个虚表。SHOW CREATE VIEW 视图名称;
视图通常只用于查询数据,不用于操作数据
)若操作的视图的字段不存在于基表中,则操作失败
不可更新的情况:
一对一
的关系。另外当视图定义出现如
下情况时,视图不支持更新操作: JOIN联合查询
,视图将不支持INSERT和DELETE操作; 数学表达式
或 子查询
,视图将不支持INSERT,也DISTINCT 、 聚合函数 、 GROUP BY 、 HAVING 、 UNION
等,视图将不支持INSERT、UPDATE、DELETE;不可更新视图
; 总结:虽然可以更新视图数据,但总的来说,视图作为 虚拟表
,主要用于 方便查询
,不建议更新视图的
数据。对视图数据的更改,都是通过对实际数据表里数据的操作来完成的
。
方式1:CREATE OR REPLACE VIEW
# 当字段存在的时候就修改字段,不存在就创建字段
CREATE OR REPLACE VIEW 视图(视图列表)
AS
SELECT 字段1,字段2...
FROM 表
方式2:ALTER VIEW
ALTER VIEW 视图
AS
SELECT语句
DROP VIEW IF EXISTS 视图名称;
访问限制
在某些数据的结果集上,而这些数据的结果集可以使用视图来实现。用
户不必直接查询或操作数据表。这也可以理解为视图具有 隔离性
。视图相当于在用户和实际的数据表之
间加了一层虚拟表。
同时,MySQL可以根据权限将用户对数据的访问限制在某些视图上,用户不需要查询数据表,可以直接 通过视图获取数据表中的信息
。这在一定程度上保障了数据表中数据的安全性。预先编译
的 SQL 语句的封装。
执行过程:存储过程预先存储在 MySQL 服务器上,需要执行的时候,客户端只需要向服务器端发出调用存储过程的命令,服务器端就可以把预先存储好的这一系列 SQL 语句全部执行。视图是虚拟表
,通常不对底层数据表直接操作,而存储过程是程序化的SQL,可以直接操作底层数据表
,相比于面向集合的操作方式,能够实现一些更复杂的数据处理。没有返回值
的。CREATE PROCEDURE 存储过程名(IN|OUT|INOUT 参数名 参数类型,...)
[characteristics ...]
BEGIN
存储过程体
END
说明:
characteristics
表示创建存储过程时指定的对存储过程的约束条件,其取值信息如下:LANGUAGE SQL
| [NOT] DETERMINISTIC
| { CONTAINS SQL | NO SQL | READS SQL DATA | MODIFIES SQL DATA }
| SQL SECURITY { DEFINER | INVOKER }
| COMMENT 'string'
其中:
LANGUAGE SQL
:说明存储过程执行体是由SQL语句组成的,当前系统支持的语言为SQL。 [NOT] DETERMINISTIC
:指明存储过程执行的结果是否确定。DETERMINISTIC表示结果是确定的。每次执行存储过程时,相同的输入会得到相同的输出。NOT DETERMINISTIC表示结果是不确定的,相同的输入可能得到不同的输出。如果没有指定任意一个值,默认为NOT DETERMINISTIC。{ CONTAINS SQL | NO SQL | READS SQL DATA | MODIFIES SQL DATA }
:指明子程序使用SQL语句的限制。 SQL SECURITY { DEFINER | INVOKER }
:执行当前存储过程的权限,即指明哪些用户能够执行当前存储过程。 DEFINER
表示只有当前存储过程的创建者或者定义者才能执行当前存储过程;INVOKER
表示拥有当前存储过程的访问权限的用户能够执行当前存储过程。DELIMITER 新的结束标记
DELIMITER结束标记示例:
DELIMITER $
CREATE PROCEDURE 存储过程名(IN|OUT|INOUT 参数名1 参数类型1,...)
[characteristics ...]
BEGIN
sql语句1;
sql语句2;
END $
# 创建存储过程select_all_data(),查看 emps 表的所有数据
ELIMITER $
CREATE PROCEDURE select_all_data()
BEGIN
SELECT * FROM emps;
END $
# 最后再改回“;”分号,为了不影响后续使用。
DELIMITER ;
CALL 存储过程名(实参列表)
逐步推进
,就可以完成对存储过程中所有操作的调试了。当然,你也可以把存储过程中的 SQL 语句复制出来,逐段单独调试。# SET GLOBAL log_bin_trust_function_creators = 1; 报错解决方案1
DELIMITER //
CREATE FUNCTION 函数名(参数名 参数类型,...)
RETURNS 返回值类型
[characteristics ...]
# 报错解决方案2:
# LANGUAGE SQL
# NOT DETERMINISTIC
# READS SQL DATA
# SQL SECURITY DEFINER
# COMMENT '查询部门平均工资'
BEGIN
函数体 #函数体中肯定有 RETURN 语句
END //
DELIMITER ;
说明:
1、参数列表:指定参数为IN、OUT或INOUT只对PROCEDURE是合法的,FUNCTION中总是默认为IN参数。
2、RETURNS type 语句表示函数返回数据的类型;
RETURNS子句只能对FUNCTION做指定,对函数而言这是 强制 的。它用来指定函数的返回类型,而且函数体必须包含一个 RETURN value 语句。
3、characteristic 创建函数时指定的对函数的约束。取值与创建存储过程时相同,这里不再赘述。
4、函数体也可以用BEGIN…END来表示SQL代码的开始和结束。如果函数体只有一条语句,也可以省略BEGIN…END。
5、若在创建存储函数中报错 `you might want to use the less safe log_bin_trust_function_creators variable `,有两种处理方法:
方式1:加上必要的函数特性“[NOT] DETERMINISTIC”和“{CONTAINS SQL | NO SQL | READS SQL DATA |MODIFIES SQL DATA}”
方式2:在创建函数前加上SET GLOBAL log_bin_trust_function_creators = 1;
用户自己定义
的,而内部函数是MySQL
的 开发者定义
的
SELECT 函数名(实参列表);
关键字 | 调用语法 | 返回值 | 应用场景 | |
---|---|---|---|---|
存储过程 | PROCEDURE | CALL 存储过程() | 理解为有0个或多个 | 一般用于更新 |
存储函数 | FUNCTION | SELECT 函数() | 只能是一个 | 一般用于查询结果为一个值并返回时 |
存储函数可以放在查询语句中使用,存储过程不行
。反之,存储过程的功能更加强大,包括能够
执行对表的操作(比如创建表,删除表等)和事务操作,这些功能是存储函数不具备的。SHOW CREATE PROCEDURE | FUNCTION 存储过程名 或 函数名
SHOW PROCEDURE | FUNCTION STATUS [LIKE '函数名|存储过程名']
若不指定具体的函数或存储过程则查询数据库中所有的函数或存储过程。SELECT * FROM information_schema.Routines
WHERE ROUTINE_NAME='存储过程或函数的名' [AND ROUTINE_TYPE = {'PROCEDURE|FUNCTION'}];
ALTER {PROCEDURE | FUNCTION} 存储过程或函数的名 [characteristic ...]
{ CONTAINS SQL | NO SQL | READS SQL DATA | MODIFIES SQL DATA }| SQL SECURITY { DEFINER | INVOKER }| COMMENT 'string'
DROP {PROCEDURE | FUNCTION} [IF EXISTS] 存储过程或函数的名
系统变量
以及 用户自定义变量
。服务器
层面。启动MySQL服务,生成MySQL服务实例期间,MySQL将为MySQL服务器内存中的系统变量赋值,这些系统变量定义了当前MySQL服务实例的属性、特征。这些系统变量的值要么是 编译MySQL时参数
的默认值,要么是 配置文件
(例如my.ini等)中的参数值。大家可以通过网址https://dev.mysql.com/doc/refman/8.0/en/serversystemvariables.html
查看MySQL文档的系统变量。global
关键字声明全局变量
。session
关键字声明local变量
。属于特殊的全局系统变量。
默认为会话系统变量
。 两个“@”
开头, “@@”首先标记会话系统变量,如果会话系统变量不存在,则标记全局系统变量。
SHOW GLOBAL VARIABLES;
查看所有全局变量。SHOW SESSION VARIABLES;
或SHOW VARIABLES;
查看所有会话变量。SHOW GLOBAL VARIABLES LIKE '%标识符%';
查看满足条件的部分系统变量。SHOW SESSION VARIABLES LIKE '%标识符%';
#查看满足条件的部分会话变量。SELECT @@global.变量名;
查看指定的系统变量的值。SELECT @@session.变量名;
或者SELECT @@变量名;
查看指定的会话变量的值。SET @@global.变量名=变量
SET GLOBAL 变量名=变量值;
SET @@session.变量名=变量值;
SET SESSION 变量名=变量值;
SET PERSIST GLOBAL 变量名=变量值;
SET @用户变量 = 值;
SET @用户变量 := 值;
SELECT @用户变量 := 表达式 [FROM 等子句];
SELECT 表达式 INTO @用户变量 [FROM 等子句];
SELECT @用户变量
DECLARE 变量名 类型 [default 值]; # 如果没有DEFAULT子句,初始值为NUL
SET 变量名=值;
或SET 变量名:=值;
SELECT 字段名或表达式 INTO 变量名 FROM 表
SELECT 局部变量名;
语法格式
BEGIN
#声明局部变量
DECLARE 变量名1 变量数据类型 [DEFAULT 变量默认值];
DECLARE 变量名2,变量名3,... 变量数据类型 [DEFAULT 变量默认值];
#为局部变量赋值
SET 变量名1 = 值;
SELECT 值 INTO 变量名2 [FROM 子句];
#查看局部变量的值
SELECT 变量1,变量2,变量3;
END
作用域 | 定义位置 | 语法 | |
---|---|---|---|
会话用户变量 | 当前会话 | 会话的任何地方 | 加@符号,不用指定类型 |
局部变量 | 定义它的BEGIN END | 中 BEGIN END的第一句话 | 一般不用加@,需要指定类型 |
定义条件
是事先定义程序执行过程中可能遇到的问题, 处理程序
定义了在遇到问题时应当采取的处理方式,并且保证存储过程或函数在遇到警告或错误时能继续执行。这样可以增强存储程序处理问题的能力,避免程序异常停止运行。定义条件与处理程序其实就是异常处理
错误名字
和 指定的 错误条件
关联起来。这个名字可以随后被用在定义处理程序的DECLARE HANDLER
语句中。DECLARE 错误名称 CONDITION FOR 错误码(或错误条件)
MySQL_error_code
和sqlstate_value
都可以表示MySQL的错误。 DECLARE 错误名称 CONDITION FOR sqlstate_value 错误码(或错误条件)
sqlstate_value在定义和使用是,前边必须加上 sqlstate_value,这样才能加后边的错误码识别为字符串
。DECLARE 处理方式 HANDLER FOR 错误类型 处理语句
CONTINUE
:表示遇到错误不处理,继续执行。EXIT
:表示遇到错误马上退出。UNDO
:表示遇到错误后撤回之前的操作。MySQL中暂时不支持这样的操作。SQLSTATE '字符串错误码'
:表示长度为5的sqlstate_value类型的错误代码;MySQL_error_code
:匹配数值类型错误代码;SQLWARNING
:匹配所有以01开头的SQLSTATE错误代码;NOT FOUND
:匹配所有以02开头的SQLSTATE错误代码;SQLEXCEPTION
:匹配所有没有被SQLWARNING或NOT FOUND捕获的SQLSTATE错误代码;SET 变量 = 值
”这样的简单语句,也可以是使用 BEGIN ... END
编写的复合语句。定义异常处理程序的6种方式:
#方法1:捕获sqlstate_value
DECLARE CONTINUE HANDLER FOR SQLSTATE '42S02' SET @info = 'NO_SUCH_TABLE';
#方法2:捕获mysql_error_value
DECLARE CONTINUE HANDLER FOR 1146 SET @info = 'NO_SUCH_TABLE';
#方法3:先定义条件,再调用
DECLARE no_such_table CONDITION FOR 1146;
DECLARE CONTINUE HANDLER FOR NO_SUCH_TABLE SET @info = 'NO_SUCH_TABLE';
#方法4:使用SQLWARNING
DECLARE EXIT HANDLER FOR SQLWARNING SET @info = 'ERROR';
#方法5:使用NOT FOUND
DECLARE EXIT HANDLER FOR NOT FOUND SET @info = 'NO_SUCH_TABLE';
#方法6:使用SQLEXCEPTION
DECLARE EXIT HANDLER FOR SQLEXCEPTION SET @info = 'ERROR';
案例示例:
创建一个名称为“InsertDataWithCondition”的存储过程,代码如下。
在存储过程中,定义处理程序,捕获sqlstate_value值,当遇到sqlstate_value值为23000时,执行EXIT操
作,并且将@proc_value的值设置为-1。
#准备工作
CREATE TABLE departments
AS
SELECT * FROM atguigudb.`departments`;
ALTER TABLE departments
ADD CONSTRAINT uk_dept_name UNIQUE(department_id);
# 创建一个名称为“InsertDataWithCondition”的存储过程
DELIMITER //
CREATE PROCEDURE InsertDataWithCondition()
BEGIN
# 定义条件
DECLARE duplicate_entry CONDITION FOR SQLSTATE '23000' ;
# 捕获sqlstate_value值,当遇到sqlstate_value值为23000时,执行EXIT操作,并且将@proc_value的值设置为-1
DECLARE EXIT HANDLER FOR duplicate_entry SET @proc_value = -1;
SET @x = 1;
INSERT INTO departments(department_name) VALUES('测试');
SET @x = 2;
INSERT INTO departments(department_name) VALUES('测试');
SET @x = 3;
END //
DELIMITER ;
调用存储过程:
UPDATE employees SET email = NULL WHERE last_name = 'Abel';
SET @x = 2;
UPDATE employees SET email = 'aabbel' WHERE last_name = 'Abel';
SET @x = 3;
END //
DELIMITER ;
CALL UpdateDataWithCondition();
条件判断语句
:IF 语句和 CASE 语句循环语句
:LOOP、WHILE 和 REPEAT跳转语句
:ITERATE 和 LEAVE 语句IF 语句的语法结构是:
IF 表达式1 THEN 操作1
[ELSEIF 表达式2 THEN 操作2]……
[ELSE 操作N]
END IF;
根据表达式的结果为TRUE或FALSE执行相应的语句。这里“[]”中的内容是可选的。
特点:① 不同的表达式对应不同的操作 ② 使用在begin end中
CASE 语句的语法结构1:
#情况一:类似于switch
CASE 表达式
WHEN 值1 THEN 结果1或语句1(如果是语句,需要加分号)
WHEN 值2 THEN 结果2或语句2(如果是语句,需要加分号)
...
ELSE 结果n或语句n(如果是语句,需要加分号)
END [case](如果是放在begin end中需要加上case,如果放在select后面不需要)
CASE 语句的语法结构2:
#情况二:类似于多重if
CASE
WHEN 条件1 THEN 结果1或语句1(如果是语句,需要加分号)
WHEN 条件2 THEN 结果2或语句2(如果是语句,需要加分号)
...
ELSE 结果n或语句n(如果是语句,需要加分号)
END [case](如果是放在begin end中需要加上case,如果放在select后面不需要)
LOOP语句的基本格式如下:
[标签名称]:LOOP
循环体
退出条件 # 注意:循环体一定要有退出条件
END LOOP [标签名称]
其中,标签名称,可自定义,可以省略,可用于结束循环时指定结束的某个循环。
示例:
DECLARE id INT DEFAULT 0;
add_loop:LOOP
# 循环体
SET id = id +1;
# 退出条件
IF id >= 10 THEN LEAVE add_loop;
END IF;
END LOOP add_loop;
WHILE语句的基本格式如下:
[标签名称:] WHILE 循环条件 DO
循环体
END WHILE [标签名称];
①标签名称需自定义,可省。
②如果循环条件结果为真,WHILE语句内的语句或语句群被执行,直至循环条件为假,退出循环。
示例:
DELIMITER //
CREATE PROCEDURE test_while()
BEGIN
DECLARE i INT DEFAULT 0;
# 循环
WHILE i < 10 DO
SET i = i + 1;
END WHILE;
SELECT i;
END //
DELIMITER ;
#调用
CALL test_while();
先执行一次,再判断
REPEAT语句的基本格式如下:
[标签名称:] REPEAT
循环体
UNTIL 结束循环表达式
END REPEAT [标签名称];
①标签名称需自定义,可以省略;
②REPEAT语句内的语句或语句群被重复,直至结束循环表达式为真。
LEAVE 标签名
示例:
DELIMITER //
CREATE PROCEDURE leave_begin(IN num INT)
begin_label: BEGIN
# 小于0时,退出循环
IF num<=0
THEN LEAVE begin_label;
ELSEIF num=1
THEN SELECT AVG(salary) FROM employees;
ELSEIF num=2
THEN SELECT MIN(salary) FROM employees;
ELSE
SELECT MAX(salary) FROM employees;
END IF;
SELECT COUNT(*) FROM employees;
END //
DELIMITER ;
ITERATE 标签名
示例:
DELIMITER //
CREATE PROCEDURE test_iterate()
BEGIN
DECLARE num INT DEFAULT 0;
my_loop:LOOP
SET num = num + 1;
#小于10时,结束本次循环
IF num < 10
THEN ITERATE my_loop;
ELSEIF num > 15
THEN LEAVE my_loop;
END IF;
SELECT '尚硅谷:让天下没有难学的技术';
END LOOP my_loop;
END //
DELIMITER ;
随意定位到某一条记录
,或者逐条处理数据
,并对记录的数据进行处理。游标让 SQL 这种面向集合的语言有了面向过程开发的能力
。充当了指针的作用
,我们可以通过操作游标来对数据行进行操作。DECLARE 游标名 CURSOR FOR SELECT语句;
这个语法适用于 MySQL,SQL Server,DB2 和 MariaDB。
DECLARE cursor_name CURSOR IS select_statement;
这个语法适用于Oracle 或者 PostgreSQLOPEN 游标名
FETCH 游标名 INTO 变量 [,变量] ...
这句的作用是使用游标名这个游标来读取当前行,并且将数据保存到变量中(读取游标,并赋值给变量
),游标指针指到下一行。如果游标读取的数据行有多个列名,则在 INTO 关键字后面赋值给多个变量名即可。通常写在循环中。
注意:游标的查询结果集中的字段数,必须跟 INTO 后面的变量数一致,CLOSE 游标名;
有 OPEN 就会有 CLOSE,也就是打开和关闭游标。当我们使用完游标后需要关闭掉该游标。因为游标会占用系统资源
,如果不及时关闭,游标会一直保持到存储过程结束
,影响系统运行的效率。而关闭游标的操作,会释放游标占用的系统资源。关闭游标之后,我们就不能再检索查询结果中的数据行,如果需要检索只能再次打开游标。示例:
创建存储过程“get_count_by_limit_total_salary()”,声明IN参数 limit_total_salary,DOUBLE类型;
声明OUT参数total_count,INT类型。函数的功能可以实现累加薪资最高的几个员工的薪资值,
直到薪资总和达到limit_total_salary参数的值,返回累加的人数给total_count。
DELIMITER //
CREATE PROCEDURE get_count_by_limit_total_salary(IN limit_total_salary DOUBLE,OUT
total_count INT)
BEGIN
DECLARE sum_salary DOUBLE DEFAULT 0; #记录累加的总工资
DECLARE cursor_salary DOUBLE DEFAULT 0; #记录某一个工资值
DECLARE emp_count INT DEFAULT 0; #记录循环个数
#定义游标
DECLARE emp_cursor CURSOR FOR SELECT salary FROM employees ORDER BY salary DESC;
#打开游标
OPEN emp_cursor;
REPEAT
#使用游标(从游标中获取数据)
FETCH emp_cursor INTO cursor_salary;
SET sum_salary = sum_salary + cursor_salary;
SET emp_count = emp_count + 1;
UNTIL sum_salary >= limit_total_salary
END REPEAT;
SET total_count = emp_count;
#关闭游标
CLOSE emp_cursor;
END //
DELIMITER ;
逐条读取
结果集中的数据,提供了完美的解决方案。跟在应用层面实现相同的功能相比,游标可以在存储程序中使用,效率高,程序也更加简洁。加锁
,这样在业务并发量大的时候,不仅会影响业务之间的效率,还会 消耗系统资源
,造成内存不足,这是因为游标是在内存中进行的处理。商品信息
和 库存信息
分别存放在 2 个不同的数据表中,我们在添加一条新商品记录的时候,为了保证数据的完整性,必须同时在库存表中添加一条库存记录。事务
包裹起来,确保这两个操作成为一个 原子操作
,即要么全部执行,要么全部不执行。要是遇到特殊情况,可能还需要对数据进行手动维护,这样就很 容易忘记其中的一步
,导致数据缺失。你可以创建一个触发器,让商品信息数据的插入操作自动触发库存数据的插入操作
。这样一来,就不用担心因为忘记添加库存数据而导致的数据缺失了。事件来触发
某个操作,这些事件包括 INSERT
、UPDATE
、 DELETE
事件。所谓事件就是指用户的动作或者触发某项行为。如果定义了触发程序,当数据库执行这些语句时候,就相当于事件发生了,就会 `自动 激发触发器执行相应的操作。语法结构:
CREATE TRIGGER 触发器名称
{BEFORE|AFTER} {INSERT|UPDATE|DELETE} ON 表名
FOR EACH ROW
触发器执行的语句块;
说明:
① 表名 :表示触发器监控的对象。
② BEFORE|AFTER :表示触发的时间。BEFORE 表示在事件之前触发;AFTER 表示在事件之后触发。
③ INSERT|UPDATE|DELETE :表示触发的事件。
INSERT 表示插入记录时触发;
UPDATE 表示更新记录时触发;
DELETE 表示删除记录时触发。
④ 触发器执行的语句块 :可以是单条SQL语句,也可以是由BEGIN…END结构组成的复合语句块。
示例:
定义触发器“salary_check_trigger”,基于员工表“employees”的INSERT事件,
在INSERT之前检查将要添加的新员工薪资是否大于他领导的薪资,如果大于领导薪资,
则报sqlstate_value为'HY000'的错误,从而使得添加失败。
DELIMITER //
CREATE TRIGGER salary_check_trigger
BEFORE INSERT ON employees FOR EACH ROW
BEGIN
DECLARE mgrsalary DOUBLE;
SELECT salary INTO mgrsalary FROM employees WHERE employee_id = NEW.manager_id;
IF NEW.salary > mgrsalary THEN # NEW关键字代表INSERT添加语句的新记录。
SIGNAL SQLSTATE 'HY000' SET MESSAGE_TEXT = '薪资高于领导薪资错误';
END IF;
END //
DELIMITER ;
查看触发器是查看数据库中已经存在的触发器的定义、状态和语法信息等。
SHOW TRIGGERS\G
SHOW CREATE TRIGGER 触发器名
SELECT * FROM information_schema.TRIGGERS;
DROP TRIGGER IF EXISTS 触发器名称;
不受应用层的控制
。这对系统维护是非常有挑战的。基于子表的UPDATE和DELETE语句定义的触发器并不会被激活
。MySQL从5.7版本开始直接跳越发布了8.0版本,可见这是一个令人兴奋 的里程碑版本,MySQL从8.0版本的更能上做了显著的改进增强,开发者对MySQL的源代码进行了重构,最突出的一点就是MySQL Optiizer优化器进行了改进。不仅在速度上得到了改善,还未用户带来了更好的性能和体验。
更简便的NoSQL支持
,NoSQL泛指非关系型数据库和数据储存,随着互联网平台的规模飞速发展,传统的关系型数据库已近越来越得不到满足需求。从5.6 版本开始,MySQL就开始支持简单的NoSQL的存储功能。MySQL 8.0 对这一功能做了优化,以更灵活的方式实现NoSQL的功能,不再依赖模式(schema)。更好的索引
,在查询中,只正确的使用索引可以提高查询的效率。MySQL8中新增了隐藏索引
和降序索引
。隐藏索引可以用来测试去掉索引对查询性能的影响。在查询中混合存在多列索引时,使用降序索引可以提高查询性能。完善的JSON支持
,M有SQL从5.7开始支持原生JSON数据的存储,MySQL对这一切功能做了优化,增加了聚合函数JSON_ARRAYAGG()
和JSON_OBJECTAGG()
,将参数聚合为JSON数组或对象,新增了行内操作符 ->>,是列路径运算符 -> 的增强,对JSON排序做了提升,并优化了JSON的更新操作。安全和账户管理
,MySQL 8中新增了caching_sha2_password授权插件、角色、密码历史记录和FIPS模式支持,这些特性提高了数据库安全的性能,使数据库管理员能更灵活的进行账户管理工作。InnoDB的变化
,InnoDB是MySQL默认的储存引擎,是事务型数据库的首选引擎,支持事务安全表(AICID),支持特性高了数据库的安全性和性能,并支持原子数据定义语言(DDL),提高了数据安全性,对事务提供更好的支持。数据字典
,在之前的MySQL版本中,字典数据都存储在元数据文件和非事务表中,从MySQL 8 开始新增了事务数据字典,在这个字典里存储着数据库对象信息,这些数据字典储存在内部事务表中。原子数据定义语句
MySQL 8开始支持原子数据定义语句(Automic DDL),即 原子DDL 。目前,只有InnoDB存储引擎支持原子DDL。原子数据定义语句(DDL)将与DDL操作相关的数据字典更新、存储引擎操作、二进制日志写入结合到一个单独的原子事务中,这使得即使服务器崩溃,事务也会提交或回滚。使用支持原子操作的存储引擎所创建的表,在执行DROP TABLE、CREATE TABLE、ALTER TABLE、
RENAME TABLE、TRUNCATE TABLE、CREATE TABLESPACE、DROP TABLESPACE等操作时,都支持原子操作,即事务要么完全操作成功,要么失败后回滚,不再进行部分提交。 对于从MySQL 5.7复制到MySQL 8版本中的语句,可以添加 IF EXISTS 或 IF NOT EXISTS 语句来避免发生错误。资源管理
MySQL 8开始支持创建和管理资源组,允许将服务器内运行的线程分配给特定的分组,以便线程根据组内可用资源执行。组属性能够控制组内资源,启用或限制组内资源消耗。数据库管理员能够根据不同的工作负载适当地更改这些属性。 目前,CPU时间是可控资源,由“虚拟CPU”这个概念来表示,此术语包含CPU的核心数,超线程,硬件线程等等。服务器在启动时确定可用的虚拟CPU数量。拥有对应权限的数据库管理员可以将这些CPU与资源组关联,并为资源组分配线程。 资源组组件为MySQL中的资源组管理提供了SQL接口。资源组的属性用于定义资源组。MySQL中存在两个默认组,系统组和用户组,默认的组不能被删除,其属性也不能被更改。对于用户自定义的组,资源组创建时可初始化所有的属性,除去名字和类型,其他属性都可在创建之后进行更改。 在一些平台下,或进行了某些MySQL的配
置时,资源管理的功能将受到限制,甚至不可用。例如,如果安装了线程池插件,或者使用的是macOS系统,资源管理将处于不可用状态。在FreeBSD和Solaris系统中,资源线程优先级将失效。在Linux系统中,只有配置了CAP_SYS_NICE属性,资源管理优先级才能发挥作用。字符集支持
MySQL 8中默认的字符集由 latin1 更改为 utf8mb4 ,并首次增加了日语所特定使用的集合,utf8mb4_ja_0900_as_cs。优化器增强
MySQL优化器开始支持隐藏索引和降序索引。隐藏索引不会被优化器使用,验证索引的必要性时不需要删除索引,先将索引隐藏,如果优化器性能无影响就可以真正地删除索引。降序索引允许优化器对多个列进行排序,并且允许排序顺序不一致。公用表表达式
公用表表达式(Common Table Expressions)简称为CTE,MySQL现在支持递归和非递归两种形式的CTE。CTE通过在SELECT语句或其他特定语句前 使用WITH语句对临时结果集
进行命名。
基础语法如下:
WITH cte_name (col_name1,col_name2 ...) AS (Subquery)
SELECT * FROM cte_name;
Subquery代表子查询,子查询前使用WITH语句将结果集命名为cte_name,在后续的查询中即可使用cte_name进行查询。正则表达式支持
MySQL在8.0.4以后的版本中采用支持Unicode的国际化组件库实现正则表达式操作,这种方式不仅能提供完全的Unicode支持,而且是多字节安全编码。MySQL增加了REGEXP_LIKE()、EGEXP_INSTR()、REGEXP_REPLACE()和 REGEXP_SUBSTR()等函数来提升性能。另外,regexp_stack_limit和regexp_time_limit 系统变量能够通过匹配引擎来控制资源消耗。内部临时表
TempTable存储引擎取代MEMORY存储引擎成为内部临时表的默认存储引擎
。TempTable存储引擎为VARCHAR和VARBINARY列提供高效存储。internal_tmp_mem_storage_engine会话变量定义了内部临时表的存储引擎,可选的值有两个,TempTable和MEMORY,其中TempTable为默认的存储引擎。temptable_max_ram系统配置项定义了TempTable存储引擎可使用的最大内存数量。日志记录
在MySQL 8中错误日志子系统由一系列MySQL组件构成。这些组件的构成由系统变量log_error_services来配置,能够实现日志事件的过滤和写入。
WITH cte_name (col_name1,col_name2 ...) AS (Subquery)SELECT * FROM cte_name;
备份锁
新的备份锁允许在线备份期间执行数据操作语句,同时阻止可能造成快照不一致的操作。新
备份锁由 LOCK INSTANCE FOR BACKUP 和 UNLOCK INSTANCE 语法提供支持,执行这些操作需要备份管理
员特权。增强的MySQL复制
MySQL 8复制支持对 JSON文档
进行部分更新的 二进制日志记录
,该记录 使用紧凑 的二进制格式
,从而节省记录完整JSON文档的空间。当使用基于语句的日志记录时,这种紧凑的日志记
录会自动完成,并且可以通过将新的binlog_row_value_options系统变量值设置为PARTIAL_JSON来启用。在MySQL 5.7版本上开发的应用程序如果使用了MySQL8.0 移除的特性,语句可能会失败,或者产生不同的执行结果。为了避免这些问题,对于使用了移除特性的应用,应当尽力修正避免使用这些特性,并尽可能使用替代方法。
查询缓存
查询缓存已被移除,删除的项有:
(1)语句
:FLUSH QUERY CACHE和RESET QUERYCACHE。
(2)系统变量
:query_cache_limit、query_cache_min_res_unitquery_cache_size、
query_cache_type、query_cache_wlock_invalidate。
(3)状态变量
:Qcache_free_blocks、Qcache_free_memory、Qcache_hits、Qcache_inserts、Qcache_lowmem_prunes、Qcache_not_cached、Qcache_queries_in_cache、Qcache_total_blocks。
(4)线程状态
:checking privileges on cachedquery、checking query cache for query、invalidating query cache entries、sending cached result toclient、storing result in query cache、waiting for query cache lock。加密相关
删除的加密相关的内容有:ENCODE()、DECODE()、ENCRYPT()、DES_ENCRYPT()和DES_DECRYPT()函数,配置项des-key-file,系统变量have_crypt,FLUSH语句的DES_KEY_FILE选项,HAVE_CRYPT CMake选项。 对于移除的ENCRYPT()函数,考虑使用SHA2()替代,对于其他移除的函数,使用AES_ENCRYPT()和AES_DECRYPT()替代。空间函数相关
在MySQL 5.7版本中,多个空间函数已被标记为过时。这些过时函数在MySQL 8中都已被移除,只保留了对应的ST_和MBR函数。\N和NULL
在SQL语句中,解析器不再将\N视为NULL,所以在SQL语句中应使用NULL代替\N。这项变化不会影响使用LOAD DATA INFILE或者SELECT…INTO OUTFILE操作文件的导入和导出。在这类操作中,NULL仍等同于\N。mysql_install_db
在MySQL分布中,已移除了mysql_install_db程序,数据字典初始化需要调用带着–initialize或者–initialize-insecure选项的mysqld来代替实现。另外,–bootstrap和INSTALL_SCRIPTDIRCMake也已被删除。通用分区处理程序
通用分区处理程序已从MySQL服务中被移除。为了实现给定表分区,表所使用的存储引擎需要自有的分区处理程序。 提供本地分区支持的MySQL存储引擎有两个,即InnoDB和NDB,而在MySQL 8中只支持InnoDB。系统和状态变量信息
在INFORMATION_SCHEMA数据库中,对系统和状态变量信息不再进行维护。GLOBAL_VARIABLES、SESSION_VARIABLES、GLOBAL_STATUS、SESSION_STATUS表都已被删除。另外,系统变量show_compatibility_56也已被删除。被删除的状态变量有Slave_heartbeat_period、Slave_last_heartbeat,Slave_received_heartbeats、Slave_retried_transactions、Slave_running。以上被删除的内容都可使用性能模式中对应的内容进行替代。mysql_plugin工具
mysql_plugin工具用来配置MySQL服务器插件,现已被删除,可使用–plugin-load或–plugin-load-add选项在服务器启动时加载插件或者在运行时使用INSTALL PLUGIN语句加载插件来替代该工具。MySQL从8.0版本开始支持窗口函数。窗口函数的作用类似于在查询中对数据进行分组,不同的是,分组操作会把分组的结果聚合成一条记录,而窗口函数是将结果置于每一条数据记录中。
静态窗口函数
的窗口大小是固定的,不会因为记录的不同而不同;动态窗口函数
的窗口大小会随着记录的不同而变化。特点
是可以分组,而且可以在分组内排序。另外,窗口函数不会因为分组而减少原表中的行数,这对我们在原表数据的基础上进行统计和排序非常有用。函数 OVER([PARTITION BY 字段名 ORDER BY 字段名 ASC|DESC])
或
函数 OVER 窗口名 … WINDOW 窗口名 AS ([PARTITION BY 字段名 ORDER BY 字段名 ASC|DESC])
WINDOW 窗口名 AS ...
语法解析,这部分语法是在末尾声明一个语句,可以在前面使用,在前面只需要通过窗口名即可调用。因而当一条语句的前半部分有重复的子查询语句的时候,就可以使用该语法,尤其在多表查询的时候。1.
ROW_NUMBER()函数
SELECT ROW_NUMBER() OVER(PARTITION BY category_id ORDER BY price DESC) AS row_num,
,在排序完后会在结果集第一列前插入一列 row_num 显示序号。 2.RANK()函数
3.DENSE_RANK()函数
1.
PERCENT_RANK()函数
等级值百分比
函数。按照如下方式进行计算。 2.CUME_DIST()函数
查询小于或等于某个值的比例
。1.
LAG(expr,n)函数
2.LEAD(expr,n)函数
1.
FIRST_VALUE(expr)函数
2.LAST_VALUE(expr)函数
举例:按照价格排序,查询第1个商品的价格信息。
1.
NTH_VALUE(expr,n)函数
2.NTILE(n)函数
临时结果集
,作用范围是当前语句
。语法结构:
WITH CTE名称
AS (子查询)
SELECT|DELETE|UPDATE 语句;
示例:
作用
是可以替代子查询,而且可以被多次引用。递归公用表表达式对查询有一个共同根节点的树形结构数据非常高效,可以轻松搞定其他查询方式难以处理的查询。语法格式:
WITH RECURSIVE
CTE名称 AS (子查询)
SELECT|DELETE|UPDATE 语句;
示例:
WITH RECURSIVE cte
AS
(
SELECT employee_id,last_name,manager_id,1 AS n FROM employees WHERE employee_id = 100
-- 种子查询,找到第一代领导
UNION ALL
SELECT a.employee_id,a.last_name,a.manager_id,n+1 FROM employees AS a JOIN cte
ON (a.manager_id = cte.employee_id) -- 递归查询,找出以递归公用表表达式的人为领导的人
)
SELECT employee_id,last_name FROM cte WHERE n >= 3;
关键字 | 描述 |
---|---|
NULL | 数据列可包含NULL值 |
NOT NULL | 数据列不允许包含NULL值 |
DEFAULT | 默认值 |
PRIMARY KEY | 主键 |
AUTO_INCREMENT | 自动递增,适用于整数类型 |
UNSIGNED | 无符号 |
CHARACTER SET ‘字符集’ | 指定一个字符集 |
扫码关注腾讯云开发者
领取腾讯云代金券
Copyright © 2013 - 2025 Tencent Cloud. All Rights Reserved. 腾讯云 版权所有
深圳市腾讯计算机系统有限公司 ICP备案/许可证号:粤B2-20090059 深公网安备号 44030502008569
腾讯云计算(北京)有限责任公司 京ICP证150476号 | 京ICP备11018762号 | 京公网安备号11010802020287
Copyright © 2013 - 2025 Tencent Cloud.
All Rights Reserved. 腾讯云 版权所有