mysql语法总结及例子

1. DDL相关

 a.

-- 查询所有数据库
show databases;
-- 删除数据库
drop database ladeng;

 b.

-- use `数据库名称`;  表示使用此数据库
use ladeng;

 c.

-- 查看表结构
show create table student;

d.  往表中添加新字段

ALTER TABLE `report_user_manage` ADD is_delete tinyint(2) NOT NULL COMMENT '是否删除 0-未删除 1-已删除';
ALTER TABLE `report_user_manage` ADD creator varchar(50) NOT NULL COMMENT '创建人';
ALTER TABLE `report_user_manage` ADD gmt_create datetime NOT NULL COMMENT '创建时间';
-- 当数据新插入时更新时间默认为null, 当数据修改时更新时间会自动变化
ALTER TABLE `report_user_manage` ADD gmt_modified datetime DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP COMMENT
'更新时间';

e

-- 如果源字段值是其它类型如:vachar且有值, 修改为新字段bigint类型,则会报报,需要先清空这个字段值才能修改字段名称; 如果源字段值都为null或者类型也为bigint则直接修改不会报错
ALTER TABLE 表名 CHANGE  `源字段名称` `目标字段名称`  bigint(20)  DEFAULT NULL COMMENT '逾期已还用户数';

f.

-- 删除表
drop table 表名

 

2. 查询表总共有多少条数据, 一般有如下3中写法,  推荐优先使用 count(1) 或 count(id) 

-- count(1):性能高
select count(1) from student;
-- count(带索引的字段):性能高, count(未带索引的字段):性能一般, count(字段)时需要注意,如果该字段值存在有null则统计有误差.
select count(字段名称) from student; 如: select count(id) from student; -- count(*): * 会扫描所有列,故性能比前面几种低 select count(*) from student;

 3. 单条插入

-- 如果选中了数据库则数据库名称可以省略; 
-- 如果数据库名称或表名称是中文名或者字段名称是关键字,则尽量加上 `` 包裹起来
-- 如果id是自增,则id默认一般不能插入,除非自己修改数据库配置允许插入id;
-- 所有字段值都可以用单引号包围, 如果字段类型不是varchar类型,也能使用''包裹字段值, 底层做了类型强制转换成varchar INSERT INTO 数据库名称.表名称 (字段名称1, 字段名称2) VALUES(
'字段1值', '字段2值'); INSERT INTO `ladeng`.`student` (`name`, `age`, `score`, `courseId`) VALUES ('小明', '19', '100', '3');

 4. 批量插入(批量插入条数看插入的字段个数和字段值的字符数,  方式一: 推荐500-5000条,  方式二推荐: 100-1000条, 避免内存占用过高)

--  方式一(多条插入语句中间用分号; 分隔): 如: 下面这2条同时执行
INSERT INTO `ladeng`.`student` (`name`, `age`, `score`, `courseId`) VALUES ('小红', '18', '15', '3');
INSERT INTO `ladeng`.`student` (`name`, `age`, `score`, `courseId`) VALUES ('小丽', '22', '25', '3');

-- 方式二: values后面跟多条数据,中间用顿号、分隔
INSERT INTO `ladeng`.`student` (`name`, `age`, `score`, `courseId`) VALUES ('小红', '18', '15', '3'),('小丽', '22', '25', '3');

 5. 单条更新语句

UPDATE `ladeng`.`student` SET `score`='79', age = 19  WHERE `id` = '1';

6. 批量更新语句: 参照批量插入的第一种方式

7.  删除语句

delete from student where id in (14, 15);
-- 方式一: 删除表中所有数据,(会记录删除日志,可根据日志恢复, 删除后id不会重置为0)
delete from 表名

-- 方式二: 删除表中所有数据:(不会记录删除日志,不能恢复, 删除后id会重置为0, 原理类似于删除表,然后再创建表,性能高)
truncate table 表名;

8.  between ... and ... 在什么之间, 如: 查询年龄>= 19并且年龄<=22数据

select * from student where age between 19 and 22  等价于
select * from student where age >=19 and age <= 22 

9. 聚焦函数(count, sum,avg, max, min, group by), 将多行数据聚焦成一个值

