导图社区 零基础也能学的:MySQL基础学习(一)
这张思维导图整理了:MySQL入门、查询语言基础(包含理论、练习题)、MySQL安装几个部分的基础知识。MySQL是一门偏向实际应用的课程。和其他语言一样,光靠看书和听课肯定不够,主要还是得靠练习。
编辑于2021-04-23 11:44:46mysql
DQL--数据查询语言
基础查询
1、查询单个或多个字段 select 列名1,列名2 from 表名select Age,Name from TB_students
2、查询常量值 select 100; select'JACK';
3、查询表达式 select 100%98;select 100+98;
4、查询函数 select version();
5、起别名,便于理解AS select 100%98 as 结果; select last_name AS 姓,first_name AS 名 from employees;
6、去重 distinct select distinct 字段名 from 表名;
7、+
null+任何值都为null
ifnull函数,判断字段或表达式是否为null,为null返回固定值,否则返回原本值 ifnull(原值,0)
#2. 查询员工号为 176 的员工的姓名和部门号和年薪 SELECT employee_id, last_name, department_id, SALARY * 12+ salary * 12 * 'ifnull(commission_pct,0)' AS 年薪 FROM employees WHERE employee_id = 100;
concat拼接查询出的值
SELECT CONCAT(`first_name`,',',`last_name`,',',`job_id`,',',IFNULL(commission_pct,0)) AS out_put FROM employees;
条件查询
1、简单条件运算符 > < = != <>
2、逻辑运算符 && || !等同于and or not
3、模糊查询
1、where:select 列名1,列名2 from 表名 where 列名3 运算符 值
1)查询姓氏为张的学生信息 SELECT ID,Name FROM Students WHERE Name LIKE '张%' 2)查询名字最后一个为“生”的同学 SELECT ID,Name FROM Students WHERE Name LIKE '%生' 3)查询名字中含有“生”的学生信息 SELECT ID,Name FROM Students WHERE Name LIKE '%生%' 4)查询姓名为两个字,且姓张学生信息 SELECT ID,Name FROM Students WHERE Name LIKE '张_' 5)查询姓氏为张、李的学生信息 这个可以使用or关键字,但是使用通配符更简单高效 SELECT ID,Name FROM Students WHERE Name LIKE '[张李]%' 6)查询姓氏非张、李的学生信息 这个也可以使用NOT LIKE 来实现,用下面方法更好。 SELECT ID,Name FROM Students WHERE Name LIKE '[^张李]%' 或者: SELECT ID,Name FROM Students WHERE Name LIKE '[!张李]%'
2、and在where语句中把多个条件结合起来。表示和的意思,多个条件都成立
1)查询年龄大于18且姓张的学生信息 SELECT ID,Name FROM Students WHERE Age>18 AND Name LIKE '张%'
3、IN 操作符允许我们在 WHERE 子句中规定多个值。表示:在哪些值当中。
1)查询年龄是18、19、20的学生信息 SELECT ID,Name FROM Students WHERE Age IN (18,19,20)
4、or在where语句中把两个或多个条件结合起来。或关系,多个条件,一个成立即可
select ID,Name,Age from TB_students where name like '张%' or name like '王%' or Age=18 1)查询姓氏为张、李的学生信息 SELECT ID,Name FROM Students WHERE Name LIKE '张%' OR Name LIKE '李%'
5、like
1、通配符% _
%代表任意多个字符,包含0字符
_任意单字符
2、反义符 \ ,在特殊符号之前,使符号变为普通字符
6、between and 等价于>= and <=,包含临界值,注意临界值顺序---小数值在前
7、is null
练习题
#1. 查询工资大于 12000 的员工姓名和工资 SELECT last_name,salary FROM employees WHERE salary>12000; #2. 查询员工号为 176 的员工的姓名和部门号和年薪 SELECT employee_id, last_name, department_id, SALARY * 12+ salary * 12 * 'is null(commission_pct,0)' AS 年薪 FROM employees WHERE employee_id = 176; #年薪=工资+奖金,奖金率为null时,取值为0。否则运算结果为null #3. 选择工资不在 5000 到 12000 的员工的姓名和工资 SELECT last_name, salary FROM employees WHERE salary < 5000 OR salary > 12000; #4. 选择在 20 或 50 号部门工作的员工姓名和部门号 SELECT last_name, department_id FROM employees WHERE department_id BETWEEN 20 AND 50; #5. 选择公司中没有管理者的员工姓名及 job_id SELECT last_name, job_id FROM employees WHERE manager_id IS NULL; #6. 选择公司中有奖金的员工姓名,工资和奖金级别 SELECT last_name, salary FROM employees WHERE commission_pct IS NOT NULL; #7. 选择员工姓名的第三个字母是 a 的员工姓名 SELECT last_name FROM employees WHERE last_name LIKE '__a%'; #8. 选择姓名中有字母 a 和 e 的员工姓名 SELECT * FROM employees WHERE first_name LIKE '%a%e%' OR first_name LIKE '%e%a%'; #9. 显示出表 employees 表中 first_name 以 'e' 结尾的员工信息 SELECT * FROM employees WHERE first_name LIKE ‘%-’; #10. 显示出表 employees 部门编号在 80-100 之间 的姓名、职位 SELECT last_name, job_id FROM employees WHERE department_id BETWEEN 80 AND 100; #11. 显示出表 employees 的 manager_id 是 100,101,110 的员工姓名、职位 SELECT last_name,job_id,manager_id FROM employees WHERE manager_id IN (100,101,110);
4、排序查询
1、语法 sellect 查询列表 from 表名【where 筛选条件】order by排序的字段或表达式
2、ASC --升序 DESC --降序
3、order by可以接字段、表达式、别名、函数、多个字段(,连接)
练习题
#1、按单个字段排序 SELECT * FROM employees ORDER BY salary DESC; #2、添加筛选条件再排序 #案例:查询部门编号>=90的员工信息,并按员工编号降序 SELECT * FROM employees WHERE department_id>=90 ORDER BY employee_id DESC; #3、按表达式排序 #案例:查询员工信息 按年薪降序 SELECT *,salary*12*(1+IFNULL(commission_pct,0)) FROM employees ORDER BY salary*12*(1+IFNULL(commission_pct,0)) DESC; #4、按别名排序 #案例:查询员工信息 按年薪升序 SELECT *,salary*12*(1+IFNULL(commission_pct,0)) 年薪 FROM employees ORDER BY 年薪 ASC; #5、按函数排序 #案例:查询员工名,并且按名字的长度降序 SELECT LENGTH(last_name),last_name FROM employees ORDER BY LENGTH(last_name) DESC; #6、按多个字段排序 #案例:查询员工信息,要求先按工资降序,再按employee_id升序 SELECT * FROM employees ORDER BY salary DESC,employee_id ASC; 查询有相中包含e的员工信息,并按邮箱字节数降序,再按部门号升序 SELECT * FROM employees WHERE email LIKE '%e%' ORDER BY LENGTH(email) DESC,department_id ASC;
5、函数
单行函数
1、字符函数
1、length 获取参数值的字节个数SELECT LENGTH('john'); SELECT LENGTH('张三丰hahaha'); 中文单个汉字=3字节
2、concat 拼接字符串 SELECT CONCAT(last_name,'_',first_name) 姓名 FROM employees;
3、upper、lower SELECT UPPER('john'); SELECT LOWER('joHn'); #示例:将姓变大写,名变小写,然后拼接 SELECT CONCAT(UPPER(last_name),LOWER(first_name)) 姓名 FROM employees;
4、substr、substring 注意:索引从1开始 #截取从指定索引处后面所有字符 SELECT SUBSTR('李莫愁爱上了陆展元',7) out_put; 截取从指定索引处指定字符长度的字符 SELECT SUBSTR('李莫愁爱上了陆展元',1,3) out_put; 姓名中首字符大写,其他字符小写然后用_拼接,显示出来 SELECT CONCAT(UPPER(SUBSTR(last_name,1,1)),'_',LOWER(SUBSTR(last_name,2))) out_put FROM employees;
5、instr 返回子串第一次出现的索引,如果找不到返回0 SELECT INSTR('杨不殷六侠悔爱上了殷六侠','殷八侠') AS out_put;
6、trim 前后删除字符 SELECT LENGTH(TRIM(' 张翠山 ')) AS out_put; 删除前后空格 SELECT TRIM('aa' FROM 'aaaaaaaaa张aaaaaaaaaaaa翠山aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa') AS out_put; 删除前后aa
7、lpad 用指定的字符实现左填充指定长度 SELECT LPAD('殷素素',2,'*') AS out_put;
8、rpad 用指定的字符实现右填充指定长度 SELECT RPAD('殷素素',12,'ab') AS out_put;
9、replace 替换 SELECT REPLACE('张无忌爱上了周芷若','周芷若','赵敏') AS out_put;
2、数学函数
1、round 四舍五入 select round(1.56); select (1.568,2),保留两位小数
2、ceil 向上取整,返回>=该参数的最小整数 select ceil(1.005);
3、floor 向下取整
3、日期函数
1、now返回当前系统时间 select now();--包含年月份时分秒
2、curdate--不包含时间 select curdate();
3、curtime--不包含年月日 select crutime();
4、获取指定的部分,年、月、日、小时、分钟、秒 year、month\monthname、day、hour、minute、second
SELECT YEAR(NOW()) 年; SELECT YEAR('1998-1-1') 年; SELECT YEAR(hiredate) 年 FROM employees; SELECT MONTH(NOW()) 月; SELECT MONTHNAME(NOW()) 月;
5、str_to_date 将字符通过指定的格式转换成日期 SELECT STR_TO_DATE('1998-3-2','%Y-%c-%d') AS out_put;
6、date_format 将日期转换成字符 SELECT DATE_FORMAT(NOW(),'%y年%m月%d日') AS out_put; #查询有奖金的员工名和入职日期(xx月/xx日 xx年) SELECT last_name,DATE_FORMAT(hiredate,'%m月/%d日 %y年') 入职日期 FROM employees WHERE commission_pct IS NOT NULL;
标注

