- 十、合计/统计函数
- 十一、字符串相关函数
- 11.1 CHARSET(str) 返回字串字符集
- 11.2 CONCAT (string2 [,… ]) 连接字符串
- 11.3 INSTR 返回 substring 在 string 中出现的位置
- 11.4 UCASE (string2 ) 转换成大写
- 11.5 LCASE (string2 ) 转换成小写
- 11.6 LENGTH (string ) string 长度[按照字节]
- 11.7 REPLACE 替换字符串内容
- 11.8 STRCMP (string1 ,string2 ) 逐字符比较两字串大小
- 11.9 SUBSTRING 截取字符串中的内容
- 11.10 LTRIM RTRIM TRIM 去除首尾空格
- 十二、数学相关函数
- 12.1 ABS(num) 绝对值
- 12.2 BIN (decimal_number )十进制转二进制
- 12.3 CEILING (number2 ) 向上取整
- 12.4 CONV(number2,from_base,to_base) 进制转换
- 12.5 FLOOR (number2 ) 向下取整
- 12.6 FORMAT (number,decimal_places ) 保留小数位数(四舍五入)
- 12.7 HEX (DecimalNumber ) 转十六进制
- 12.8 LEAST (number , number2 [,..]) 求最小值
- 12.9 MOD (numerator ,denominator ) 求余
- 12.10 RAND([seed]) RAND([seed]) 返回随机数 其范围为 0 ≤ v ≤ 1.0
- 十三、时间日期相关函数
- 十四、加密和系统函数
- 十五、流程控制函数
十、合计/统计函数
10.1 count 函数:返回行的总数

-- 演示 mysql 的统计函数的使用-- 统计一个班级共有多少学生?SELECT COUNT(*) FROM student;-- 统计数学成绩大于 90 的学生有多少个?SELECT COUNT(*) FROM studentWHERE math > 90-- 统计总分大于 250 的人数有多少?SELECT COUNT(*) FROM studentWHERE (math + english + chinese) > 250-- count(*) 和 count(列) 的区别-- 解释 :count(*) 返回满足条件的记录的行数-- count(列): 统计满足条件的某列有多少个,但是会排除 为 null 的情况CREATE TABLE t15 (`name` VARCHAR(20));INSERT INTO t15 VALUES('tom');INSERT INTO t15 VALUES('jack');INSERT INTO t15 VALUES('mary');INSERT INTO t15 VALUES(NULL);SELECT * FROM t15;SELECT COUNT(*) FROM t15; -- 4SELECT COUNT(`name`) FROM t15;-- 3
10.2 sum 函数:返回满足 where 条件的列和
一般用在数值列
-- 演示 sum 函数的使用-- 统计一个班级数学总成绩?SELECT SUM(math) FROM student;-- 统计一个班级语文、英语、数学各科的总成绩SELECT SUM(math) AS math_total_score,SUM(english),SUM(chinese) FROM student;-- 统计一个班级语文、英语、数学的成绩总和SELECT SUM(math + english + chinese) FROM student;-- 统计一个班级语文成绩平均分SELECT SUM(chinese)/ COUNT(*) FROM student;SELECT SUM(`name`) FROM student;
10.3 avg 函数:返回满足 where 条件的一列的平均值

-- 演示 avg 的使用-- 练习:-- 求一个班级数学平均分?SELECT AVG(math) FROM student;-- 求一个班级总分平均分SELECT AVG(math + english + chinese) FROM student;
10.4 max/min 函数:返回满足 where 条件的一列的最大/最小值

-- 演示 max 和 min 的使用-- 求班级最高分和最低分(数值范围在统计中特别有用)SELECT MAX(math + english + chinese), MIN(math + english + chinese)FROM student;-- 求出班级数学最高分和最低分SELECT MAX(math) AS math_high_socre, MIN(math) AS math_low_socreFROM student;
10.5 使用 group by 子句对列进行分组

10.6 使用 having 子句对分组后的结果进行过滤

