SQL经典语句
一. 基础篇
1. 创建数据库
1 | CREATE DATABASE database-name |
2. 删除数据库
1 | DROP database dbname |
3. 备份sql server
创建 备份数据的 device
1 | USE master |
开始 备份
1 | BACKUP DATABASE pubs TO testBack |
4. 创建新表
1 | create table tabname(col1 type1 [not null] [primary key],col2 type2 [not null],..) |
根据已有的表创建新表:
1 | create table tab_new like tab_old (使用旧表创建新表) |
5. 删除新表
1 | DROP TABLE tabname |
6. 增加一个列
Alter table tabname add column col type
注:列增加后将不能删除。DB2中列加上后数据类型也不能改变,唯一能改变的是增加varchar类型的长度。
7. 添加主键
ALTER TABLE tabname ADD PRIMARY KEY (col)
删除主键
ALTER TABLE tabname DROP PRIMARY KEY (col)
8. 创建索引
create [unique] index idxname on tabname(col….)
删除索引
drop index idxname
注:索引是不可更改的,想更改必须删除重新建。
9. 创建视图
create view viewname as select statement
删除视图
drop view viewname
10. 几个简单的基本sql语句
选择
select * from table1 where 范围
插入
insert into table1(field1,field2) values(value1,value2)
删除
delete from table1 where 范围
更新
update table1 set field1=value1 where 范围
查找
select * from table1 where field1 like ’%value1%’
排序
select * from table1 order by field1,field2 [desc]
总数
select count as totalcount from table1
求和
select sum(field1) as sumvalue from table1
平均
select avg(field1) as avgvalue from table1
最大
select max(field1) as maxvalue from table1
最小
select min(field1) as minvalue from table1
11. 几个高级查询运算词
- A: UNION 运算符
UNION 运算符通过组合其他两个结果表(例如 TABLE1 和 TABLE2)并消去表中任何重复行而派生出一个结果表。当 ALL 随 UNION一起使用时(即 UNION ALL),不消除重复行。两种情况下,派生表的每一行不是来自 TABLE1 就是来自 TABLE2。 - B: EXCEPT 运算符
EXCEPT 运算符通过包括所有在 TABLE1 中但不在 TABLE2 中的行并消除所有重复行而派生出一个结果表。当 ALL 随 EXCEPT 一起使用时 (EXCEPT ALL),不消除重复行。 - C: INTERSECT 运算符
INTERSECT 运算符通过只包括 TABLE1 和 TABLE2 中都有的行并消除所有重复行而派生出一个结果表。当 ALL 随 INTERSECT 一起使用时 (INTERSECT ALL),不消除重复行。注:使用运算词的几个查询结果行必须是一致的。
12. 使用外连接
A. left (outer) join:
左外连接(左连接):结果集几包括连接表的匹配行,也包括左连接表的所有行。select a.a, a.b, a.c, b.c, b.d, b.f from a LEFT OUT JOIN b ON a.a = b.c
B:right (outer) join:
右外连接(右连接):结果集既包括连接表的匹配连接行,也包括右连接表的所有行。C:full/cross (outer) join:
全外连接:不仅包括符号连接表的匹配行,还包括两个连接表中的所有记录。
13. 分组:Group by
一张表,一旦分组 完成后,查询后只能得到组相关的信息。
组相关的信息:(统计信息) count,sum,max,min,avg 分组的标准)
在SQLServer中分组时:不能以text,ntext,image类型的字段作为分组依据
在selecte统计函数中的字段,不能和普通的字段放在一起;
14. 对数据库进行操作
分离数据库: sp_detach_db; 附加数据库:sp_attach_db 后接表明,附加需要完整的路径名
15. 如何修改数据库的名称
sp_renamedb ‘old_name’, ‘new_name’
16. 查询字段ID最大值的记录
无索引
select time from table t1 where not exists (select 1 from table where id > t1.id)
有索引
select time from table where id in (select max(id) from table)
17. 查询某一列的重复项内容
SELECT * FROM documentation AS A WHERE (SELECT COUNT(*) FROM documentation WHERE fileName=A.fileName)>1
18. 判断表中是否存在记录
SELECT COUNT (*) FROM tableName WHERE conditions
二. 提升篇
1. 复制表(只复制结构,源表名:a 新表名:b) (Access可用)
法一
select * into b from a where 1<>1(仅用于SQlServer)
法二
select top 0 * into b from a
2. 拷贝表(拷贝数据,源表名:a 目标表名:b) (Access可用)
insert into b(a, b, c) select d,e,f from b;
3. 跨数据库之间表的拷贝(具体数据使用绝对路径) (Access可用)
insert into b(a, b, c) select d,e,f from b in ‘具体数据库’ where 条件
例子:..from b in ‘“&Server.MapPath(“.”)&”\data.mdb” &”‘ where..
4. 子查询(表名1:a 表名2:b)
select a,b,c from a where a IN (select d from b ) 或者: select a,b,c from a where a IN (1,2,3)
5. 显示文章、提交人和最后回复时间
select a.title,a.username,b.adddate from table a,(select max(adddate) adddate from table where table.title=a.title) b
6. 外连接查询(表名1:a 表名2:b)
select a.a, a.b, a.c, b.c, b.d, b.f from a LEFT OUT JOIN b ON a.a = b.c
7. 在线视图查询(表名1:a )
select * from (SELECT a,b,c FROM a) T where t.a > 1;
8. between的用法,between限制查询数据范围时包括了边界值,not between不包括
select * from table1 where time between time1 and time2
select a,b,c, from table1 where a not between 数值1 and 数值2
9. in 的使用方法
select * from table1 where a [not] in (‘值1’,’值2’,’值4’,’值6’)
10. 两张关联表,删除主表中已经在副表中没有的信息
delete from table1 where not exists ( select * from table2 where table1.field1=table2.field1 )
11. 四表联查问题
select * from a left inner join b on a.a=b.b right inner join c on a.a=c.c inner join d on a.a=d.d where .....
12. 日程安排提前五分钟提醒
select * from 日程安排 where datediff('minute',f开始时间,getdate())>5
13. 一条sql语句搞定数据库分页
select top 10 b.* from (select top 20 主键字段,排序字段 from 表名 order by 排序字段 desc) a,表名 b where b.主键字段 = a.主键字段 order by a.排序字段
关于数据库分页:
declare @start int,@end int
@sql nvarchar(600)
set @sql=’select top’+str(@end-@start+1)+’+from T where rid not in(select top’+str(@str-1)+’Rid from T where Rid>-1)’
exec sp_executesql @sql
注意:在top后不能直接跟一个变量,所以在实际应用中只有这样的进行特殊的处理。Rid为一个标识列,如果top后还有具体的字段,这样做是非常有好处的。因为这样可以避免 top的字段如果是逻辑索引的,查询的结果后实际表中的不一致(逻辑索引中的数据有可能和数据表中的不一致,而查询时如果处在索引则首先查询索引)
14. 前10条记录
select top 10 * form table1 where 范围
15. 选择在每一组b值相同的数据中对应的a最大的记录的所有信息(类似这样的用法可以用于论坛每月排行榜,每月热销产品分析,按科目成绩排名,等等.)
select a,b,c from tablename ta where a=(select max(a) from tablename tb where tb.b=ta.b)
16. 包括所有在 TableA 中但不在 TableB和TableC 中的行并消除所有重复行而派生出一个结果表
(select a from tableA ) except (select a from tableB) except (select a from tableC)
17. 随机取出10条数据
select top 10 * from tablename order by newid()
18. 随机选择记录
select newid()
19. 删除重复记录
方法一:
delete from tablename where id not in (select max(id) from tablename group by col1,col2,…)
方法二:
select distinct * into temp from tablename
delete from tablename
insert into tablename select * from temp
评价: 这种操作牵连大量的数据的移动,这种做法不适合大容量的数据操作
方法三: 例如:在一个外部表中导入数据,由于某些原因第一次只导入了一部分,但很难判断具体位置,这样只有在下一次全部导入,这样也就产生好多重复的字段,怎样删除重复字段
alter table tablename
add column_b int identity(1,1) –添加一个自增列
delete from tablename where column_b not in(
select max(column_b) from tablename group by column1,column2,…)
alter table tablename drop column column_b
20. 列出数据库里所有的表名
select name from sysobjects where type='U' // U代表用户
21. 列出表里的所有的列名
select name from syscolumns where id=object_id('TableName')
22. 列示type、vender、pcs字段,以type字段排列,case可以方便地实现多重选择,类似select 中的case。
1 | select type,sum(case vender when 'A' then pcs else 0 end),sum(case vender when 'C' then pcs else 0 end),sum(case vender when 'B' then pcs else 0 end) FROM tablename group by type |
显示结果:
type | vender | pcs |
---|---|---|
电脑 | A | 1 |
电脑 | A | 1 |
光盘 | B | 2 |
光盘 | A | 2 |
手机 | B | 3 |
手机 | C | 3 |
23. 初始化表
1 | TRUNCATE TABLE table1 |
24. 选择从10到15的记录
1 | select top 5 * from (select top 15 * from table order by id asc) table_别名 order by id desc |
25. 临时表
创建临时表
- 方法一:例:
1
2
3
4
5
6create table #临时表名(字段1 约束条件,
字段2 约束条件,
.....)
create table ##临时表名(字段1 约束条件,
字段2 约束条件,
.....)1
2
3
4
5
6create table #Tmp --创建临时表#Tmp
(
ID int IDENTITY (1,1) not null, --创建列ID,并且每次新增一条记录就会加1
WokNo varchar(50),
primary key (ID) --定义ID为临时表#Tmp的主键
); - 方法二:
1
2select * into #临时表名 from 你的表;
select * into ##临时表名 from 你的表;注:以上的#代表局部临时表,##代表全局临时表
查询临时表
1 | select * from #临时表名; |
删除临时表
1 | drop table #临时表名; |
26. 关联表更新
SQL SERVER
1 | UPDATE A SET A.b=B.d FROM tableA A INNER JOIN tableB B ON A.a = B.a; |
MYSQL
1 | UPDATE tableA A INNER JOIN tableB B ON A.a = B.a SET A.b=B.d; |
27. 查看表的所有外键关系
1 | select t1.*,t2.name,t3.name from dbo.sysforeignkeys t1 left join sysobjects t2 on t1.fkeyid=t2.id |
28. 删除所有约束
1 | DECLARE c1 cursor for |
29. 删除数据库所有表
1 | declare @tname varchar(8000) |
30. 删除外键约束
得到某个表被哪些外键引用,并且显示出外键表的表名
1 | SELECT |
通过外键表的表名和外键名称执行以下语句即可删除外键
1 | ALTER TABLE dbo.CONTRACTINFO DROP CONSTRAINT FK_CONTRACTINFO_ATTACHMENTDOC |
31. 查找指定数据库表的列名、数据类型、数据长度、注释
SQL SERVER 2000
1 | SELECT sysobjects.name AS 表名, syscolumns.name AS 列名, systypes.name AS 数据类型, syscolumns.length AS 数据长度, sysproperties.[value] AS 注释 |
SQL SERVER 2005及以上
1 | SELECT ta.name AS 表名, c.name AS 列名, t.name AS 数据类型, c.max_length AS 数据长度, ex.[value] AS 注释 |
32. 同步表数据
1 | INSERT 表2 |
33. 判断某列中是否包含中文字符或者英文字符
1 | SELECT * FROM 表名 WHERE 某列 LIKE '%[吖-座]%' |
34. 行转列,将多行数据合并成一行(SQL SERVER 2005以上支持)
例如,表Table1中有两列数据:
|code |name|
|——- |—|
|AAA |姓名1|
|AAA |姓名2|
|AAA |姓名3|
|BBB |姓名4|
|BBB |姓名5|
行转列,想变成如下格式:
|code |name|
|——- |—|
|AAA |姓名1,姓名2,姓名3|
|BBB |姓名4,姓名5|
可用如下SQL实现:
1 | SELECT code, name = (STUFF((SELECT ',' + name |
35. 去除字段中空格、换行符、回车符
1 | UPDATE xx_sjb SET sl = LTRIM(RTRIM(REPLACE(REPLACE(sl, CHAR(10), ''), CHAR (13), ''))) |
三. 技巧篇
1. 1=1,1=2的使用,在SQL语句组合时用的较多
“where 1=1” 表示选择全部
“where 1=2” 表示全部不选
2. 收缩数据库
- 重建索引
1
2DBCC REINDEX
DBCC INDEXDEFRAG - 收缩数据和日志
1
2DBCC SHRINKDB
DBCC SHRINKFILE
3. 压缩数据库
1 | dbcc shrinkdatabase(dbname) |
4. 转移数据库给新用户以已存在用户权限
1 | exec sp_change_users_login 'update_one','newname','oldname' |
5. 检查备份集
1 | RESTORE VERIFYONLY from disk='E:\dvbbs.bak' |
6. 修复数据库
1 | ALTER DATABASE [dvbbs] SET SINGLE_USER |
7. 日志清除
1 | SET NOCOUNT ON |
8. 更改某个表
1 | exec sp_changeobjectowner 'tablename','dbo' |
9. 存储更改全部表
1 | CREATE PROCEDURE dbo.User_ChangeObjectOwnerBatch |
10. SQL SERVER中直接循环写入数据
1 | declare @i int |
案例:有如下表,要求就表中所有沒有及格的成绩,在每次增长0.1的基础上,使他们刚好及格:
| Name | score |
| ——- | —: |
|Zhangshan |80|
|Lishi |59|
|Wangwu |50|
|Songquan |69|
1 | while((select min(score) from tb_table)<60) |
11. 查看数据库属性
1 | sp_helpdb 数据库名 |
12. 按姓氏笔画排序:
1 | Select * From TableName Order By CustomerName Collate Chinese_PRC_Stroke_ci_as //从少到多 |
13. 数据库加密:
1 | select encrypt('原始密码') |
14. 取回表中字段:
1 | declare @list varchar(1000), |
15. 查看硬盘分区:
1 | EXEC master..xp_fixeddrives |
16. 比较A,B表是否相等:
1 | if (select checksum_agg(binary_checksum(*)) from A) |
17. 杀掉所有的事件探察器进程:
1 | DECLARE hcforeach CURSOR GLOBAL FOR SELECT 'kill '+RTRIM(spid) FROM master.dbo.sysprocesses |
18. 记录搜索:
- 开头到N条记录
1
Select Top N * From 表
- N到M条记录(要有主索引ID)
1
Select Top M-N * From 表 Where ID in (Select Top M ID From 表) Order by ID Desc
- N到结尾记录
1
Select Top N * From 表 Order by ID Desc
19. 获取当前数据库中的所有用户表
1 | select Name from sysobjects where xtype='u' and status>=0 |
20. 获取某一个表的所有字段
1 | select name from syscolumns where id=object_id('表名') |
21. 查看与某一个表相关的视图、存储过程、函数
1 | select a.* from sysobjects a, syscomments b where a.id = b.id and b.text like '%表名%' |
22. 查看当前数据库中所有存储过程
1 | select name as 存储过程名称 from sysobjects where xtype='P' |
23. 查询用户创建的所有数据库
1 | select * from master..sysdatabases D where sid not in(select sid from master..syslogins where name='sa') |
24. 查询某一个表的字段和数据类型
1 | select column_name,data_type from information_schema.columns |
25. 不同服务器数据库之间的数据操作
25.1 创建链接服务器
- 创建一个链接名
1
EXEC sp_addlinkedserver 'LinkName ', ' ', 'SQLOLEDB ', '远程服务器名或IP地址 ' --有自定义实例名时要加上"/实例名"
- 创建登录信息(或叫创建链接服务器登录名映射)(只需选择一种方式即可)
- 以windows认证的方式登录
1
EXEC sp_addlinkedsrvlogin 'LinkName' --或EXEC sp_addlinkedsrvlogin 'LinkName','true'
- 以SQL认证的方式登录
1
EXEC sp_addlinkedsrvlogin 'LinkName ', 'false ',null, '用户名 ', '密码 '
- 以windows认证的方式登录
25.2 链接服务器相关数据操作
- 查询示例
1
SELECT * FROM LinkName.数据库名.架构名.表名
25.3 删除链接服务器
- 删除登录信息(或叫删除链接服务器登录名映射)
1
EXEC sp_droplinkedsrvlogin 'LinkName',NULL
- 删除链接服务器名称
1
EXEC sp_dropserver 'LinkName ', 'droplogins' --如果指定droplogins,则在删除链接服务器之前要删除登录名映射
25.4 其它连接远程/局域网数据方法:OPENROWSET/OPENQUERY/opendatasource
OPENROWSET方法(不需要用到创建好的链接名。如果连接的实例名不是默认的,需要在”sql服务器名或IP地址”后加上”/实例名”)
- 查询示例
- Windows认证方式查询(以下方法之一即可)
1
2
3
4SELECT * FROM OPENROWSET('SQLOLEDB', 'server=sql服务器名或IP地址;Trusted_Connection=yes',数据库名.架构名.表名)
SELECT * FROM OPENROWSET('SQLNCLI', 'server=sql服务器名或IP地址;Trusted_Connection=yes',数据库名.架构名.表名)
SELECT * FROM OPENROWSET('SQLOLEDB', 'server=sql服务器名或IP地址;Trusted_Connection=yes','SELECT * FROM 数据库名.架构名.表名')
SELECT * FROM OPENROWSET('SQLNCLI', 'server=sql服务器名或IP地址;Trusted_Connection=yes','SELECT * FROM 数据库名.架构名.表名') - SQL认证方式查询(以下方法之一即可)
1
2
3
4
5
6
7
8SELECT * FROM OPENROWSET('SQLOLEDB', 'server=sql服务器名或IP地址;uid=用户名;pwd=密码',数据库名.架构名.表名)
SELECT * FROM OPENROWSET('SQLNCLI', 'server=sql服务器名或IP地址;uid=用户名;pwd=密码',数据库名.架构名.表名)
SELECT * FROM OPENROWSET('SQLOLEDB', 'server=sql服务器名或IP地址;uid=用户名;pwd=密码','SELECT * FROM 数据库名.架构名.表名')
SELECT * FROM OPENROWSET('SQLNCLI', 'server=sql服务器名或IP地址;uid=用户名;pwd=密码','SELECT * FROM 数据库名.架构名.表名')
SELECT * FROM OPENROWSET('SQLOLEDB', 'sql服务器名';'用户名';'密码',数据库名.架构名.表名)
SELECT * FROM OPENROWSET('SQLNCLI', 'sql服务器名';'用户名';'密码',数据库名.架构名.表名)
SELECT * FROM OPENROWSET('SQLOLEDB', 'sql服务器名';'用户名';'密码','SELECT * FROM 数据库名.架构名.表名')
SELECT * FROM OPENROWSET('SQLNCLI', 'sql服务器名';'用户名'; '密码','SELECT * FROM 数据库名.架构名.表名')
- Windows认证方式查询(以下方法之一即可)
- 生成本地表
1
SELECT * INTO 表 FROM openrowset( 'SQLOLEDB ', 'sql服务器名 '; '用户名 '; '密码 ',数据库名.dbo.表名)
- 把本地表导入远程表
1
2INSERT OPENROWSET('SQLOLEDB', 'server=sql服务器名或IP地址;uid=用户名;pwd=密码',数据库名.架构名.表名)
SELECT * FROM 本地表 - 关联更新本地表
1
2
3
4UPDATE b
SET b.列A = a.列A
FROM OPENROWSET( 'SQLOLEDB ', 'sql服务器名 '; '用户名 '; '密码 ',数据库名.dbo.表名) AS a
INNER JOIN 本地表 b ON a.column1 = b.column1
- 查询示例
OPENQUERY方法(需要先创建一个链接)
- 查询示例
1
SELECT * FROM OPENQUERY(LinkName,'SELECT * FROM 数据库名.架构名.表名')
- 把本地表导入远程表
1
INSERT OPENQUERY(LinkName, 'SELECT * FROM 数据库名.架构名.表名') SELECT * FROM 本地表
- 关联更新本地表
1
2
3
4UPDATE b
SET b.列A = a.列A
FROM OPENQUERY(ITSV, 'SELECT * FROM 数据库.dbo.表名 ') AS a
INNER JOIN 本地表 b ON a.column1 = b.column1
- 查询示例
OPENDATASOURCE方法(不需要用到创建好的链接名。如果连接的实例名不是默认的,需要在”sql服务器名或IP地址”后加上”/实例名”)
- 查询示例
- Windows认证方式查询(以下方法之一即可)
1
2SELECT * FROM OPENDATASOURCE('SQLOLEDB', 'server=sql服务器名或IP地址;Trusted_Connection=yes').数据库名.架构名.表名
SELECT * FROM OPENDATASOURCE('SQLNCLI', 'server=sql服务器名或IP地址;Trusted_Connection=yes').数据库名.架构名.表名 - SQL认证方式查询(以下方法之一即可)
1
2
3
4SELECT * FROM OPENDATASOURCE('SQLOLEDB', 'server=sql服务器名或IP地址;uid=用户名;pwd=密码').数据库名.架构名.表名
SELECT * FROM OPENDATASOURCE('SQLNCLI', 'server=sql服务器名或IP地址;uid=用户名;pwd=密码').数据库名.架构名.表名
SELECT * FROM OPENDATASOURCE('SQLOLEDB', 'Data Source=sql服务器名或IP地址;uid=用户名;pwd=密码').数据库名.架构名.表名
SELECT * FROM OPENDATASOURCE('SQLNCLI', 'Data Source=sql服务器名或IP地址;uid=用户名;pwd=密码').数据库名.架构名.表名
- Windows认证方式查询(以下方法之一即可)
- 导入示例
1
2INSERT OPENDATASOURCE('SQLOLEDB', 'server=sql服务器名或IP地址;uid=用户名;pwd=密码').数据库名.架构名.表名
SELECT * FROM 本地表如果出现如下错误:SQL Server 阻止了对组件“Ad Hoc Distributed Queries”的STATEMENT“OpenRowset/OpenDatasource”的访问,因为此组件已作为此服务器安全配置的一部分而被关闭。系统管理员可以通过使用 sp_configure 启用“Ad Hoc Distributed Queries”。有关启用“Ad Hoc Distributed Queries”
则解决办法是参照下面命令:
- 查询示例
- 开启Ad Hoc Distributed Queries组件,在sql查询编辑器中执行如下语句:
1
2
3
4exec sp_configure 'show advanced options',1
reconfigure
exec sp_configure 'Ad Hoc Distributed Queries',1
reconfigure - 关闭Ad Hoc Distributed Queries组件,在sql查询编辑器中执行如下语句:
1
2
3
4exec sp_configure 'Ad Hoc Distributed Queries',0
reconfigure
exec sp_configure 'show advanced options',0
reconfigure
26. 删除数据库下面的所有表
1 | use 数据库名(是要删除表的所在的那个数据库的名称) |
27. 查看数据库文件使用情况
1 | SELECT A.name AS "逻辑名称" |
28. 修改sa的密码
1 | EXEC sp_password 'OldPassword','NewPassword','sa'; |
不知道OldPassword时,可用NULL代替。
29. 查询表的索引
1 | SELECT 表名 = c.name, |
30. 查询没有索引的表
1 | SELECT * |
常识
- 在SQL查询中,
FROM
后最多可以跟256张表或视图 - 在SQL语句中使用
ORDER BY
查询时,先排序,后取值 - 在SQL中,一个字段的最大容量是8000