1.函数
从函数定义的角度出发,我们可以将函数分成 内置函数 和 自定义函数 。在 SQL 语言中,同样也包括了 内置函数和自定义函数。内置函数是系统内置的通用函数,而自定义函数是我们根据自己的需要编写 的,本章及下一章讲解的是 SQL 的内置函数。
单行函数 操作数据对象 接受参数返回一个结果 只对一行进行变换 每行返回一个结果 可以嵌套 参数可以是一列或一个值
2.基本函数
SELECT ABS(-123),ABS(32),SIGN(-23),SIGN(43),PI(),CEIL(32.32),CEILING(-43.23),FLOOR(32.32), FLOOR(-43.23),MOD(12,5) FROM DUAL;
SELECT RAND(),RAND(),RAND(10),RAND(10),RAND(-1),RAND(-1) FROM DUAL;
SELECT ROUND(12.33),ROUND(12.343,2),ROUND(12.324,-1),TRUNCATE(12.66,1),TRUNCATE(12.66,-1) FROM DUAL;
RADIANS(x) 将角度转化为弧度,其中,参数x为角度值 DEGREES(x) 将弧度转化为角度,其中,参数x为弧度值
SELECT RADIANS(30),RADIANS(60),RADIANS(90),DEGREES(2*PI()),DEGREES(RADIANS(90)) FROM DUAL; #单行函数可以嵌套 select truncate(round(123.456,2),0) from dual;
3.字符串函数
ASCII(S) 返回字符串S中的第一个字符的ASCII码值 #注意是中文的逗号 SELECT ASCII('Abcdfsf'), FROM DUAL;
CHAR_LENGTH(s) 返回字符串s的字符数。作用与CHARACTER_LENGTH(s)相同 LENGTH(s) 返回字符串s的字节数,和字符集有关 #一个汉字三个字符 select ASCII('Abcdfsf'),CHAR_LENGTH('hello'),CHAR_LENGTH('我们'), LENGTH('hello'),LENGTH('我们') FROM DUAL;
CONCAT(s1,s2,......,sn) 连接s1,s2,......,sn为一个字符串 CONCAT_WS(x, s1,s2,......,sn) 同CONCAT(s1,s2,...)函数,但是每个字符串之间要加上x
#xxx worked for yyy select concat(emp.last_name,' worked for ',mgr.last_name) "details" from employees emp join employees mgr where emp.`manager_id` =mgr.`employee_id`;
INSERT(str, idx, len, replacestr) 将字符串str从第idx位置开始,len个字符长的子串替换为字符串replacestr REPLACE(str, a, b) 用字符串b替换字符串str中所有出现的字符串a #字符串的索引是从1开始的 select insert('helloworld',2,3,'aaaa'), replace('hello','lol','mmm') from dual;
UPPER(s) 或 UCASE(s) 将字符串s的所有字母转成大写字母 LOWER(s) 或LCASE(s) 将字符串s的所有字母转成小写字母 select upper('HeLLo'),lower('HeLLo') FROM dual;
LEFT(str,n) 返回字符串str最左边的n个字符 RIGHT(str,n) 返回字符串str最右边的n个字符
Lpad:右对齐 Rpad:左对齐 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)相同。未找到,返回0
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
4.日期和时间函数
4.1 获取日期、时间
select curdate(),CURRENT_DATE(),CURTIME(),NOW(),SYSDATE(),UTC_DATE(),UTC_TIME() from dual;
4.2 日期与时间戳的转换
毫秒级10位时间戳
4.3 获取月份、星期、星期数、天数等函数
4.4 时间和秒钟转换的函数
select TIME_TO_SEC(NOW())
from DUAL;
SELECT SEC_TO_TIME(78774);
select TIME_TO_SEC(NOW()) from DUAL; SELECT SEC_TO_TIME(78774);
4.5 计算日期和时间的函数
查询 7 天内的新增用户数有多少? SELECT COUNT(*) as num FROM new_user WHERE TO_DAYS(NOW())-TO_DAYS(regist_time)<=7
GET_FORMAT函数中date_type和format_type参数取值如下:
select DATE_FORMAT(NOW(),'%H:%i:%s');
SELECT STR_TO_DATE('1/23/2023','%m/%d/%Y') FROM DUAL;
SELECT STR_TO_DATE('20230121174706','%Y%m%d%H%i%s') FROM DUAL;
流程处理函数可以根据不同的条件,执行不同的处理流程,可以在SQL语句中实现不同的条件选择。
MySQL中的流程处理函数主要包括IF()、IFNULL()和CASE()函数。
-- 练习:查询部门号为 10,20, 30 的员工信息, 若部门号为 10, 则打印其工资的 1.1 倍, 20 号部门, 则打印其工资的 1.2 倍, 30 号部门打印其工资的 1.3 倍数, 其他部门打印其工资的1.4倍 select employee_id,last_name,department_id,salary, CASE department_id WHEN 10 THEN salary*1.1 wHEN 20 THEN salary*1.2 WHEN 30 THEN salary*1.3 ELSE salary*1.4 END "details" from employees;
5.加密与解密函数
单行函数练习题
1.显示系统时间(注:日期+时间)
SELECT NOW() FROM DUAL;
2.查询员工号,姓名,工资,以及工资提高百分之20%后的结果(new
salary)
SELECT employee_id, last_name, salary, salary * 1.2 "new salary" FROM employees;
3.将员工的姓名按首字母排序,并写出姓名的长度(length)
SELECT last_name, LENGTH(last_name) FROM employees ORDER BY last_name DESC;
4.查询员工id,last_name,salary,并作为一个列输出,别名为OUT_PUT
SELECT CONCAT(employee_id, ',' , last_name , ',', salary) OUT_PUT FROM employees;
5.查询公司各员工工作的年数、工作的天数,并按工作年数的降序排序。
SELECT DATEDIFF(SYSDATE(), hire_date) / 365 worked_years, DATEDIFF(SYSDATE(), hire_date) worked_days FROM employees ORDER BY worked_years DESC
6.查询员工姓名,hire_date , department_id,满足以下条件:雇用时间在
1997年之后,department_id 为80 或 90 或110, commission_pct不为空
SELECT last_name, hire_date, department_id FROM employees #WHERE hire_date >= '1997-01-01' #WHERE hire_date >= STR_TO_DATE('1997-01-01', '%Y-%m-%d') WHERE DATE_FORMAT(hire_date,'%Y') >= '1997' AND department_id IN (80, 90, 110) AND commission_pct IS NOT NULL
7.查询公司中入职超过10000天的员工姓名、入职时间
SELECT last_name,hire_date FROM employees #WHERE TO_DAYS(NOW()) - to_days(hire_date) > 10000; WHERE DATEDIFF(NOW(),hire_date) > 10000;
SELECT CONCAT(last_name, ' earns ', TRUNCATE(salary, 0) , ' monthly but wants ', TRUNCATE(salary * 3, 0)) "Dream Salary" FROM employees;
SELECT last_name Last_name, job_id Job_id, CASE job_id WHEN 'AD_PRES' THEN 'A' WHEN 'ST_MAN' THEN 'B' WHEN 'IT_PROG' THEN 'C' WHEN 'SA_REP' THEN 'D' WHEN 'ST_CLERK' THEN 'E' ELSE 'F' END "grade" FROM employees;