-- 相关表的创建-- 部门表CREATE TABLE dept( /*部门表*/deptno MEDIUMINT UNSIGNED NOT NULL DEFAULT 0,dname VARCHAR(20) NOT NULL DEFAULT "",loc VARCHAR(13) NOT NULL DEFAULT "");INSERT INTO dept VALUES(10, 'ACCOUNTING', 'NEW YORK'),(20, 'RESEARCH', 'DALLAS'),(30, 'SALES', 'CHICAGO'),(40, 'OPERATIONS', 'BOSTON');SELECT * FROM dept;-- 员工表CREATE TABLE emp(empno MEDIUMINT UNSIGNED NOT NULL DEFAULT 0, /*编号*/ename VARCHAR(20) NOT NULL DEFAULT "", /*名字*/job VARCHAR(9) NOT NULL DEFAULT "",/*工作*/mgr MEDIUMINT UNSIGNED ,/*上级编号*/hiredate DATE NOT NULL,/*入职时间*/sal DECIMAL(7,2) NOT NULL,/*薪水*/comm DECIMAL(7,2) ,/*红利 奖金*/deptno MEDIUMINT UNSIGNED NOT NULL DEFAULT 0 /*部门编号*/);-- 添加测试数据INSERT INTO emp VALUES(7369, 'SMITH', 'CLERK', 7902, '1990-12-17', 800.00,NULL , 20),(7499, 'ALLEN', 'SALESMAN', 7698, '1991-2-20', 1600.00, 300.00, 30),(7521, 'WARD', 'SALESMAN', 7698, '1991-2-22', 1250.00, 500.00, 30),(7566, 'JONES', 'MANAGER', 7839, '1991-4-2', 2975.00,NULL,20),(7654, 'MARTIN', 'SALESMAN', 7698, '1991-9-28',1250.00,1400.00,30),(7698, 'BLAKE','MANAGER', 7839,'1991-5-1', 2850.00,NULL,30),(7782, 'CLARK','MANAGER', 7839, '1991-6-9',2450.00,NULL,10),(7788, 'SCOTT','ANALYST',7566, '1997-4-19',3000.00,NULL,20),(7839, 'KING','PRESIDENT',NULL,'1991-11-17',5000.00,NULL,10),(7844, 'TURNER', 'SALESMAN',7698, '1991-9-8', 1500.00, NULL,30),(7900, 'JAMES','CLERK',7698, '1991-12-3',950.00,NULL,30),(7902, 'FORD', 'ANALYST',7566,'1991-12-3',3000.00, NULL,20),(7934,'MILLER','CLERK',7782,'1992-1-23', 1300.00, NULL,10);SELECT * FROM emp;-- 工资级别#工资级别表CREATE TABLE salgrade(grade MEDIUMINT UNSIGNED NOT NULL DEFAULT 0, /*工资级别*/losal DECIMAL(17,2) NOT NULL, /* 该级别的最低工资 */hisal DECIMAL(17,2) NOT NULL /* 该级别的最高工资*/);INSERT INTO salgrade VALUES (1,700,1200);INSERT INTO salgrade VALUES (2,1201,1400);INSERT INTO salgrade VALUES (3,1401,2000);INSERT INTO salgrade VALUES (4,2001,3000);INSERT INTO salgrade VALUES (5,3001,9999);SELECT * FROM salgrade;SELECT * FROM dept;SELECT * FROM emp;
# 演示 group by + havingGROUP by 用于对查询的结果分组统计-- having 子句用于限制分组显示结果.-- ?如何显示每个部门的平均工资和最高工资-- 老韩分析: avg(sal) max(sal)-- 按照部分来分组查询SELECT AVG(sal), MAX(sal) , deptnoFROM emp GROUP BY deptno;-- 使用数学方法,对小数点进行处理SELECT FORMAT(AVG(sal),2), MAX(sal) , deptnoFROM emp GROUP BY deptno;-- ?显示每个部门的每种岗位的平均工资和最低工资-- 老师分析 1. 显示每个部门的平均工资和最低工资-- 2. 显示每个部门的每种岗位的平均工资和最低工资SELECT AVG(sal), MIN(sal) , deptno, jobFROM emp GROUP BY deptno, job;-- ?显示平均工资低于 2000 的部门号和它的平均工资 // 别名-- 老师分析 [写 sql 语句的思路是化繁为简,各个击破]-- 1. 显示各个部门的平均工资和部门号-- 2. 在 1 的结果基础上,进行过滤,保留 AVG(sal) < 2000-- 3. 使用别名进行过滤SELECT AVG(sal), deptnoFROM emp GROUP BY deptnoHAVING AVG(sal) < 2000;-- 使用别名SELECT AVG(sal) AS avg_sal, deptnoFROM emp GROUP BY deptnoHAVING avg_sal < 2000;
十一、字符串相关函数
11.1 CHARSET(str) 返回字串字符集
-- 演示字符串相关函数的使用 , 使用 emp 表来演示-- CHARSET(str) 返回字串字符集SELECT CHARSET(ename) FROM emp;
11.2 CONCAT (string2 [,… ]) 连接字符串
-- CONCAT (string2 [,... ]) 连接字串, 将多个列拼接成一列SELECT CONCAT(ename, ' 工作是 ', job) FROM emp;
11.3 INSTR 返回 substring 在 string 中出现的位置
-- INSTR (string ,substring ) 返回 substring 在 string 中出现的位置,没有返回 0-- dual 亚元表, 系统表 可以作为测试表使用SELECT INSTR('hanshunping', 'ping') FROM DUAL;
11.4 UCASE (string2 ) 转换成大写
-- UCASE (string2 ) 转换成大写SELECT UCASE(ename) FROM emp;
11.5 LCASE (string2 ) 转换成小写
-- LCASE (string2 ) 转换成小写SELECT LCASE(ename) FROM emp;-- LEFT (string2 ,length )从 string2 中的左边起取 length 个字符-- RIGHT (string2 ,length ) 从 string2 中的右边起取 length 个字符SELECT LEFT(ename, 2) FROM emp;
11.6 LENGTH (string ) string 长度[按照字节]
-- LENGTH (string ) string 长度[按照字节]SELECT LENGTH(ename) FROM emp;
11.7 REPLACE 替换字符串内容
-- REPLACE (str ,search_str ,replace_str )-- 在 str 中用 replace_str 替换 search_str-- 如果是 manager 就替换成 经理SELECT ename, REPLACE(job,'MANAGER', '经理') FROM emp;
11.8 STRCMP (string1 ,string2 ) 逐字符比较两字串大小
-- STRCMP (string1 ,string2 ) 逐字符比较两字串大小SELECT STRCMP('hsp', 'hsp') FROM DUAL;
11.9 SUBSTRING 截取字符串中的内容
-- SUBSTRING (str , position [,length ])-- 从 str 的 position 开始【从 1 开始计算】,取 length 个字符-- 从 ename 列的第一个位置开始取出 2 个字符SELECT SUBSTRING(ename, 1, 2) FROM emp;
11.10 LTRIM RTRIM TRIM 去除首尾空格
-- LTRIM (string2 ) RTRIM (string2 ) TRIM(string)-- 去除前端空格或后端空格SELECT LTRIM(' 韩顺平教育') FROM DUAL;SELECT RTRIM('韩顺平教育 ') FROM DUAL;SELECT TRIM(' 韩顺平教育 ') FROM DUAL;
练习:以首字母小写的方式显示所有员工 emp 表的姓名
-- 思路:先取出 ename 的第一个字符,转成小写的-- 把他和后面的字符串进行拼接输出即可SELECT CONCAT(LCASE(SUBSTRING(ename,1,1)), SUBSTRING(ename,2)) AS new_nameFROM emp;SELECT CONCAT(LCASE(LEFT(ename,1)), SUBSTRING(ename,2)) AS new_nameFROM emp;
十二、数学相关函数
12.1 ABS(num) 绝对值
-- ABS(num) 绝对值SELECT ABS(-10) FROM DUAL;
12.2 BIN (decimal_number )十进制转二进制
-- BIN (decimal_number )十进制转二进制SELECT BIN(10) FROM DUAL;
12.3 CEILING (number2 ) 向上取整
-- CEILING (number2 ) 向上取整, 得到比 num2 大的最小整数SELECT CEILING(-1.1) FROM DUAL;
12.4 CONV(number2,from_base,to_base) 进制转换
-- CONV(number2,from_base,to_base) 进制转换-- 下面的含义是 8 是十进制的 8, 转成 2 进制输出SELECT CONV(8, 10, 2) FROM DUAL;-- 下面的含义是 8 是 16 进制的 8, 转成 2 进制输出SELECT CONV(16, 16, 10) FROM DUAL;
12.5 FLOOR (number2 ) 向下取整
-- FLOOR (number2 ) 向下取整,得到比 num2 小的最大整数SELECT FLOOR(-1.1) FROM DUAL;
12.6 FORMAT (number,decimal_places ) 保留小数位数(四舍五入)
-- FORMAT (number,decimal_places ) 保留小数位数(四舍五入)SELECT FORMAT(78.125458,2) FROM DUAL;
12.7 HEX (DecimalNumber ) 转十六进制
-- HEX (DecimalNumber ) 转十六进制
12.8 LEAST (number , number2 [,..]) 求最小值
-- LEAST (number , number2 [,..]) 求最小值SELECT LEAST(0,1, -10, 4) FROM DUAL;
12.9 MOD (numerator ,denominator ) 求余
-- MOD (numerator ,denominator ) 求余SELECT MOD(10, 3) FROM DUAL;
12.10 RAND([seed]) RAND([seed]) 返回随机数 其范围为 0 ≤ v ≤ 1.0
-- RAND([seed]) RAND([seed]) 返回随机数 其范围为 0 ≤ v ≤ 1.0-- 老韩说明-- 1. 如果使用 rand() 每次返回不同的随机数 ,在 0 ≤ v ≤ 1.0-- 2. 如果使用 rand(seed) 返回随机数, 范围 0 ≤ v ≤ 1.0, 如果 seed 不变,-- 该随机数也不变了SELECT RAND() FROM DUAL;SELECT FLOOR(RAND(3)*10) FROM DUAL; -- 3就是一个种子数,生成种子后,同一个种子的随机数不变
十三、时间日期相关函数

