Mysql语句快速复习教程(全)
程序员文章站
2024-02-16 19:51:28
基本操作
查看数据库
show databases;
指定字符集
...
基本操作
查看数据库
<code>show databases; </code>
指定字符集
<code>create database day15 default character set utf8 </code>
查看字符集
<code>show create database day15; </code>
删除
<code>drop database day15 </code>
修改字符集
<code>alter database day15 default character set gbk; </code>
使用数据库
<code>use day15; </code>
查看所有表
<code>show tables; </code>
创建表
<code>create table student( id int, name varchar(20), gender varchar(2), age int ) </code>
查看表结构:
<code>desc student; </code>
删除表
<code>drop table student; </code>
增
<code>-- 插入所有字段。一定依次按顺序插入 insert into student values(1,'张三','男',20); -- 注意不能少或多字段值 -- insert into student values(2,'李四','女'); -- 插入部分字段 insert into student(id,name) values(2,'李四'); </code>
改
<code>-- 修改所有数据(建议少用) update student set gender='女'; -- 带条件的修改(推荐使用) update student set gender='男' where id=1; -- 修改id为1的学生,修改性别为男 -- 修改多个字段,注意: set 字段名=值,字段名=值,.... update student set gender='男',age=30 where id=2; </code>
删
<code>-- 删除所有数据(建议少用) delete from student; -- 带条件的删除(推荐使用) delete from student where id=2; -- 和另一种方式比较 -- delete from: 可以全表删除 1)可以带条件删除 3)使用delete from删除的数据可以回滚(事务) -- truncate table: 可以全表删除 1)不能带条件删除 2)即可以删除表的数据,也可以删除表的约束 3)使用truncate table删除的数据不能回滚 truncate table student; </code>
查
<code>-- 2.1 查询所有列 select * from student; -- 2.2 查询指定列 select id,name,gender from student; -- 2.3 查询时指定别名(as) -- 注意: 在多表查询是经常使用表的别名 select id as '编号',name as '姓名' from student; -- 2.4 查询时添加常量列 -- 需求: 在查询student表时添加一个班级列,内容为“java就业班” select id,name,gender,age,'java就业班' as '年级' from student; -- 2.5 查询时合并列 -- 需求: 查询每个学生的servlet和jsp的总成绩 select id,name,(servlet+jsp) as '总成绩' from student; -- 注意:合并列只能合并数值类型的字段 select id,(name+servlet) from student; -- 2.6 查询时去除重复记录(distinct) -- 需求: 查询学生的性别 男 女 select distinct gender from student; -- 另一种语法 select distinct(gender) from student; -- 需求: 查询学生所在的地区 select distinct address from student; -- 2.7 条件查询(where) -- 2.7.1 逻辑条件: and(与) or(或) -- 需求: 查询id为2,且姓名为李四的学生 select * from student where id=2 and name='李四'; -- 交集 -- 需求: 查询id为2,或姓名为张三的学生 select * from student where id=2 or name='张三'; -- 并集 -- 2.7.2 比较条件: > < >= <= = <>(不等于) between and (等价于>= 且 <=) -- 需求: 查询servlet成绩大于70分的学生 select * from student where servlet>70; -- 需求: 查询jsp成绩大于等于75,且小于等于90分的学生 select * from student where jsp>=75 and jsp<=90; -- 另一个语法 select * from student where jsp between 75 and 90; -- (包前包后) select * from student where gender<>'男'; -- 2.7.3 判空条件(null 空字符串): is null / is not null / ='' / <>'' -- 需求: 查询地址为空的学生(包括null和空字符串) -- null vs 空字符串 -- null:表示没有值 -- 空字符串:有值的! -- 判断null select * from student where address is null ; -- 判断空字符串 select * from student where address=''; select * from student where address is null or address=''; -- (包括null和空字符串) -- 需求: 查询有地址的学生(不包括null和空字符串) select * from student where address is not null and address<>''; -- 2.7.4 模糊条件: like -- 通常使用以下替换标记: -- % : 表示任意个字符 -- _ : 表示一个字符 -- 需求: 查询姓‘张'的学生 select * from student where name like '李%'; -- 需求: 查询姓‘李',且姓名只有两个字的学生 select * from student where name like '李_'; -- 2.8 聚合查询(使用聚合函数的查询) -- 常用的聚合函数: sum() avg() max() min() count() -- 需求:查询学生的servlet的总成绩 (sum() :求和函数) select sum(servlet) as 'servlet的总成绩' from student; -- 需求: 查询学生的servlet的平均分 select avg(servlet) as 'servlet的平均分' from student; -- 需求: 查询当前servlet最高分 select max(servlet) as '最高分' from student; -- 需求: 查询最低分 select min(servlet) as '最低分' from student; -- 需求: 统计当前有多少学生(count(字段)) select count(*) from student; select count(id) from student; -- 注意:count()函数统计的数量不包含null的数据 -- 使用count统计表的记录数,要使用不包含null值的字段 select count(age) from student; select * from student; -- 2.9 分页查询(limit 起始行,查询几行) -- 起始行从0开始 -- 分页:当前页 每页显示多少条 -- 分页查询当前页的数据的sql: select * from student limit (当前页-1)*每页显示多少条,每页显示多少条; -- 需求: 查询第1,2条记录(第1页的数据) select * from student limit 0,2; -- 查询第3,4条记录(第2页的数据) select * from student limit 2,2; -- 查询第5,6条记录(第3页的数据) select * from student limit 4,2; -- 查询第7,8条记录 (没有记录不显示) select * from student limit 6,2; -- 2.10 查询排序(order by ) -- 语法 :order by 字段 asc/desc -- asc: 顺序,正序。数值:递增,字母:自然顺序(a-z) -- desc: 倒序,反序。数值:递减,字母:自然反序(z-a) -- 默认情况下,按照插入记录顺序排序 select * from student; -- 需求: 按照id顺序排序 select * from student order by id asc; select * from student order by id; -- 默认正序 select * from student order by id desc;-- 反序 -- 注意:多个排序条件 -- 需求: 按照servlet正序,按照jsp的倒序 select * from student order by servlet asc,jsp desc; -- 2.11 分组查询(group by) -- 需求: 查询男女的人数 -- 预期结果: -- 男 3 --- 女 2 -- 1) 把学生按照性别分组(group by gender) -- 2) 统计每组的人数(count(*)) select gender,count(*) from student group by gender; -- 2.12 分组查询后筛选 -- 需求: 查询总人数大于2的性别 -- 1) 查询男女的人数 -- 2)筛选出人数大于2的记录(having) --- 注意: 分组之前条件使用where关键字,分组之前条件使用having关键字 select gender,count(*) from student where group by gender having count(*)>2; </code>
练习:
<code>create table student2( id int, name varchar(20), chinese float, english float, math float ); </code> insert into student2(id,name,chinese,english,math) values(1,'张小明',89,78,90); insert into student2(id,name,chinese,english,math) values(2,'李进',67,53,95); insert into student2(id,name,chinese,english,math) values(3,'王五',87,78,77); insert into student2(id,name,chinese,english,math) values(4,'李一',88,98,92); insert into student2(id,name,chinese,english,math) values(5,'李来财',82,84,67); insert into student2(id,name,chinese,english,math) values(6,'张进宝',55,85,45); insert into student2(id,name,chinese,english,math) values(7,'黄蓉',75,65,30); – 查询表中所有学生的信息。 select * from student2; – 查询表中所有学生的姓名和对应的英语成绩。 select name,english from student2; – 过滤表中英语成绩的重复数据 select distinct(english) from student2; – 使用别名表示学生分数。 select name as ‘姓名',chinese as ‘语文',english as ‘英语',math as ‘数学' from student2; – 查询姓名为李一的学生成绩 select * from student2 where name='李一'; – 查询英语成绩大于等于90分的同学 select * from student2 where english>=90; – 查询总分大于200分的所有同学 select * from student2 where (chinese+english+math)>200; – 查询所有姓李的学生英语成绩。 select name,english from student2 where name like ‘李%'; – 查询英语>80或者总分>200的同学 select * from student2 where english>80 or (chinese+english+math)>200 – 统计每个学生的总分。 select id,name,(chinese+english+math) as ‘总成绩' from student2; – 在所有学生总分数上加10分特长分。 select id,name,(chinese+english+math+10) as ‘总成绩' from student2;
数据约束
<code>-- 1.1 默认值 create table student( id int, name varchar(20), address varchar(20) default '广州天河' -- 默认值 ) -- 当字段没有插入值的时候,mysql自动给该字段分配默认值 insert into student(id,name) values(1,'张三'); -- 注意:默认值的字段允许为null insert into student(id,name,address) value(2,'李四',null); insert into student(id,name,address) value(3,'王五','广州番禺'); -- 1.2 非空 -- 需求: gender字段必须有值(不为null) create table student( id int, name varchar(20), gender varchar(2) not null -- 非空 ) -- 非空字段必须赋值 insert into student(id,name) values(1,'李四'); -- 非空字符不能插入null insert into student(id,name,gender) values(1,'李四',null); -- 1.3 唯一 create table student( id int unique, -- 唯一 name varchar(20) ) insert into student(id,name) values(1,'zs'); insert into student(id,name) values(1,'lisi'); -- error 1062 (23000): duplicate entry '1' for key 'id' -- 1.4 主键(非空+唯一) create table student( id int primary key, -- 主键 name varchar(20) ) insert into student(id,name) values(1,'张三'); insert into student(id,name) values(2,'张三'); -- insert into student(id,name) values(1,'李四'); -- 违反唯一约束: duplicate entry '1' for key 'primary' -- insert into student(name) value('李四'); -- 违反非空约束: error 1048 (23000): column 'id' cannot be null -- 1.5 自增长 create table student( id int(4) zerofill primary key auto_increment, -- 自增长,从0开始 zerofill 零填充 name varchar(20) ) -- 自增长字段可以不赋值,自动递增 insert into student(name) values('张三'); -- 不能影响自增长约束 delete from student; -- 可以影响自增长约束 truncate table student; -- 1.6 外键约束 -- 员工表 create table employee( id int primary key, empname varchar(20), deptname varchar(20) -- 部门名称 ) insert into employee values(1,'张三','软件开发部'); insert into employee values(2,'李四','软件开发部'); insert into employee values(3,'王五','应用维护部'); -- 解决数据冗余高的问题:给冗余的字段放到一张独立表中 -- 独立设计一张部门表 create table dept( id int primary key, deptname varchar(20) ) -- 修改员工表 create table employee( id int primary key, empname varchar(20), deptid int,-- 把部门名称改为部门id -- 声明一个外键约束 constraint emlyee_dept_fk foreign key(deptid) references dept(id) on update cascade on delete cascade -- on cascade update :级联修改 ) -- 问题: 该记录业务上不合法,员工插入了一个不存在的部门数据 insert into employee values(5,'陈六',4); -- 违反外键约束: cannot add or update a child row: a foreign key constraint fails (`day16`.`employee`, references `dept` (`id`)) -- 1)当有了外键约束,添加数据的顺序: 先添加主表,再添加副表数据 -- 2)当有了外键约束,修改数据的顺序: 先修改副表,再修改主表数据 -- 3)当有了外键约束,删除数据的顺序: 先删除副表,再删除主表数据 -- 修改部门(不能直接修改主表) update dept set id=4 where id=3; -- 先修改员工表 update employee set deptid=2 where id=4; -- 删除部门 delete from dept where id=2; -- 先删除员工表 delete from employee where deptid=2; -- 级联修改(修改) -- 直接修改部门 update dept set id=5 where id=4; -- 级联删除 -- 直接删除部门 delete from dept where id=1; </code>
关联查询
<code>-- 需求:查询员工及其所在部门(显示员工姓名,部门名称) -- 2.1 交叉连接查询(不推荐。产生笛卡尔乘积现象: 4 * 4=16,有些是重复记录) select empname,deptname from employee,dept; -- 需求:查询员工及其所在部门(显示员工姓名,部门名称) -- 多表查询规则: 1)确定查询哪些表 2)确定哪些哪些字段 3)表与表之间连接条件 (规律:连接条件数量是表数量-1) -- 2.2 内连接查询:只有满足条件的结果才会显示(使用最频繁) select empname,deptname -- 2)确定哪些哪些字段 from employee,dept -- 1)确定查询哪些表 where employee.deptid=dept.id -- 3)表与表之间连接条件 -- 内连接的另一种语法 select empname,deptname from employee inner join dept on employee.deptid=dept.id; -- 使用别名 select e.empname,d.deptname from employee e inner join dept d on e.deptid=d.id; -- 需求: 查询每个部门的员工 -- 预期结果: -- 软件开发部 张三 -- 软件开发部 李四 -- 应用维护部 王五 -- 秘书部 陈六 -- 总经办 null -- 2.2 左[外]连接查询: 使用左边表的数据去匹配右边表的数据,
如果符合连接条件的结果则显示,如果不符合连接条件则显示null
-- (注意: 左外连接:左表的数据一定会完成显示!) select d.deptname,e.empname from dept d left outer join employee e on d.id=e.deptid;
如果符合连接条件的结果则显示,如果不符合连接条件则显示null
-- (注意: 右外连接:右表的数据一定会完成显示!) select d.deptname,e.empname from employee e right outer join dept d on d.id=e.deptid; -- 2.4 自连接查询 -- 需求:查询员工及其上司 -- 预期结果: -- 张三 null -- 李四 张三 -- 王五 李四 -- 陈六 王五 select e.empname,b.empname from employee e left outer join employee b on e.bossid=b.id; </code>
存储过程
<code>-- 声明结束符 -- 创建存储过程 delimiter $ create procedure pro_test() begin -- 可以写多个sql语句; select * from employee; end $ -- 执行存储过程 call pro_test(); -- 3.1 带有输入参数的存储过程 -- 需求:传入一个员工的id,查询员工信息 delimiter $ create procedure pro_findbyid(in eid int) -- in: 输入参数 begin select * from employee where id=eid; end $ -- 调用 call pro_findbyid(4); -- 3.2 带有输出参数的存储过程 delimiter $ create procedure pro_testout(out str varchar(20)) -- out:输出参数 begin -- 给参数赋值 set str='helljava'; end $ -- 删除存储过程 drop procedure pro_testout; -- 调用 -- 如何接受返回参数的值?? -- ***mysql的变量****** -- 全局变量(内置变量):mysql数据库内置的变量 (所有连接都起作用) -- 查看所有全局变量: show variables -- 查看某个全局变量: select @@变量名 -- 修改全局变量: set 变量名=新值 -- character_set_client: mysql服务器的接收数据的编码 -- character_set_results:mysql服务器输出数据的编码 -- 会话变量: 只存在于当前客户端与数据库服务器端的一次连接当中。 如果连接断开,那么会话变量全部丢失! -- 定义会话变量: set @变量=值 -- 查看会话变量: select @变量 -- 局部变量: 在存储过程中使用的变量就叫局部变量。只要存储过程执行完毕, 局部变量就丢失!! -- 1)定义一个会话变量name, 2)使用name会话变量接收存储过程的返回值 call pro_testout(@name); -- 查看变量值 select @name; -- 3.3 带有输入输出参数的存储过程 delimiter $ create procedure pro_testinout(inout n int) -- inout: 输入输出参数 begin -- 查看变量 select n; set n =500; end $ -- 调用 set @n=10; call pro_testinout(@n); select @n; -- 3.4 带有条件判断的存储过程 -- 需求:输入一个整数,如果1,则返回“星期一”, 如果2,返回“星期二”,如果3,返回“星期三”。 其他数字,返回“错误输入”; delimiter $ create procedure pro_testif(in num int,out str varchar(20)) begin if num=1 then set str='星期一'; elseif num=2 then set str='星期二'; elseif num=3 then set str='星期三'; else set str='输入错误'; end if; end $ call pro_testif(4,@str); select @str; -- 3.5 带有循环功能的存储过程 -- 需求: 输入一个整数,求和。例如,输入100,统计1-100的和 delimiter $ create procedure pro_testwhile(in num int,out result int) begin -- 定义一个局部变量 declare i int default 1; declare vsum int default 0; while i<=num do set vsum = vsum+i; set i=i+1; end while; set result=vsum; end $ drop procedure pro_testwhile; call pro_testwhile(100,@result); select @result; -- 3.6 使用查询的结果赋值给变量(into) delimiter $ create procedure pro_findbyid2(in eid int,out vname varchar(20) ) begin select empname into vname from employee where id=eid; end $ call pro_findbyid2(1,@name); select @name; -- 练习: 编写一个存储过程 如果学生的英语平均分小于等于70分,则输出'一般' 如果学生的英语平均分大于70分,且小于等于90分,则输出‘良好' 如果学生的英语平均分大于90分,则输出‘优秀' delimiter $ create procedure pro_testavg(out str varchar(20)) begin -- 定义局部变量,接收平均分 declare savg double; -- 计算英语平方分 select avg(english) into savg from student2; if savg<=70 then set str='一般'; elseif savg>70 and savg<=90 then set str='良好'; else set str='优秀'; end if; end $ call pro_testavg(@str); select @str; </code>
触发器
<code>select * from employee; -- 日志表 create table test_log( id int primary key auto_increment, content varchar(100) ) -- 需求: 当向员工表插入一条记录时,希望mysql自动同时往日志表插入数据 -- 创建触发器(添加) create trigger tri_empadd after insert on employee for each row -- 当往员工表插入一条记录时 insert into test_log(content) values('员工表插入了一条记录'); -- 插入数据 insert into employee(id,empname,deptid) values(7,'扎古斯',1); insert into employee(id,empname,deptid) values(8,'扎古斯2',1); -- 创建触发器(修改) create trigger tri_empupd after update on employee for each row -- 当往员工表修改一条记录时 insert into test_log(content) values('员工表修改了一条记录'); -- 修改 update employee set empname='eric' where id=7; -- 创建触发器(删除) create trigger tri_empdel after delete on employee for each row -- 当往员工表删除一条记录时 insert into test_log(content) values('员工表删除了一条记录'); -- 删除 delete from employee where id=7; select * from employee; select * from test_log; </code>
mysql权限问题
<code> -- mysql数据库权限问题:root :拥有所有权限(可以干任何事情) -- 权限账户,只拥有部分权限(curd)例如,只能操作某个数据库的某张表 -- 如何修改mysql的用户密码? -- password: md5加密函数(单向加密) select password('root'); -- *81f5e21e35407d884a6cd4a731aebfb6af209e1b -- mysql数据库,用户配置 : user表 use mysql; select * from user; -- 修改密码 update user set password=password('123456') where user='root'; -- 分配权限账户(select insert delete update drop create all) grant select on day16.employee to 'eric'@'localhost' identified by '123456'; grant delete on day16.employee to 'eric'@'localhost' identified by '123456'; </code>
备份和还原
<code>-备份 mysqldump -u root -p day17 > c:/back.sql -恢复 mysql -u root -p day17 < d:/back.sql </code>