SQLServer 业 ,精于勤 荒于嬉.
- SQLServer 触发器详解
-
发表日期:2022-10-19 11:20:47 | 来源: | 分类:SQLServer
-
示例1
create TRIGGER [dbo].[trigger_House(触发器的名字)] ON [dbo].[House(监听的表名)] AFTER INSERT, UPDATE -- 插入/修改后触发 ,delete 删除后触发 AS BEGIN SET NOCOUNT ON; -- 你的业务逻辑 update House set modify_microtime = getdate() where HID in (select HID from inserted); -- 注意这句非常重要 where HID in (select HID from inserted) 查找更新的记录行 -- 如果House表一次更新的是多行,那么触发的inserted 表也是多行,所以这里用了where in 而不是 = END
示例2
drop trigger 触发器名字
- SQLServer 自定义函数详解
-
发表日期:2022-08-22 11:12:52 | 来源: | 分类:SQLServer
-
示例1
CREATE FUNCTION [dbo].[函数名] ( @Id int,@date1 datetime --入参 ) RETURNS varchar(1000) -- 返回值 AS BEGIN declare @Str varchar(8000) -- 业务逻辑 -- SQL思路拔高可以参考存储过程那篇文章,写的非常细 -- http://www.canquick.com/article/ARTICLE_B176685DBB653F60CE765750.html RETURN @Str -- 返回值 END go grant execute, view definition on 函数名to [] go
示例2
CREATE FUNCTION [dbo].[MaxNumber]( @no1 float, @no2 float, @no3 float, @no4 float, @no5 float ) RETURNS float AS BEGIN declare @maxno float if isnull(@no1, 0) > isnull(@no2, 0) set @maxno = isnull(@no1, 0) else set @maxno = isnull(@no2, 0) if isnull(@maxno, 0) < isnull(@no3, 0) set @maxno = isnull(@no3, 0) if isnull(@maxno, 0) < isnull(@no4, 0) set @maxno = isnull(@no4, 0) if isnull(@maxno, 0) < isnull(@no5, 0) set @maxno = isnull(@no5, 0) return @maxno END go
- SQLServer 大坑之分页
-
发表日期:2022-08-20 10:46:18 | 来源: | 分类:SQLServer
-
示例1
SELECT T1.* FROM (SELECT result.*, ROW_NUMBER() OVER ( ORDER BY rand()) AS ROW_NUMBER FROM (SELECT * FROM [Users] WHERE ustate = 1) AS result ) AS T1 WHERE (T1.ROW_NUMBER BETWEEN 1 AND 10)
示例2
SELECT top 10 * FROM [Users] WHERE ustate = 1 and uid not in( SELECT top ( N * 10) UID FROM [Users] WHERE ustate = 1 --N:0第一页 1第二页 类推 )
示例3
select top 10 * from Users where uid > ( select isnull(max(uid),0) from ( select top N uid from Users order by uid --N:0第一页 1第二页 类推 ) A ) order by uid
- SQLServer Link DB Server
-
发表日期:2022-08-20 09:56:53 | 来源: | 分类:SQLServer
-
示例1
--查看当前链接情况: select * from sys.servers; --使用 sp_helpserver 来显示可用的服务器 Exec sp_helpserver --显示使用sp_addlinkedserver来增加服务器链接 EXEC sp_addlinkedserver @server='LINKABC',--被访问的服务器别名 @srvproduct='', @provider='SQLOLEDB', @datasrc='192.168.0.10\sql2008' --要访问的服务器IP[\实例名有就写没有] --然后使用sp_addlinkedsrvlogin 来增加用户登录链接 EXEC sp_addlinkedsrvlogin 'LINKABC', --被访问的服务器别名 'false', NULL, 'root', --帐号 '123456' --密码 访问: select * from [LINKABC].[abcDB].dbo.Users where uid = 1; 也支持IP形式: select * from [192.168.0.10].[abcDB].dbo.Users where uid = 1; -- 删除数据库链接 --与创建相反,要先删除登录账户 exec sp_droplinkedsrvlogin 'LINKABC',null --然后再删除数据库链接 exec sp_dropserver 'LINKABC'
- SQLServer 存储过程详解
-
发表日期:2022-08-19 15:57:32 | 来源: | 分类:SQLServer
-
示例1
-- 如果没有参数 可以不写参数 CREATE PROCEDURE [存储名] @参数1 int, @参数2 varchar(50), @参数3 varchar(2000),@参数4 varchar(100) as -- 存储的主体 go
示例2
exec [存储过程名字] @参数1=1,@参数2='aaa', @参数3sql='bbbb',@参数4where ='ccccc'; 或者简写,即按字段顺序 exec [存储过程名字] 1,'aaa', 'bbbb','ccccc'; exec 也可以写成全称 execute
示例3
-- 参数可以传入SQL语句直接在存储过程里执行,通常不建议这样干! exec (@参数); -- 传入的参数也可以是半截sql,然后再拼起来执行,通常不建议这样干! -- 而且通常这么干都是在代码里把接收到的参数拼起来传,十有八九有SQL注入漏洞 exec ('delete from TestTab2 where a=b and d=c and '+@参数4where);
示例4
-- 定义临时表 DECLARE @table_tmp table ( a int, b VARCHAR(50), c int, d int, e VARCHAR(50), f VARCHAR(50), g DECIMAL DEFAULT 0 ); -- 写法用法和普通表没什么区别,考验你的SQL基础能力罢了 insert into @table_tmp (a, b, c, d) select uid,username,age,10 from Users where type=3; update @table_tmp set e=(select role_name from user_profile where uid = a) where age > 18; UPDATE t set t.f = r.role_name FROM @table_resoult as t, user_role as r where r.uid = t.a; update @table_tmp set g=(select sum(money) from user_account where uid = a) where g = 0; update c set c.d=b.score from @table_tmp c join ( select * from xxxxx ) b on c.a = b.uid and c.e = b.role; --最后达到你想要的结果集了,你可以把虚拟表的数据插入到目标真实表 insert into targetTab(a1,b1,c1,d1,e1,f1,g1) select a,b,c,d,e,f,g from @table_tmp; -- 或者存储过程直接返回临时表的结果 select * from @table_tmp;
示例5
--可以定义变量 declare @var1 varchar(50) --给定义的变量赋值 select @username = username from users where UID = 1; --使用变量 update user_house set update_time=getdate(), username=@username where uid = 1;
示例6
--开启事务 begin tran -- 这里进行业务操作 -- …… --如果有错 if (0 <> @@ERROR) begin --事务回滚 rollback tran; -- 返回错误码 select -1; end else begin --提交事务 commit tran; -- 可以返回修改行数或 其它内容 select @@ROWCOUNT; end
示例7
--定义变量 declare @uid int,@username varchar(50); -- 定义游标 填充数据 declare c1 cursor for select UID,UName from Users where uuid <> 0; --开启游标 open c1 aa:fetch next from c1 into @uid,@username -- 循环 while @@fetch_Status = 0 begin -- 业务逻辑 end -- 关闭游标 close c1 -- 销毁游标 deallocate c1
- SQLServer 大坑之事务回滚占自增ID
-
发表日期:2022-08-18 11:33:59 | 来源: | 分类:SQLServer
-
主键如果是自增的,当你使用事务时,即便是事务回滚了,你的下一个ID会被占用,再次插入会跳一个数。
这个大坑在一些业务场景下极其讨厌,比如这张表的ID是某编号你希望是连续的那就很膈应了。
你可以看到上面的例子,执行多次后ID全是间隔的,就是因为回滚事务的那条也占用了ID自增。
注意:mysql也是如此
- SQLServer 存储报错:SQLSTATE[IMSSP]: The active result for the query contains no fields.
-
发表日期:2022-08-16 17:00:00 | 来源: | 分类:SQLServer
-
存储报错:SQLSTATE[IMSSP]: The active result for the query contains no fields.
原因:
一般执行execute语句,也就是增删改语句会返回修改记录数的行数。执行select 是返回记录。
存储里可能是先执行了增删改,最后是select 返回结果,就会报这个错。
在存储里增加这句话,让存储不返回影响的记录数就可以了。
SET NOCOUNT ON
- SQLServer 大坑之导入数据
-
发表日期:2022-08-16 16:49:34 | 来源: | 分类:SQLServer
-
Cannot insert explicit value for identity column in table 'test' when IDENTITY_INSERT is set to OFF
这句报错的大致意思就是这张表有自增列,不能直接导入。
SQLServer 导入表,如果这张表的ID是自增那就非常烦,导入会报错。提示你必须先允许插入自增,插入完成再关闭。
set IDENTITY_INSERT 表名 on
insert into ……
insert into ……
set IDENTITY_INSERT 表名 off
当然要是一张表还好办,我要导入N张表,可是把我烦死了,而且单表SQL文件特别大,打开都费劲。
- SQLServer 大坑之浮点数显示问题
-
发表日期:2022-08-15 16:47:01 | 来源: | 分类:SQLServer
-
以下情况在php+SQLserver下经过验证是存在问题的,其它语言请自酌。
一、字段类型是float类型时
输出
显示正常,十分具有迷惑性,开发的时候你以为是没问题的,
但是
输出
150.12 显示为 15.119999999999999
这是太坑了,我们的项目是重构一个用了15年+的项目,库不变只做程序,项目里见到最多的字段就是金额小数点,做到后面发现这个问题,所有输出的位置到处都得格式化去改!
还有一个问题是,如果你希望的是x.xx这种保留两位的格式,不好意思,不建议用float类型!因为12.00 进库即存为12,出库也12,不格式化没法是12.00!
对比之下mysql是没有问题的,因为是是这样定义的float(9,2)保留两位!
二、字段类型是 decimal(18, 2) 类型时
0.00 显示为 .00
输出:
导致于 所有显示地方都得把 .00 换成 0.00。
(后发现可能是php-SQLserver驱动的兼容问题一个项目用的pdo_dblib显示是好的,一个用的pdo_sqlsrv显示不正常,可能跟驱动版本有关系,反正用php+SQLserver的小伙伴格外注意一下),
- SQLServer 大坑之数字乘除计算
-
发表日期:2022-08-15 12:16:52 | 来源: | 分类:SQLServer
-
10 / 4 * 4 = ? , 小学毕业都知道应该还是10,
我们用mysql试一下:
依然是10,其实当计算出小数,mysql会自动转型为float,但是注意,SQLserver不会!
得出的结果是8! 为什么是8呢,有点离谱。
因为SQLserver是强类型的,10/4 = 2 ,2*4=8,因此如果你的字段是int类型的你想在SQL里做乘除那你可得想好了,记得先把计算的字段转型为float
- SQLServer 大坑之null 判断
-
发表日期:2022-08-15 12:00:34 | 来源: | 分类:SQLServer
-
例如
三条数据,我们找出 misMoney是null的
然而一条也没出来!
mysql 的null 你写 where a = null / where a <> null 都没毛病,当然我一直也认为其实能这样判断是最好最方便,但是SQLserver不支持!
尤其现在绝大多数程序员及项目都用的是mysql,所以很多程序员没用过甚至不了解SQLserver。我面试过很多程序员,答对的很少,全然不知SQL语句 null 的正确判断写法。
正确写法
一定要有 is null 或 is not null !
参考另一篇文章:
- SQLServer 大坑之不等于判断
-
发表日期:2022-08-15 11:50:20 | 来源: | 分类:SQLServer
-
例如有三条数据:
我们想找出 posMoney和misMoney不一致的数据,逻辑很简单 我们通常想到且写的是:
where posMoney <> misMoney ,理论上出的结果是王五和李四。
然后我们看结果
只有一条王五!! 20000和null 的竟然出不来!
这就是SQLserver的大坑,一定要注意,mysql是没有问题的,但是SQLserver是不能和可能存在null列进行对比,
当然我事先是知道这个的,但是!!有时候我们认为的数据实际不应该有null,但现实是前几天发现业务系统里这列数据居然因为某些异常情况存在很多null值,以至于我这样写成为了被错误。这是最坑的,这才是更是应该注意的,所以代码最好把所有可能性想到,避免让别人的错误成为自己的错。
所以,在SQLserver里如果是金额等严谨性判断<> 最好这样写:
不管你认为可不可能有null 总之先把null替换为数字0或空字符串'',然后再对比 <>,多么痛的领悟……