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是某编号你希望是连续的那就很膈应了。

image.png

你可以看到上面的例子,执行多次后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类型时

image.png

image.png  

输出

image.png 

 显示正常,十分具有迷惑性,开发的时候你以为是没问题的,

但是

image.png

输出

image.png 

150.12 显示为 15.119999999999999

这是太坑了,我们的项目是重构一个用了15年+的项目,库不变只做程序,项目里见到最多的字段就是金额小数点,做到后面发现这个问题,所有输出的位置到处都得格式化去改!

还有一个问题是,如果你希望的是x.xx这种保留两位的格式,不好意思,不建议用float类型!因为12.00 进库即存为12,出库也12,不格式化没法是12.00!

对比之下mysql是没有问题的,因为是是这样定义的float(9,2)保留两位!


二、字段类型是 decimal(18, 2) 类型时

0.00 显示为 .00

image.png

输出:

image.png

导致于 所有显示地方都得把 .00 换成 0.00。

(后发现可能是php-SQLserver驱动的兼容问题一个项目用的pdo_dblib显示是好的,一个用的pdo_sqlsrv显示不正常,可能跟驱动版本有关系,反正用php+SQLserver的小伙伴格外注意一下),

阅读全文 »

SQLServer 大坑之数字乘除计算

发表日期:2022-08-15 12:16:52 | 来源: | 分类:SQLServer

10 / 4 * 4 = ?  , 小学毕业都知道应该还是10,

我们用mysql试一下:

image.png

依然是10,其实当计算出小数,mysql会自动转型为float,但是注意,SQLserver不会!

image.png

得出的结果是8! 为什么是8呢,有点离谱。

因为SQLserver是强类型的,10/4 = 2 ,2*4=8,因此如果你的字段是int类型的你想在SQL里做乘除那你可得想好了,记得先把计算的字段转型为float

image.png

阅读全文 »

SQLServer 大坑之null 判断

发表日期:2022-08-15 12:00:34 | 来源: | 分类:SQLServer

例如

image.png

三条数据,我们找出 misMoney是null的

image.png

然而一条也没出来!

mysql 的null 你写 where a = null / where a <> null 都没毛病,当然我一直也认为其实能这样判断是最好最方便,但是SQLserver不支持!

尤其现在绝大多数程序员及项目都用的是mysql,所以很多程序员没用过甚至不了解SQLserver。我面试过很多程序员,答对的很少,全然不知SQL语句 null 的正确判断写法。

正确写法

 image.png

一定要有 is null  或 is not null !

参考另一篇文章:

SQLServer大坑之<>判断

阅读全文 »

SQLServer 大坑之不等于判断

发表日期:2022-08-15 11:50:20 | 来源: | 分类:SQLServer

例如有三条数据:

image.png

我们想找出 posMoney和misMoney不一致的数据,逻辑很简单 我们通常想到且写的是:

where posMoney <> misMoney ,理论上出的结果是王五和李四。

然后我们看结果

image.png

只有一条王五!!  20000和null 的竟然出不来!

这就是SQLserver的大坑,一定要注意,mysql是没有问题的,但是SQLserver是不能和可能存在null列进行对比,

当然我事先是知道这个的,但是!!有时候我们认为的数据实际不应该有null,但现实是前几天发现业务系统里这列数据居然因为某些异常情况存在很多null值,以至于我这样写成为了被错误。这是最坑的,这才是更是应该注意的,所以代码最好把所有可能性想到,避免让别人的错误成为自己的错。

所以,在SQLserver里如果是金额等严谨性判断<> 最好这样写:

image.png

不管你认为可不可能有null 总之先把null替换为数字0或空字符串'',然后再对比 <>,多么痛的领悟……

阅读全文 »

全部博文(1580)
集速网 copyRight © 2015-2022 宁ICP备15000399号-1 宁公网安备 64010402001209号
与其临渊羡鱼,不如退而结网
欢迎转载、分享、引用、推荐、收藏。