7、datediff 计算两个时间的差值
4、其他函数
select version();select database(); select user();
5、流程控制函数
1、if 函数--if else的效果SELECT IF(10<5,'大','小'); SELECT last_name,commission_pct,IF(commission_pct IS NULL,'没奖金,呵呵','有奖金,嘻嘻') 备注 FROM employees;
2、case函数
1、case 要判断的字段或表达式 when 常量1 then 要显示的值1或语句1; when 常量2 then 要显示的值2或语句2; ... else 要显示的值n或语句n; end
案例:查询员工的工资,要求 部门号=30,显示的工资为1.1倍 部门号=40,显示的工资为1.2倍 部门号=50,显示的工资为1.3倍 其他部门,显示的工资为原工资 SELECT salary 原始工资,department_id, CASE department_id WHEN 30 THEN salary*1.1 WHEN 40 THEN salary*1.2 WHEN 50 THEN salary*1.3 ELSE salary END AS 新工资 FROM employees;
2、case when 条件1 then 要显示的值1或语句1 when 条件2 then 要显示的值2或语句2 。。。 else 要显示的值n或语句n end
SELECT last_name,salary, CASE WHEN salary>20000 THEN 'A级别' WHEN salary>15000 THEN 'B级别' WHEN salary>10000 THEN 'C级别' ELSE 'D级别' END AS 新工资 FROM employees; CASE 简单表达式,它通过将表达式与一组简单的表达式进行比较来确定结果。 CASE 搜索表达式,它通过计算一组布尔表达式来确定结果。 简单表达式语法: CASE input_expression WHEN when_expression THEN result_expression [ ...n ] [ ELSE else_result_expression ] END 搜索式语法: CASE WHEN Boolean_expression THEN result_expression [ ...n ] [ ELSE else_result_expression ] END 1)查询学习信息,如果Sex为0则显示为男,如果为1显示为女,其他显示为其他。 SELECT ID, Name, CASE Sex WHEN '0' THEN '男' WHEN '1' THEN '女' ELSE '其他' END AS Sex FROM Students 2)查询学生信息,根据年龄统计是否成年,大于等于18为成年,小于18为未成年 SELECT ID, Name, CASE WHEN Age>=18 THEN '成年' ELSE '未成年'END AS 是否成年 FROM Students 3)统计成年未成年学生的个数 要求结果 成年 未成年 23 6 SQL语句 SELECT SUM(CASE WHEN Age>=18 THEN 1 ELSE 0 END) AS '成年',SUM(CASE WHEN Age<18 THEN 1 ELSE 0 END) AS '未成年' FROM Students 4)行列转换。统计男女生中未成年、成年的人数 结果如下: 性别 未成年 成年 男 3 13 女 2 18 SQL语句: SELECT CASE WHEN Sex=0 THEN '男' ELSE '女' END AS '性别', SUM(CASE WHEN Age<18 THEN 1 ELSE 0 END) AS '未成年', SUM(CASE WHEN Age>=18 THEN 1 ELSE 0 END) AS '成年' FROM Students GROUP BY Sex
分组函数
SUM查询合计值
1)查询ID为1001的学生的各科总成绩 SC即为学生的成绩表,字段:StudentID,CourseID,Score. SELECT SUM(Score) AS TotalScore FROM SC WHERE StudentID='1001'
AVG列平均值
1)查询学生的平均年龄 SELECT AVG(Age) AS AgeAverage FROM Students 2)求课程ID为C001的平均成绩 SELECT AVG(Score) FROM SC WHERE CourseID='C001'
COUNT()函数返回匹配指定条件的行数
1)查询学生总数 SELECT COUNT(ID) FROM Students 2)查询学生年龄分布的总数 SELECT COUNT(DISTINCT Age) FROM Students 3)查询男生总数 SELECT COUNT(ID) FROM Students WHERE Sex='男' 4)查询男女生各有多少人 SELECT Sex,COUNT(ID) FROM Students GROUP BY Sex
max、min、
HAVING WHERE关键字无法与合计函数一起使用
语法: SELECT column_name, aggregate_function(column_name) FROM table_name WHERE column_name operator value GROUP BY column_name HAVING aggregate_function(column_name) operator value 1)查询平均成绩大等于于60的学生ID及平均成绩 SELECT StudentID,AVG(Score) FROM SC GROUP BY StudentID HAVING AVG(Score)>=60 2)还是用HAVING的SQL语句中,可以有普通的WHERE条件 查询平均成绩大于等于60,且学生ID等于1的学生的ID及平均成绩。 SELECT StudentID,AVG(Score) FROM SC WHERE StudentID='1' GROUP BY StudentID HAVING AVG(Score)>=60 3)查询总成绩在600分以上(包括600)的学生ID SELECT StudentID FROM SC GROUP BY StudentID HAVING SUM(Score)>=600
GROUP BY 结合合计函数,更具一个或多个列对结果集进行分组
查询男女生分布,上面已经给了答案。 SELECT Sex,COUNT(ID) FROM Students GROUP BY Sex 2) 查询学生的城市分布情况 SELECT City,COUNT(ID) FROM Students GROUP BY City 3)学生的平均成绩,查询结果包括:学生ID,平均成绩 SELECT StudentID,AVG(Score) FROM SC GROUP BY StudentID 4)删除学生信息中重复记录 根据列进行分组,如果全部列相同才定义为重复,则就需要GROUP BY所有字段。否则可按指定字段进行处理。 DELETE FROM Students WHERE ID NOT IN (SELECT MAX(ID) FROM Students GROUP BY ID,Name,Age,Sex,City,MajorID)
TOP 返回记录的数目
1)查询年龄最大的三名学生信息 SELECT TOP 3 ID,Name FROM Students ORDER BY Age DESC 2)还是上一道题,如果有相同年龄的如何处理呢? SELECT ID,Name,Age FROM Students WHERE Age IN (SELECT TOP 3 Age FROM Students)
分组查询
1、语法 select 查询列表 from 表 【where 筛选条件】 group by 分组的字段 【order by 排序的字段】;
2、特点 ①和分组函数一同查询的字段必须是group by后出现的字段 ②筛选分为两类:分组前筛选和分组后筛选

