- 复制表结构
- 指定IP和端口
- 时间自动更新
- 补充自增主键
- 分组排序取前N条记录
- 查询结果加序号
- LPAD
- find_in_set
- 日期
- 中文排序
- 每月的天数是否正确
- 每年的天数是否正确
- 修改字段值为一个查询值
- 正则实现 like
- 统计关联记录数
复制表结构
CREATE TABLE 【table_name】 LIKE 【table_template】;
该语句创建的新表【table_name】带有模板【table_template】的所有属性、主键、索引等
指定IP和端口
mysql -uroot -p -h192.168.3.146 -P3306
时间自动更新
ALTER TABLE `tb_user` CHANGE `modified_time` `modified_time` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP
补充自增主键
alter table `表格名` add column `列名` int not null auto_increment primary key comment '主键' first;
分组排序取前N条记录
建表并初始化数据:
CREATE TABLE lin(
id integer NOT NULL,
type varchar(1) ,
CONSTRAINT lin_pkey PRIMARY KEY (id)
);
INSERT INTO lin(id, type) VALUES (1, 'a'),(2, 'a'),(3, 'a'),(4, 'b'),(5, 'b'),(6, 'b'),(7, 'b'),(8, 'c'),(9, 'c'),(10, 'c');
执行查询:
select * from lin a where (select count(1) from lin b where a.type = b.type and a.id < b.id) < 2
执行结果为:
+----+------+
| id | type |
+----+------+
| 2 | a |
| 3 | a |
| 6 | b |
| 7 | b |
| 9 | c |
| 10 | c |
+----+------+
SQL解释:
- 子查询是查询所有和当前记录type一样但是id比当前记录大的记录的总和
- type和当前记录一样,但是id比当前记录大的记录如果大于或者等于两个,说明当前至少排在第三个
- type和当前记录一样,但是id比当前记录大的记录如果小于两个,说明当前记录可以排在前两个
查询结果加序号
select (@i:=@i+1) num,u.user_name from tb_user u,(select @i:=0) t
MySQL变量不用事前声明,只需要“@变量名”使用即可
LPAD
find_in_set
日期
-
时间函数
select now(), sysdate(), curdate(), curtime(); "now()": "2018-01-06 10:05:19", "sysdate()": "2018-01-06 10:05:19", "curdate()": "2018-01-06", "curtime()": "10:05:19"
-
now()和sysdate()的区别
now() 在执行开始时值就得到了,sysdate() 在函数执行时动态获取值MariaDB [(none)]> select sysdate(), sleep(3), sysdate(); +---------------------+----------+---------------------+ | sysdate() | sleep(3) | sysdate() | +---------------------+----------+---------------------+ | 2018-01-12 17:23:05 | 0 | 2018-01-12 17:23:08 | +---------------------+----------+---------------------+ 1 row in set (3.01 sec) MariaDB [(none)]> select now(), sleep(3), now(); +---------------------+----------+---------------------+ | now() | sleep(3) | now() | +---------------------+----------+---------------------+ | 2018-01-12 17:23:13 | 0 | 2018-01-12 17:23:13 | +---------------------+----------+---------------------+ 1 row in set (3.00 sec)
-
日期和毫秒数转换
select UNIX_TIMESTAMP('2018-01-06 10:04:30'); select from_unixtime(1515204270);
-
为日期增加/减少时间间隔
select date_add(now(), interval 1 day); select date_add(now(), interval -1 day); 增加/减少的时间单位可选: day、hour、minute、second、microsecond、week、month、quarter、year
中文排序
GBK编码时一级汉字3755个采用拼音排序,二级汉字不是
但考虑到常用汉字都是一级汉字,所以中文排序依赖GBK编码基本上可以满足需求
若数据库采用GBK编码,则直接order by [field] desc/asc
即可
若数据库采用utf8编码,则使用order by convert([field] using gbk) desc/asc
即可
示例:
INSERT INTO `tb_order` (`id`, `name`) VALUES
(1,'赵'),
(2,'钱'),
(3,'孙'),
(4,'李'),
(5,'赵赵'),
(6,'赵钱'),
(7,'赵孙'),
(8,'赵李'),
(9,'啦'),
(10,'lb'),
(11,'la');
结果如下:
MariaDB [learn]> select * from tb_order order by convert(name using gbk) asc;
+----+--------+
| id | name |
+----+--------+
| 11 | la |
| 10 | lb |
| 9 | 啦 |
| 4 | 李 |
| 2 | 钱 |
| 3 | 孙 |
| 1 | 赵 |
| 8 | 赵李 |
| 6 | 赵钱 |
| 7 | 赵孙 |
| 5 | 赵赵 |
+----+--------+
每月的天数是否正确
每天记录一条记录,现在需要校验是否有某天遗漏
select *
from (select year,month,count(*) as days from tb_init group by year,month) AS A
where (month='01' and days!=31)
or (month='03' and days!=31)
or (month='05' and days!=31)
or (month='07' and days!=31)
or (month='08' and days!=31)
or (month='10' and days!=31)
or (month='12' and days!=31)
or (month='04' and days!=30)
or (month='06' and days!=30)
or (month='09' and days!=30)
or (month='11' and days!=30)
or (month='02' and days!= case when year%4=0 and year%100!=0 or year%400=0 then 29 else 28 end);
每年的天数是否正确
select *
from (select year,count(*) as days from tb_init group by year) AS A
where days!= case when year%4=0 and year%100!=0 or year%400=0 then 366 else 365 end;
修改字段值为一个查询值
update tb_auth a inner join (select auth,dynasty from tb_poetry group by auth) b
set a.dynasty = b.dynasty
where a.name = b.auth and a.dynasty is null;
正则实现 like
select * from tb_order;
#下面两条SQL等价
select * from tb_order where name like '%赵%' or name like '%l%';
select * from tb_order where name regexp '赵|l';
第一条SQL结果如下:
1 赵
2 钱赵
3 孙赵
4 李
5 赵赵
6 赵钱
7 赵孙
8 赵李
9 啦
10 lb
11 la
第二、三条SQL结果如下:
1 赵
2 钱赵
3 孙赵
5 赵赵
6 赵钱
7 赵孙
8 赵李
10 lb
11 la
统计关联记录数
若关联了记录,则统计其数量;若没有关联记录,则为0
思路:
先统计数量再关联,为空则置0
SELECT u.*,IFNULL(c.num,0) as num
FROM tb_user u LEFT JOIN (SELECT user_id,count(*) num FROM tb_user_log GROUP BY user_id) c ON u.user_id = c.user_id