- 日期时间相关函数-- CURRENT_DATE ( ) 当前日期SELECT CURRENT_DATE() FROM DUAL;-- CURRENT_TIME ( )当前时间SELECT CURRENT_TIME() FROM DUAL;-- CURRENT_TIMESTAMP ( ) 当前时间戳SELECT CURRENT_TIMESTAMP() FROM DUAL;-- 创建测试表 信息表CREATE TABLE mes(id INT ,content VARCHAR(30),send_time DATETIME);-- 添加一条记录INSERT INTO mesVALUES(1, '北京新闻', CURRENT_TIMESTAMP());INSERT INTO mes VALUES(2, '上海新闻', NOW());INSERT INTO mes VALUES(3, '广州新闻', NOW());SELECT * FROM mes;SELECT NOW() FROM DUAL;-- 上应用实例-- 显示所有新闻信息,发布日期只显示 日期,不用显示时间.SELECT id, content, DATE(send_time)FROM mes;-- 请查询在 10 分钟内发布的新闻, 思路一定要梳理一下.SELECT *FROM mesWHERE DATE_ADD(send_time, INTERVAL 10 MINUTE) >= NOW();SELECT *FROM mesWHERE send_time >= DATE_SUB(NOW(), INTERVAL 10 MINUTE);-- 请在 mysql 的 sql 语句中求出 2011-11-11 和 1990-1-1 相差多少天SELECT DATEDIFF('2011-11-11', '1990-01-01') FROM DUAL;-- 请用 mysql 的 sql 语句求出你活了多少天? [练习] 1986-11-11 出生SELECT DATEDIFF(NOW(), '1986-11-11') FROM DUAL;-- 如果你能活 80 岁,求出你还能活多少天.[练习] 1986-11-11 出生-- 先求出活 80 岁 时, 是什么日期 X-- 然后在使用 datediff(x, now()); 1986-11-11->datetime-- INTERVAL 80 YEAR : YEAR 可以是 年月日,时分秒-- '1986-11-11' 可以 date,datetime timestampSELECT DATEDIFF(DATE_ADD('1986-11-11', INTERVAL 80 YEAR), NOW())FROM DUAL;SELECT TIMEDIFF('10:11:11', '06:10:10') FROM DUAL;-- YEAR|Month|DAY| DATE (datetime )SELECT YEAR(NOW()) FROM DUAL;SELECT MONTH(NOW()) FROM DUAL;SELECT DAY(NOW()) FROM DUAL;SELECT MONTH('2013-11-10') FROM DUAL;-- unix_timestamp() : 返回的是 1970-1-1 到现在的秒数SELECT UNIX_TIMESTAMP() FROM DUAL;-- FROM_UNIXTIME() : 可以把一个 unix_timestamp 秒数[时间戳],转成指定格式的日期-- %Y-%m-%d 格式是规定好的,表示年月日-- 意义:在开发中,可以存放一个整数,然后表示时间,通过 FROM_UNIXTIME 转换--SELECT FROM_UNIXTIME(1618483484, '%Y-%m-%d') FROM DUAL;SELECT FROM_UNIXTIME(1618483100, '%Y-%m-%d %H:%i:%s') FROM DUAL;SELECT * FROM mysql.user \G
十四、加密和系统函数