3、练习题
问题1:分组函数做筛选能不能放在where后面 答:不能 问题2:where——group by——having 一般来讲,能用分组前筛选的,尽量使用分组前筛选,提高效率 3、分组可以按单个字段也可以按多个字段 4、可以搭配着排序使用 */ #引入:查询每个部门的员工个数 SELECT COUNT(*) FROM employees group by department_id; #1.简单的分组 #案例1:查询每个工种的员工平均工资 SELECT AVG(salary),job_id FROM employees GROUP BY job_id; #案例2:查询每个位置的部门个数 SELECT COUNT(*),location_id FROM departments GROUP BY location_id; #2、可以实现分组前的筛选 #案例1:查询邮箱中包含a字符的 每个部门的最高工资 SELECT MAX(salary),department_id FROM employees WHERE email LIKE '%a%' GROUP BY department_id; #案例2:查询有奖金的每个领导手下员工的平均工资 SELECT AVG(salary),manager_id FROM employees WHERE commission_pct IS NOT NULL GROUP BY manager_id; #3、分组后筛选 #案例:查询哪个部门的员工个数>5 #①查询每个部门的员工个数 SELECT COUNT(*),department_id FROM employees GROUP BY department_id; #② 筛选刚才①结果 SELECT COUNT(*),department_id FROM employees GROUP BY department_id HAVING COUNT(*)>5; #案例2:每个工种有奖金的员工的最高工资>12000的工种编号和最高工资 SELECT job_id,MAX(salary) FROM employees WHERE commission_pct IS NOT NULL GROUP BY job_id HAVING MAX(salary)>12000; #案例3:领导编号>102的每个领导手下的最低工资大于5000的领导编号和最低工资 manager_id>102 SELECT manager_id,MIN(salary) FROM employees GROUP BY manager_id HAVING MIN(salary)>5000; #4.添加排序 #案例:每个工种有奖金的员工的最高工资>6000的工种编号和最高工资,按最高工资升序 SELECT job_id,MAX(salary) m FROM employees WHERE commission_pct IS NOT NULL GROUP BY job_id HAVING m>6000 ORDER BY m ; #5.按多个字段分组 #案例:查询每个工种每个部门的最低工资,并按最低工资降序 SELECT MIN(salary),job_id,department_id FROM employees GROUP BY department_id,job_id ORDER BY MIN(salary) DESC;
排序查询
1、语法: select 查询列表 from 表名 【where 筛选条件】 order by 排序的字段或表达式;
2、特点: 1、asc代表的是升序,可以省略 desc代表的是降序 2、order by子句可以支持 单个字段、别名、表达式、函数、多个字段 3、order by子句在查询语句的最后面,除了limit子句
练习案例
#1、按单个字段排序 SELECT * FROM employees ORDER BY salary DESC; #2、添加筛选条件再排序 #案例:查询部门编号>=90的员工信息,并按员工编号降序 SELECT * FROM employees WHERE department_id>=90 ORDER BY employee_id DESC; #3、按表达式排序 #案例:查询员工信息 按年薪降序 SELECT *,salary*12*(1+IFNULL(commission_pct,0)) FROM employees ORDER BY salary*12*(1+IFNULL(commission_pct,0)) DESC; #4、按别名排序 #案例:查询员工信息 按年薪升序 SELECT *,salary*12*(1+IFNULL(commission_pct,0)) 年薪 FROM employees ORDER BY 年薪 ASC; #5、按函数排序 #案例:查询员工名,并且按名字的长度降序 SELECT LENGTH(last_name),last_name FROM employees ORDER BY LENGTH(last_name) DESC; #6、按多个字段排序 #案例:查询员工信息,要求先按工资降序,再按employee_id升序 SELECT * FROM employees ORDER BY salary DESC,employee_id ASC;
连接查询
笛卡尔乘积现象
表1 有m行,表2有n行,结果=m*n行
发生原因:没有有效的连接条件
如何避免:添加有效的连接条件
按功能分类
内连接
等值连接
非等值连接
自连接
外连接
左外连接
右外连接
全外连接
交叉连接
92标准仅支持内连接;99标准支持内连接、外连接(左外和右外)、交叉连接
分类 按年代分类
92标准
等值连接
① 多表等值连接的结果为多表的交集部分 ②n表连接,至少需要n-1个连接条件 ③ 多表的顺序没有要求 ④一般需要为表起别名 ⑤可以搭配前面介绍的所有子句使用,比如排序、分组、筛选
练习
#案例1:查询女神名和对应的男神名 SELECT NAME,boyName FROM boys,beauty WHERE beauty.boyfriend_id= boys.id; #案例2:查询员工名和对应的部门名 SELECT last_name,department_name FROM employees,departments WHERE employees.`department_id`=departments.`department_id`; #2、为表起别名 /* ①提高语句的简洁度 ②区分多个重名的字段 注意:如果为表起了别名,则查询的字段就不能使用原来的表名去限定 */ #查询员工名、工种号、工种名 SELECT e.last_name,e.job_id,j.job_title FROM employees e,jobs j WHERE e.`job_id`=j.`job_id`; #3、两个表的顺序是否可以调换 #查询员工名、工种号、工种名 SELECT e.last_name,e.job_id,j.job_title FROM jobs j,employees e WHERE e.`job_id`=j.`job_id`; #4、可以加筛选 #案例:查询有奖金的员工名、部门名 SELECT last_name,department_name,commission_pct FROM employees e,departments d WHERE e.`department_id`=d.`department_id` AND e.`commission_pct` IS NOT NULL; #案例2:查询城市名中第二个字符为o的部门名和城市名 SELECT department_name,city FROM departments d,locations l WHERE d.`location_id` = l.`location_id` AND city LIKE '_o%'; #5、可以加分组 #案例1:查询每个城市的部门个数 SELECT COUNT(*) 个数,city FROM departments d,locations l WHERE d.`location_id`=l.`location_id` GROUP BY city; #案例2:查询有奖金的每个部门的部门名和部门的领导编号和该部门的最低工资 SELECT department_name,d.`manager_id`,MIN(salary) FROM departments d,employees e WHERE d.`department_id`=e.`department_id` AND commission_pct IS NOT NULL GROUP BY department_name,d.`manager_id`; #6、可以加排序 #案例:查询每个工种的工种名和员工的个数,并且按员工个数降序 SELECT job_title,COUNT(*) FROM employees e,jobs j WHERE e.`job_id`=j.`job_id` GROUP BY job_title ORDER BY COUNT(*) DESC; #7、可以实现三表连接? #案例:查询员工名、部门名和所在的城市 SELECT last_name,department_name,city FROM employees e,departments d,locations l WHERE e.`department_id`=d.`department_id` AND d.`location_id`=l.`location_id` AND city LIKE 's%' ORDER BY department_name DESC;
非等值连接
练习
#案例1:查询员工的工资和工资级别 SELECT salary,grade_level FROM employees e,job_grades g WHERE salary BETWEEN g.`lowest_sal` AND g.`highest_sal` AND g.`grade_level`='A'; 下方为工资级别表 /* select salary,employee_id from employees; select * from job_grades; CREATE TABLE job_grades (grade_level VARCHAR(3), lowest_sal int, highest_sal int); INSERT INTO job_grades VALUES ('A', 1000, 2999); INSERT INTO job_grades VALUES ('B', 3000, 5999); INSERT INTO job_grades VALUES('C', 6000, 9999); INSERT INTO job_grades VALUES('D', 10000, 14999); INSERT INTO job_grades VALUES('E', 15000, 24999); INSERT INTO job_grades VALUES('F', 25000, 40000); */
自连接
练习
#案例:查询 员工名和上级的名称 SELECT e.employee_id,e.last_name,m.employee_id,m.last_name FROM employees e,employees m WHERE e.`manager_id`=m.`employee_id`;
99标准
mysql
windows下msi安装
windows下解压安装
引子
实际用到数据库的情况---网站、APP登录
数据容器---数组、集合:数据存储到内存中,断电后数据丢失 文件或文件夹:不易做查询、筛选
数据库
数据库的优点
1、持久化数据写入到本地
2、结构化查询、方便管理
常见概念
DB--database:存储数据的仓库,储存有组织的数据
DBMS--Database Mangerment System数据库管理系统,用于管理DB
SQL--System Query Language:结构化查询语言,与DBMS进行通信
数据库存储的特点
1、数据存到表中,表再放到库中
2、一个库中有多张表,每张表具有唯一的名来标识自己
3、表中有一个或多个列,列又称字段,相当于JAVA中的属性
4、表中的每一行数据,相当于JAVA这个i部分的对象