-- 查询总条数, 如果不存在id>2的数据则count(1)会返回0,而不是返回null
select count(1) from student where id > 2
-- 查询所有学生所有科目总分数, 如果表中没有数据则sum(score)会返回null,而不是返回0
select sum(score) from student 
-- 查询所有学生所有科目总分数, 如果没有数据就默认返回0; ifNull是用来判断是否为空,为空则赋值一个默认值
select ifNull(sum(score), 0) from student
-- 查询每个学生所有科目分数, 按照学生名字分组; group by 作用是分组,如果需要对多个字段分组,则 group by 字段1,字段2,...
select `name`, sum(score) from student group by `name`;

 10.  sql中的3种判断语句

  下面那些sql执行的结果都等于这张图片

 第一种用法(推荐使用这种,没有局限性,): case when   条件表达式1   then   值1       when  条件表达式2   then  值2     ...    else 默认值 end  as `字段别名`

-- 这3条sql意思都一样,结果如下图 (else可以省略); 等价于java中的if语句
select  
  is_delete,
  case 
    when is_delete = 0 then '未删除'
    else '已删除'
  end as is_delete_zh
from student 

select  
  is_delete,
  case 
    when is_delete = 0 then '未删除'
    when is_delete = 1 then '已删除'
  end as is_delete_zh
from student 

select  
  is_delete,
  case 
    when is_delete = 0 then '未删除'
    when is_delete = 1 then '已删除'
  else '' end as is_delete_zh
from student 

第二种用法; 

-- 这2条sql意思一样,结果如下图(else可以省略); 等价于java中的 switch语句
select 
  is_delete,
  case is_delete
    when 0 then '未删除'
    when 1 then '已删除'
  end as is_delete_zh
from student 

-- 等价于java中的 switch语句.
select 
  is_delete,
  case is_delete
       when 0 then '未删除'
       else '已删除'
  end as is_delete_zh
from student 

第三种用法:

-- 等价于java中的三目运算符
select is_delete, if(is_delete = 0, '未删除', '已删除') as is_delete_zh from student;

 11. 连表查询

  

 

性能比较: 内连接查询 > left join 或 right join > where连接查询(产生笛卡尔积,耗资源,性能很低)
     内连接查询性能与left join接近, 数据量很大时且很多数据关联字段没有关联上时, 内连接性能会高一点, 
     因为连接查询的结果会写入到一个结果集中, 内连接后的数据量少占用内存小执行效率快,且内连接写入的数据量小会节约时间
如果是多表(3张或以上)连接查询则过滤数据量的条件放在on 后面比放在where条件性能更高;
如: select a.* from a left join b on a.time > '..' and a.bid = b.id
     left join c on c.id = b.cid where ...;

 

12. DATE_FORMAT语法(日期格式化)

select createTime,DATE_FORMAT(createTime, '%Y-%m-%d'),DATE_FORMAT(createTime, '%Y-%m-%d %H:%i:%s') from user

格式描述
%a 缩写星期名
%b 缩写月名
%c 月,数值
%D 带有英文前缀的月中的天
%d 月的天,数值(00-31)
%e 月的天,数值(0-31)
%f 微秒
%H 小时 (00-23)
%h 小时 (01-12)
%I 小时 (01-12)
%i 分钟,数值(00-59)
%j 年的天 (001-366)
%k 小时 (0-23)
%l 小时 (1-12)
%M 月名
%m 月,数值(00-12)
%p AM 或 PM
%r 时间,12-小时(hh:mm:ss AM 或 PM)
%S 秒(00-59)
%s 秒(00-59)
%T 时间, 24-小时 (hh:mm:ss)
%U 周 (00-53) 星期日是一周的第一天
%u 周 (00-53) 星期一是一周的第一天
%V 周 (01-53) 星期日是一周的第一天,与 %X 使用
%v 周 (01-53) 星期一是一周的第一天,与 %x 使用
%W 星期名
%w 周的天 (0=星期日, 6=星期六)
%X 年,其中的星期日是周的第一天,4 位,与 %V 使用
%x 年,其中的星期一是周的第一天,4 位,与 %v 使用
%Y 年,4 位
%y 年,2 位

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 12. 拼接字符

 a.  concat_ws用法

   

 

   b.   concat用法(如果有一个参数值为null则拼接后的结果为null)

 

 c.   concat_group用法

 

 

 

 

 

 

 

 

12.1