-- 演示加密函数和系统函数-- USER() 查询用户-- 可以查看登录到 mysql 的有哪些用户,以及登录的 IPSELECT USER() FROM DUAL; -- 用户@IP 地址-- DATABASE()查询当前使用数据库名称SELECT DATABASE();-- MD5(str) 为字符串算出一个 MD5 32 的字符串,常用(用户密码)加密-- root 密码是 hsp -> 加密 md5 -> 在数据库中存放的是加密后的密码SELECT MD5('hsp') FROM DUAL;SELECT LENGTH(MD5('hsp')) FROM DUAL;-- 演示用户表,存放密码时,是 md5CREATE TABLE hsp_user(id INT ,`name` VARCHAR(32) NOT NULL DEFAULT '',pwd CHAR(32) NOT NULL DEFAULT '');INSERT INTO hsp_userVALUES(100, '韩顺平', MD5('hsp'));SELECT * FROM hsp_user; -- csdnSELECT * FROM hsp_user -- SQL 注入问题WHERE `name`='韩顺平' AND pwd = MD5('hsp')-- PASSWORD(str) -- 加密函数, MySQL 数据库的用户密码就是 PASSWORD 函数加密SELECT PASSWORD('hsp') FROM DUAL; -- 数据库的 *81220D972A52D4C51BB1C37518A2613706220DAC-- select * from mysql.user \G 从原文密码 str 计算并返回密码字符串-- 通常用于对 mysql 数据库的用户密码加密-- mysql.user 表示 数据库.表SELECT * FROM mysql.user
十五、流程控制函数

