前言
- 本篇文章主要介绍的关于本人从刚工作到现在使用Sql一些使用方法和经验,从最基本的SQL函数使用,到一些场景的业务场景SQL编写。
复制代码 SQL基础函数使用
1.字段转换
CASE WHEN
意义: If(a==b) a=c;
用法:
1, CASE 字段 WHEN 字段结果1 THEN 字段显示结果1 WHEN 字段结果2 THEN 字段显示结果2 END
2, CASE WHEN 字段1=字段结果1 THEN 字段显示结果1 WHEN 字段2=字段结果2 THEN 字段显示结果2 END
2.替换空值
意义: if(a==null) a=0;
MySQL:IFNULL
用法:IFNULL(字段,0)别名
Oracle:NVL
用法:NVL(字段,0)别名
Sybase: ISNULL
用法:ISNULL(字段,0)别名
3.合计函数
GROUP BY
例:SELECT 字段1, 【如sum】函数名(字段2) FROM 表名 GROUP BY 字段1
4.取某段数据
Mysql: LIMIT
用法: select * from 表 LIMIT 数字 ----取零到数字中的数据
select * from 表 LIMIT 数字1,数字2 ----取数字1到数字2中的数据
Oracle: rownum
用法:select * from 表 rownum1) [/code]9. 日期格式使用
DATE_FORMAT 可以把日志格式化成想要的格式
DATE_FORMAT(date, format)
sql设置日期格式TO_DATE(字段名,YYYY-MM-DD)字段名
时间格式化:- select substr(字段名,起始位置,长度)自定义名 from 表名
复制代码 例如:- SET @rownum=0;
- SELECT @rownum := @rownum +1 AS aid, h.* FROM household h;
复制代码 查询结果为 20181010
根据日期得到星期几DAYOFWEEK是从周日开始,所以要减一,WEEKDAY是从0开始,所有要加一
SELECT DAYOFWEEK('2021-4-22')-1,WEEKDAY('2021-4-20')+1
10.UNION和UNION All比较
- SELECT user_id FROM t_user GROUP BY user_id HAVING AVG(user_age)<22;
复制代码 11. sql之left join、right join、inner join的区别
left join(左联接) 返回包括左表中的所有记录和右表中联结字段相等的记录
左表返回的值一定大于或等于右表的值
right join(右联接) 返回包括右表中的所有记录和左表中联结字段相等的记录
右表返回的值一定大于或等于左表的值
inner join(等值连接) 只返回两个表中联结字段相等的行
左表返回的值一定等于右表返回的值
常用SQL
1.解锁sql表
MySql
1.查看是否有锁表- delete from 表名 where 主键 in
- (select 主键 from 表名 group by 删除的字段数据名 having count(1) > 1)
- and 主键 not in (select min(主键) from 表名 group by 删除的字段数据名 having count(1)>1)
复制代码 2.查询产生锁的具体sql- SELECT DATE_FORMAT(a.`update_time`,'%Y-%m-%d %H:%i:%S') AS updateTime,
- a.`update_time` FROM t_user a
复制代码 3.杀死产生锁的事物线程
根据具体的sql判断是不是死锁,具体是什么业务,是否可以进行kill。
然后根据结果 kill掉产生锁的事物线程:- SELECT DATE_FORMAT('2018-10-10 00:00:00', '%Y%m%d')
复制代码 批量kill :- UNION在进行表链接后会筛选掉重复的记录 UNION ALL只是简单的将两个结果合并后就返回
-
- 例:SELECT 字段 FROM 表1
- UNION
- SELECT 字段 FROM 表2
- SELECT 字段 FROM 表1
- UNION ALL
- SELECT 字段 FROM 表2
复制代码 SqlServer
查看被锁表:- SHOW OPEN TABLES WHERE In_use > 0;
复制代码 spid 锁表进程
tableName 被锁表名
解锁:- select a.trx_id 事务id ,a.trx_mysql_thread_id 事务线程id,a.trx_query 事务sql from INFORMATION_SCHEMA.INNODB_LOCKS b,INFORMATION_SCHEMA.innodb_trx a where b.lock_trx_id=a.trx_id;
复制代码 2.查看一个字段的在那些表中
- select concat('KILL ',a.trx_mysql_thread_id ,';') from INFORMATION_SCHEMA.INNODB_LOCKS b,INFORMATION_SCHEMA.innodb_trx a where b.lock_trx_id=a.trx_id;
复制代码 SqlServer:- select concat('KILL ',a.trx_mysql_thread_id ,';') from INFORMATION_SCHEMA.INNODB_LOCKS b,INFORMATION_SCHEMA.innodb_trx a where b.lock_trx_id=a.trx_id into outfile '/tmp/kill.txt';
复制代码 3. 查询所有表及其字段和备注
- select request_session_id spid,OBJECT_NAME(resource_associated_entity_id) tableName from sys.dm_tran_locks where resource_type='OBJECT'
复制代码 4.sql数据自我复制
- declare @spid int Set @spid = 57
- declare @sql varchar(1000)set @sql='kill '+cast(@spid as varchar)
- exec(@sql)
复制代码 5.删除重复数据,保留最小id的那一条
- SELECT DISTINCT TABLE_NAME FROM information_schema.`COLUMNS` WHERE COLUMN_NAME='ip' AND TABLE_SCHEMA='guard_scan' AND TABLE_NAME NOT LIKE 'vm%';
复制代码 6.SQL查询相隔天数语句
--今天- select table_name from user_tab_columns where COLUMN_NAME='字段'
复制代码 --昨天- SELECT t.table_name,
- t.colUMN_NAME,
- t.DATA_TYPE || '(' || t.DATA_LENGTH || ')',
- t1.COMMENTS
- FROM User_Tab_Cols t, User_Col_Comments t1
- WHERE t.table_name = t1.table_name
- AND t.column_name = t1.column_name;
复制代码 --本周- insert into test(name,age,gender)
- select name,age,gender from test
复制代码 --本月- delete from table_name as ta where ta.唯一键 <> ( select max(tb.唯一键) from table_name as tb where ta.判断重复的列 = tb.判断重复的列);
复制代码 --上一个月- select * from 表名 where to_days(时间字段名) = to_days(now());
复制代码 --本年- SELECT * FROM 表名 WHERE TO_DAYS( NOW( ) ) - TO_DAYS( 时间字段名) = 1
复制代码 --上一月- SELECT * FROM 表名 WHERE YEARWEEK( date_format( 时间字段名,'%Y-%m-%d' ) ) = YEARWEEK( now() ) ;
复制代码 --查询本季度数据- SELECT * FROM 表名 WHERE DATE_FORMAT( 时间字段名, '%Y%m' ) = DATE_FORMAT( CURDATE( ) ,'%Y%m' )
复制代码 --查询上季度数据- SELECT * FROM 表名 WHERE PERIOD_DIFF(date_format(now(),'%Y%m'),date_format(时间字段名,'%Y%m') =1
复制代码 --查询本年数据- SELECT * FROM 表名 WHERE YEAR( 时间字段名 ) = YEAR( NOW( ) )
复制代码 --查询上年数据- SELECT * FROM 表名 WHERE PERIOD_DIFF( date_format( now( ) , '%Y%m' ) , date_format( 时间字段名, '%Y%m' ) ) =1
复制代码 --查询当前这周的数据- select * from `ht_invoice_information` where QUARTER(create_date)=QUARTER(now());
复制代码 --查询上周的数据- select * from `ht_invoice_information` where QUARTER(create_date)=QUARTER(DATE_SUB(now(),interval 1 QUARTER));
复制代码 --查询当前月份的数据- select * from `ht_invoice_information` where YEAR(create_date)=YEAR(NOW());
复制代码 --查询距离当前现在6个月的数据- select * from `ht_invoice_information` where year(create_date)=year(date_sub(now(),interval 1 year));
复制代码 --查询上个月的数据- SELECT name,submittime FROM enterprise WHERE YEARWEEK(date_format(submittime,'%Y-%m-%d')) = YEARWEEK(now());
复制代码 7.mysql经纬度圆周计算
单位为米- SELECT name,submittime FROM enterprise WHERE YEARWEEK(date_format(submittime,'%Y-%m-%d')) = YEARWEEK(now())-1;
复制代码 8.mysql的ip地址段查询判断
ip查询前三段- select name,submittime from enterprise where date_format(submittime,'%Y-%m')=date_format(now(),'%Y-%m')
复制代码 示例:- select name,submittime from enterprise where submittime between date_sub(now(),interval 6 month) and now();
复制代码 9.mysql的地址段大小查询判断
使用INET_ATON函数进行转换- select name,submittime from enterprise where date_format(submittime,'%Y-%m')=date_format(DATE_SUB(curdate(), INTERVAL 1 MONTH),'%Y-%m')
复制代码 其他
MySQL
1.查询所有数据库- SELECT st_distance_sphere(POINT('114.43107891381024', '30.52764363752110'), POINT('114.42638694658900', '30.54681469735225')) AS distcance
复制代码 2.查询指定数据库中所有表名- SELECT SUBSTRING_INDEX(ip,'.',3) FROM t_ip
复制代码 3.查询指定表中的所有字段名- SELECT SUBSTRING_INDEX(tia.ip_addr,'.',3),ip_addr FROM t_ip_all tia
- WHERE SUBSTRING_INDEX(tia.ip_addr,'.',3) = '192.168.21'
复制代码 4.查询指定表中的所有字段名和字段类型- SELECT
- *
- FROM
- 表名
- WHERE
- INET_ATON(ip) between INET_ATON("192.168.21.0")
- AND INET_ATON("192.168.1.255")
复制代码 SQLServer
1.查询所有数据库2.查询当前数据库中所有表名- select table_name from information_schema.tables where table_schema='database_name' and table_type='base table';
复制代码 xtype='U':表示所有用户表,xtype='S':表示所有系统表。
3.查询指定表中的所有字段名- select column_name from information_schema.columns where table_schema='database_name' and table_name='table_name';
复制代码 4.查询指定表中的所有字段名和字段类型- select column_name,data_type from information_schema.columns where table_schema='database_name' and table_name='table_name';
复制代码 Oracle
1.查询所有数据库
由于Oralce没有库名,只有表空间,所以Oracle没有提供数据库名称查询支持,只提供了表空间名称查询。- select * from sysdatabases;
复制代码 2.查询当前数据库中所有表名- select * from sysobjects where xtype='U';
复制代码 3.查询指定表中的所有字段名- select name from syscolumns where id=Object_Id('table_name');
复制代码 4.查询指定表中的所有字段名和字段类型- select sc.name,st.name from syscolumns sc,systypes st where sc.xtype=st.xtype and sc.id in(select id from sysobjects where xtype='U' and name='table_name');
复制代码 业务场景SQL
- select * from v$tablespace;--查询表空间(需要一定权限)
复制代码 学生排名统计
一张表t,有class(班级)、name(学生)、score(成绩)字段
查询每个班级成绩最高的学生
思路:根据分组函数 group by 和最大值 max来实现。- select * from user_tables;
复制代码 查找出每个班级成绩前三的学生
思路: 通过双重子查询来查找
先对学生的成绩进行排名,相同的为一列,然后在跟进这个结果得到前三成绩的学生。- select column_name from user_tab_columns where table_name = 'table_name';--表名要全大写
复制代码 查找出每个班级成绩第二的学生。
思路: 在条件里面发现最大的,然后去除最大的就是第二大的- select column_name, data_type from user_tab_columns where table_name = 'table_name';--表名要全大写
复制代码 查询每名学生的学科总分并排名
思路: 先用group by和sum得到总分排名,然后再利用order by将结果进行排名- 这是在一些常见的场景中个人编写以及收集的一些SQL,从刚开始工作的时候就有记录,如有不妥或有更好的写法,欢迎指出~
复制代码 查询用一条SQL 语句 查询出每门课都大于80 分的学生姓名
思路一:利用group by和having 函数来查询- Select name,class,max(score) from t group by class;
复制代码 思路二:利用子查询来查询- SELECT *
- FROM( SELECT NAME,score , class,(SELECT COUNT(*)+1 FROM t WHERE score>b.score AND class = b.class ) rank
- FROM t b) e
- WHERE e.rank<=3
- ORDER BY class,rank ASC;
复制代码 根据班级进行区分- SELECT * FROM t a
- WHERE 4 >(SELECT count(*)+1 FROM t WHERE class = a.class and score>a.score)
- ORDER BY a.class,a.score DESC;
复制代码 场景二、分数相同排名相同(如果有两个第二,有第三名)
思路:需要使用一个额外的变量进行查询比较- SELECT class, MAX(score) FROM t WHERE score NOT IN (SELECT MAX(score) FROM t GROUP BY class) GROUP BY class
复制代码 组织部门查询
一张部门表,有id(主键)、name(名称)、parent_id(父级ID)字段
建表语句和测试数据- SELECT *,SUM(scroe)scroe FROM a GROUP BY sID ORDER BY scroe DESC;
复制代码 根据部门ID查询该部门下面所有的子部门
思路: 由于不清楚部门层级,这里需要使用递归查询,需要定义一个变量和FIND_IN_SET函数来实现递归查询。当查询到一个部门时,将其ID添加到一个变量中,然后继续查询其子部门,直到所有子部门都被查询到为止。- SELECT * FROM a GROUP BY scroe HAVING AVG(scroe) >80;
复制代码 如果是MySql8.0,可以使用WITH RECURSIVE关键字实现递归查询- SELECT * FROM a c WHERE scroe NOT IN (SELECT scroe FROM a b WHERE scroe<='80') GROUP BY sID;
复制代码 根据一个部门ID,查询所有的上级部门
思路: 这篇的作者讲得很详细,这里就不在赘述了.
https://www.cnblogs.com/liuxiaoji/p/15219091.html- SELECT id,NAME,score, (SELECT COUNT(*)+1 FROM t_student WHERE score>t.score) rank FROM t_student t ORDER BY rank ASC;
复制代码 如果是MySql8.0,可以使用WITH RECURSIVE关键字实现递归查询- SELECT class,NAME,score,(SELECT COUNT(*)+1 FROM t WHERE score>b.`score` AND class=b.`class` )rank FROM t b ORDER BY class,rank;
复制代码 其他
手记系列
记载个人从刚开始工作到现在各种杂谈笔记、问题汇总、经验累积的系列。
手记系列
一首很带感的动漫钢琴曲~
原创不易,如果感觉不错,希望给个推荐!您的支持是我写作的最大动力!
版权声明:
作者:虚无境
博客园出处:http://www.cnblogs.com/xuwujing
CSDN出处:http://blog.csdn.net/qazwsxpcm
个人博客出处:https://xuwujing.github.io/
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作! |