-- 日期加1天
select DATE_FORMAT(DATE_ADD(now(),INTERVAL 1 DAY), '%Y-%m-%d %H:%i:%s');
-- 日期减1天
select DATE_FORMAT(DATE_SUB(now(),INTERVAL 1 DAY), '%Y-%m-%d %H:%i:%s');

13. round用法

 项目中一个保留2位小数百分比例子(仅供参考), 如果是数字类型,小数点后面末尾的零会自动省略掉,故需要转换成varchar 如: 0.00% 或 100.00% 或 91.67%

concat_ws('', ifNull(round(ifNull(sum(if(credit_apply_status = 0, 1, 0)), 0) / count(1), 2) * 100, '0.00'), '%') as creditPassScale

另一个经典案例

 

select concat_ws('', 'insert into t2 (name, age) values(', '\'', name, '\',', age, ');') as batchInsertSql from t1;

 

 14.  distinct去重, 和 group by分组有点类似

select distinct name from user
-- 查询用户数
select count(distinct name) from user

15. like模糊查询用法, 和mysql中的正则表达式功能类似

 16.  in语法(in的参数个数最好越少越好, 个数超过2000个很容易出问题)
 

 

17. 优先级: where > group by > having > order by > limit

 

 

 

 18.  数据统计经典例子(这种可以将不同条件统计查询用一条sql查询出来)

  

select 
    ifNull(sum(case when status = 0 and age >= 18 then 1 else 0 end), 0) as '成年人未删除总数'
  , ifNull(sum(case when status = 0 and age < 18 then 1 else 0 end), 0) as '未年人未删除总数'
  , ifNull(sum(case when status = 1 and age >= 18 then 1 else 0 end), 0) as '成年人已删除总数'
  , ifNull(sum(case when status = 1 and age < 18 then 1 else 0 end), 0) as '未年人已删除总数'
from user1 

 19. 去重经典案例(这种对于一张表中有多条数据相同, 保留最大id或最小id那条数据很适用)

 

-- 每个科目前2名学生和分数
select s1.* from student1 s1 where
(select count(1) from student1 s2 where s1.course = s2.course and s1.score < s2.score) < 2
order by s1.course, s1.score desc;

 20.  union(性能比union all低)和union all用法

 

 

 

 21.  将表数据复制到另一张表

a.  新表之前存在

-- 将t1表数据插入到t2表中(t2表原本已经存在,如果不存在则会报错)
insert into t2 (username, age) select name,age from t1
-- 如果表t2是新建的一张表,并且从未插入过数据(如果之前插入过数据则会报id重复无法插入错误),则上面那条sql可以简写为:
insert into t2 select * from t1 

b.  新表之前不存在

-- 将t1表复制一份改名为tt3,但是主键及自增以及索引等都会消失,复制不过去;数据会复制过去
create table tt3 select * from t1; 

22. exist用法

-- 下面那3条语exists语句类似于java的
  List<Object>  list结果集 = new ArrayList<>();
  for (Object a list1) {
     for (Object b list2) {
       if (条件) {
         list结果集.add(a);
         break;
       }
     }  
  }

 

 

 

23.  mysql常用的系统表

  a.  从系统表获取列字段信息

select 
    table_schema, table_name, column_name, column_default, is_nullable
    , data_type, character_maximum_length, character_set_name
    , column_type, column_key, extra, column_comment
from information_schema.columns where table_schema='ladeng' and table_name= 'student'

 b.

-- 查询一个数据库下所有的表信息
select * from information_schema.TABLES where TABLE_SCHEMA = 'ladeng'

 c.

-- 查询一个数据库服务器下所有的数据库信息
select * from information_schema.SCHEMATA

 

24. 存储过程

 

25.  mysql正则表达式

 # 匹配出: /api/relation/aaa/import,  /api/relation/bbb/import,  /api/relation/aaa/bbb/import 等路径

select * from sys_path where path regexp '/api/relation/.*/import';

 # 匹配出: /api/relation/aaa/import,  /api/relation/bbb/import,   但是不匹配  /api/relation/aaa/bbb/import 等路径 (匹配任意一个或多个字符但是不包含 / )

select * from sys_path where path regexp '/api/relation/((?!/).)+/import';

26.  视图

27.  变量使用

posted @ 2019-11-01 21:15  剑阁丶神灯  阅读(555)  评论(0编辑  收藏  举报