# 演示流程控制语句# IF(expr1,expr2,expr3) 如果 expr1 为 True ,则返回 expr2 否则返回 expr3SELECT IF(TRUE, '北京', '上海') FROM DUAL;# IFNULL(expr1,expr2) 如果 expr1 不为空 NULL,则返回 expr1,否则返回 expr2SELECT IFNULL( NULL, '韩顺平教育') FROM DUAL;# SELECT CASE WHEN expr1 THEN expr2 WHEN expr3 THEN expr4 ELSE expr5 END; [类似多重分支.]# 如果 expr1 为 TRUE,则返回 expr2,如果 expr3 为 t, 返回 expr4, 否则返回 expr5SELECT CASEWHEN TRUE THEN 'jack' -- jackWHEN FALSE THEN 'tom'ELSE 'mary' END-- 1. 查询 emp 表, 如果 comm 是 null , 则显示 0.0-- 老师说明,判断是否为 null 要使用 is null, 判断不为空 使用 is notSELECT ename, IF(comm IS NULL , 0.0, comm)FROM emp;SELECT ename, IFNULL(comm, 0.0)FROM emp;-- 2. 如果 emp 表的 job 是 CLERK 则显示 职员, 如果是 MANAGER 则显示经理-- 如果是 SALESMAN 则显示 销售人员,其它正常显示SELECT ename, (SELECT CASEWHEN job = 'CLERK' THEN '职员'WHEN job = 'MANAGER' THEN '经理'WHEN job = 'SALESMAN' THEN '销售人员'ELSE job END) AS 'job'FROM emp;SELECT * FROM emp;SELECT * FROM dept;SELECT * FROM salgrade;
学习参考(致谢):
- B站 @程序员鱼皮 Java学习一条龙
- B站 @韩顺平 零基础30天学会Java
