SQL经典短小代码收集1

 1、固定列数的行列转换


student subject grade
--------- ---------- --------
student1 语文 80
student1 数学 70
student1 英语 60
student2 语文 90
student2 数学 80
student2 英语 100
……
转换为
语文 数学 英语
student1 80 70 60
student2 90 80 100
……
语句如下:select student,
sum(decode(subject,'语文', grade,null)) "语文",
sum(decode(subject,'数学', grade,null)) "数学",
sum(decode(subject,'英语', grade,null)) "英语"
from table
group by student;

-----------------------------------------------------
2、

--我来一段

--SQL2005

CREATE TABLE #t (id int IDENTITY(1,1) PRIMARY KEY,nums int)
GO

INSERT INTO #t
SELECT abs(CHECKSUM(NEWID())%100000)
GO 10000 ----生成10000笔资料,可以是任意一数字

SELECT * FROM #t
GO
DROP TABLE #t
GO


-----------------------------------------------------
SQL code
   
   

-- 日期转换参数,值得收藏
select CONVERT ( varchar , getdate (), 120 )
2004 - 09 - 12 11 : 06 : 08

select replace ( replace ( replace ( CONVERT ( varchar , getdate (), 120 ), ' - ' , '' ), ' ' , '' ), ' : ' , '' )
20040912110608

select CONVERT ( varchar ( 12 ) , getdate (), 111 )
2004 / 09 / 12

select CONVERT ( varchar ( 12 ) , getdate (), 112 )
20040912

select CONVERT ( varchar ( 12 ) , getdate (), 102 )
2004.09 . 12

其它我不常用的日期格式转换方法:

select CONVERT ( varchar ( 12 ) , getdate (), 101 )
09 / 12 / 2004

select CONVERT ( varchar ( 12 ) , getdate (), 103 )
12 / 09 / 2004

select CONVERT ( varchar ( 12 ) , getdate (), 104 )
12.09 . 2004

select CONVERT ( varchar ( 12 ) , getdate (), 105 )
12 - 09 - 2004

select CONVERT ( varchar ( 12 ) , getdate (), 106 )
12 09 2004

select CONVERT ( varchar ( 12 ) , getdate (), 107 )
09 12 , 2004

select CONVERT ( varchar ( 12 ) , getdate (), 108 )
11 : 06 : 08

select CONVERT ( varchar ( 12 ) , getdate (), 109 )
09 12 2004 1

select CONVERT ( varchar ( 12 ) , getdate (), 110 )
09 - 12 - 2004

select CONVERT ( varchar ( 12 ) , getdate (), 113 )
12 09 2004 1

select CONVERT ( varchar ( 12 ) , getdate (), 114 )
11 : 06 : 08.177

-----------------------------------------------------
SQL code
   
   

/*
标题:普通行列转换(version 2.0)
作者:爱新觉罗.毓华(十八年风雨,守得冰山雪莲花开)
时间:2008-03-09
地点:广东深圳
说明:普通行列转换(version 1.0)仅针对sql server 2000提供静态和动态写法,version 2.0增加sql server 2005的有关写法。

问题:假设有张学生成绩表(tb)如下:
姓名 课程 分数
张三 语文 74
张三 数学 83
张三 物理 93
李四 语文 74
李四 数学 84
李四 物理 94
想变成(得到如下结果):
姓名 语文 数学 物理
---- ---- ---- ----
李四 74 84 94
张三 74 83 93
-------------------
*/

create table tb(姓名 varchar ( 10 ) , 课程 varchar ( 10 ) , 分数 int )
insert into tb values ( ' 张三 ' , ' 语文 ' , 74 )
insert into tb values ( ' 张三 ' , ' 数学 ' , 83 )
insert into tb values ( ' 张三 ' , ' 物理 ' , 93 )
insert into tb values ( ' 李四 ' , ' 语文 ' , 74 )
insert into tb values ( ' 李四 ' , ' 数学 ' , 84 )
insert into tb values ( ' 李四 ' , ' 物理 ' , 94 )
go

-- SQL SERVER 2000 静态SQL,指课程只有语文、数学、物理这三门课程。(以下同)
select 姓名 as 姓名 ,
max ( case 课程 when ' 语文 ' then 分数 else 0 end ) 语文,
max ( case 课程 when ' 数学 ' then 分数 else 0 end ) 数学,
max ( case 课程 when ' 物理 ' then 分数 else 0 end ) 物理
from tb
group by 姓名

-- SQL SERVER 2000 动态SQL,指课程不止语文、数学、物理这三门课程。(以下同)
declare @sql varchar ( 8000 )
set @sql = ' select 姓名 '
select @sql = @sql + ' , max(case 课程 when ''' + 课程 + ''' then 分数 else 0 end) [ ' + 课程 + ' ] '
from ( select distinct 课程 from tb) as a
set @sql = @sql + ' from tb group by 姓名 '
exec ( @sql )

-- SQL SERVER 2005 静态SQL。
select * from ( select * from tb) a pivot ( max (分数) for 课程 in (语文,数学,物理)) b

-- SQL SERVER 2005 动态SQL。
declare @sql varchar ( 8000 )
select @sql = isnull ( @sql + ' ],[ ' , '' ) + 课程 from tb group by 课程
set @sql = ' [ ' + @sql + ' ] '
exec ( ' select * from (select * from tb) a pivot (max(分数) for 课程 in ( ' + @sql + ' )) b ' )

-- -------------------------------

/*
问题:在上述结果的基础上加平均分,总分,得到如下结果:
姓名 语文 数学 物理 平均分 总分
---- ---- ---- ---- ------ ----
李四 74 84 94 84.00 252
张三 74 83 93 83.33 250
*/

-- SQL SERVER 2000 静态SQL。
select 姓名 姓名,
max ( case 课程 when ' 语文 ' then 分数 else 0 end ) 语文,
max ( case 课程 when ' 数学 ' then 分数 else 0 end ) 数学,
max ( case 课程 when ' 物理 ' then 分数 else 0 end ) 物理,
cast ( avg (分数 * 1.0 ) as decimal ( 18 , 2 )) 平均分,
sum (分数) 总分
from tb
group by 姓名

-- SQL SERVER 2000 动态SQL。
declare @sql varchar ( 8000 )
set @sql = ' select 姓名 '
select @sql = @sql + ' , max(case 课程 when ''' + 课程 + ''' then 分数 else 0 end) [ ' + 课程 + ' ] '
from ( select distinct 课程 from tb) as a
set @sql = @sql + ' , cast(avg(分数*1.0) as decimal(18,2)) 平均分 , sum(分数) 总分 from tb group by 姓名 '
exec ( @sql )

-- SQL SERVER 2005 静态SQL。
select m. * , n.平均分 , n.总分 from
(
select * from ( select * from tb) a pivot ( max (分数) for 课程 in (语文,数学,物理)) b) m,
(
select 姓名 , cast ( avg (分数 * 1.0 ) as decimal ( 18 , 2 )) 平均分 , sum (分数) 总分 from tb group by 姓名) n
where m.姓名 = n.姓名

-- SQL SERVER 2005 动态SQL。
declare @sql varchar ( 8000 )
select @sql = isnull ( @sql + ' , ' , '' ) + 课程 from tb group by 课程
exec ( ' select m.* , n.平均分 , n.总分 from
(select * from (select * from tb) a pivot (max(分数) for 课程 in (
' + @sql + ' )) b) m ,
(select 姓名 , cast(avg(分数*1.0) as decimal(18,2)) 平均分 , sum(分数) 总分 from tb group by 姓名) n
where m.姓名 = n.姓名
' )

drop table tb

-- ----------------
--
----------------

/*
问题:如果上述两表互相换一下:即表结构和数据为:
姓名 语文 数学 物理
张三 74  83  93
李四 74  84  94
想变成(得到如下结果):
姓名 课程 分数
---- ---- ----
李四 语文 74
李四 数学 84
李四 物理 94
张三 语文 74
张三 数学 83
张三 物理 93
--------------
*/

create table tb(姓名 varchar ( 10 ) , 语文 int , 数学 int , 物理 int )
insert into tb values ( ' 张三 ' , 74 , 83 , 93 )
insert into tb values ( ' 李四 ' , 74 , 84 , 94 )
go

-- SQL SERVER 2000 静态SQL。
select * from
(
select 姓名 , 课程 = ' 语文 ' , 分数 = 语文 from tb
union all
select 姓名 , 课程 = ' 数学 ' , 分数 = 数学 from tb
union all
select 姓名 , 课程 = ' 物理 ' , 分数 = 物理 from tb
) t
order by 姓名 , case 课程 when ' 语文 ' then 1 when ' 数学 ' then 2 when ' 物理 ' then 3 end

-- SQL SERVER 2000 动态SQL。
--
调用系统表动态生态。
declare @sql varchar ( 8000 )
select @sql = isnull ( @sql + ' union all ' , '' ) + ' select 姓名 , [课程] = ' + quotename (Name , '''' ) + ' , [分数] = ' + quotename (Name) + ' from tb '
from syscolumns
where name! = N ' 姓名 ' and ID = object_id ( ' tb ' ) -- 表名tb,不包含列名为姓名的其它列
order by colid asc
exec ( @sql + ' order by 姓名 ' )

-- SQL SERVER 2005 动态SQL。
select 姓名 , 课程 , 分数 from tb unpivot (分数 for 课程 in ( [ 语文 ] , [ 数学 ] , [ 物理 ] )) t

-- SQL SERVER 2005 动态SQL,同SQL SERVER 2000 动态SQL。

-- ------------------
/*

问题:在上述的结果上加个平均分,总分,得到如下结果:
姓名 课程 分数
---- ------ ------
李四 语文 74.00
李四 数学 84.00
李四 物理 94.00
李四 平均分 84.00
李四 总分 252.00
张三 语文 74.00
张三 数学 83.00
张三 物理 93.00
张三 平均分 83.33
张三 总分 250.00
------------------
*/

select * from
(
select 姓名 as 姓名 , 课程 = ' 语文 ' , 分数 = 语文 from tb
union all
select 姓名 as 姓名 , 课程 = ' 数学 ' , 分数 = 数学 from tb
union all
select 姓名 as 姓名 , 课程 = ' 物理 ' , 分数 = 物理 from tb
union all
select 姓名 as 姓名 , 课程 = ' 平均分 ' , 分数 = cast ((语文 + 数学 + 物理) * 1.0 / 3 as decimal ( 18 , 2 )) from tb
union all
select 姓名 as 姓名 , 课程 = ' 总分 ' , 分数 = 语文 + 数学 + 物理 from tb
) t
order by 姓名 , case 课程 when ' 语文 ' then 1 when ' 数学 ' then 2 when ' 物理 ' then 3 when ' 平均分 ' then 4 when ' 总分 ' then 5 end

drop table tb




-----------------------------------------------------


SQL code
   
   

-- 按某一字段分组取最大(小)值所在行的数据
--
(爱新觉罗.毓华(十八年风雨,守得冰山雪莲花开) 2007-10-23于浙江杭州)
/*

数据如下:
name val memo
a 2 a2(a的第二个值)
a 1 a1--a的第一个值
a 3 a3:a的第三个值
b 1 b1--b的第一个值
b 3 b3:b的第三个值
b 2 b2b2b2b2
b 4 b4b4
b 5 b5b5b5b5b5
*/
-- 创建表并插入数据:
create table tb(name varchar ( 10 ),val int ,memo varchar ( 20 ))
insert into tb values ( ' a ' , 2 , ' a2(a的第二个值) ' )
insert into tb values ( ' a ' , 1 , ' a1--a的第一个值 ' )
insert into tb values ( ' a ' , 3 , ' a3:a的第三个值 ' )
insert into tb values ( ' b ' , 1 , ' b1--b的第一个值 ' )
insert into tb values ( ' b ' , 3 , ' b3:b的第三个值 ' )
insert into tb values ( ' b ' , 2 , ' b2b2b2b2 ' )
insert into tb values ( ' b ' , 4 , ' b4b4 ' )
insert into tb values ( ' b ' , 5 , ' b5b5b5b5b5 ' )
go

-- 一、按name分组取val最大的值所在行的数据。
--
方法1:
select a. * from tb a where val = ( select max (val) from tb where name = a.name) order by a.name
-- 方法2:
select a. * from tb a where not exists ( select 1 from tb where name = a.name and val > a.val)
-- 方法3:
select a. * from tb a,( select name, max (val) val from tb group by name) b where a.name = b.name and a.val = b.val order by a.name
-- 方法4:
select a. * from tb a inner join ( select name , max (val) val from tb group by name) b on a.name = b.name and a.val = b.val order by a.name
-- 方法5
select a. * from tb a where 1 > ( select count ( * ) from tb where name = a.name and val > a.val ) order by a.name
/*
name val memo
---------- ----------- --------------------
a 3 a3:a的第三个值
b 5 b5b5b5b5b5
*/

-- 二、按name分组取val最小的值所在行的数据。
--
方法1:
select a. * from tb a where val = ( select min (val) from tb where name = a.name) order by a.name
-- 方法2:
select a. * from tb a where not exists ( select 1 from tb where name = a.name and val < a.val)
-- 方法3:
select a. * from tb a,( select name, min (val) val from tb group by name) b where a.name = b.name and a.val = b.val order by a.name
-- 方法4:
select a. * from tb a inner join ( select name , min (val) val from tb group by name) b on a.name = b.name and a.val = b.val order by a.name
-- 方法5
select a. * from tb a where 1 > ( select count ( * ) from tb where name = a.name and val < a.val) order by a.name
/*
name val memo
---------- ----------- --------------------
a 1 a1--a的第一个值
b 1 b1--b的第一个值
*/

-- 三、按name分组取第一次出现的行所在的数据。
select a. * from tb a where val = ( select top 1 val from tb where name = a.name) order by a.name
/*
name val memo
---------- ----------- --------------------
a 2 a2(a的第二个值)
b 1 b1--b的第一个值
*/

-- 四、按name分组随机取一条数据。
select a. * from tb a where val = ( select top 1 val from tb where name = a.name order by newid ()) order by a.name
/*
name val memo
---------- ----------- --------------------
a 1 a1--a的第一个值
b 5 b5b5b5b5b5
*/

-- 五、按name分组取最小的两个(N个)val
select a. * from tb a where 2 > ( select count ( * ) from tb where name = a.name and val < a.val ) order by a.name,a.val
select a. * from tb a where val in ( select top 2 val from tb where name = a.name order by val) order by a.name,a.val
select a. * from tb a where exists ( select count ( * ) from tb where name = a.name and val < a.val having Count ( * ) < 2 ) order by a.name
/*
name val memo
---------- ----------- --------------------
a 1 a1--a的第一个值
a 2 a2(a的第二个值)
b 1 b1--b的第一个值
b 2 b2b2b2b2
*/

-- 六、按name分组取最大的两个(N个)val
select a. * from tb a where 2 > ( select count ( * ) from tb where name = a.name and val > a.val ) order by a.name,a.val
select a. * from tb a where val in ( select top 2 val from tb where name = a.name order by val desc ) order by a.name,a.val
select a. * from tb a where exists ( select count ( * ) from tb where name = a.name and val > a.val having Count ( * ) < 2 ) order by a.name
/*
name val memo
---------- ----------- --------------------
a 2 a2(a的第二个值)
a 3 a3:a的第三个值
b 4 b4b4
b 5 b5b5b5b5b5
*/
-- 七,如果整行数据有重复,所有的列都相同。
/*

数据如下:
name val memo
a 2 a2(a的第二个值)
a 1 a1--a的第一个值
a 1 a1--a的第一个值
a 3 a3:a的第三个值
a 3 a3:a的第三个值
b 1 b1--b的第一个值
b 3 b3:b的第三个值
b 2 b2b2b2b2
b 4 b4b4
b 5 b5b5b5b5b5
*/
-- 在sql server 2000中只能用一个临时表来解决,生成一个自增列,先对val取最大或最小,然后再通过自增列来取数据。
--
创建表并插入数据:
create table tb(name varchar ( 10 ),val int ,memo varchar ( 20 ))
insert into tb values ( ' a ' , 2 , ' a2(a的第二个值) ' )
insert into tb values ( ' a ' , 1 , ' a1--a的第一个值 ' )
insert into tb values ( ' a ' , 1 , ' a1--a的第一个值 ' )
insert into tb values ( ' a ' , 3 , ' a3:a的第三个值 ' )
insert into tb values ( ' a ' , 3 , ' a3:a的第三个值 ' )
insert into tb values ( ' b ' , 1 , ' b1--b的第一个值 ' )
insert into tb values ( ' b ' , 3 , ' b3:b的第三个值 ' )
insert into tb values ( ' b ' , 2 , ' b2b2b2b2 ' )
insert into tb values ( ' b ' , 4 , ' b4b4 ' )
insert into tb values ( ' b ' , 5 , ' b5b5b5b5b5 ' )
go

select * , px = identity ( int , 1 , 1 ) into tmp from tb

select m.name,m.val,m.memo from
(
select t. * from tmp t where val = ( select min (val) from tmp where name = t.name)
) m
where px = ( select min (px) from
(
select t. * from tmp t where val = ( select min (val) from tmp where name = t.name)
) n
where n.name = m.name)

drop table tb,tmp

/*
name val memo
---------- ----------- --------------------
a 1 a1--a的第一个值
b 1 b1--b的第一个值

(2 行受影响)
*/
-- 在sql server 2005中可以使用row_number函数,不需要使用临时表。
--
创建表并插入数据:
create table tb(name varchar ( 10 ),val int ,memo varchar ( 20 ))
insert into tb values ( ' a ' , 2 , ' a2(a的第二个值) ' )
insert into tb values ( ' a ' , 1 , ' a1--a的第一个值 ' )
insert into tb values ( ' a ' , 1 , ' a1--a的第一个值 ' )
insert into tb values ( ' a ' , 3 , ' a3:a的第三个值 ' )
insert into tb values ( ' a ' , 3 , ' a3:a的第三个值 ' )
insert into tb values ( ' b ' , 1 , ' b1--b的第一个值 ' )
insert into tb values ( ' b ' , 3 , ' b3:b的第三个值 ' )
insert into tb values ( ' b ' , 2 , ' b2b2b2b2 ' )
insert into tb values ( ' b ' , 4 , ' b4b4 ' )
insert into tb values ( ' b ' , 5 , ' b5b5b5b5b5 ' )
go

select m.name,m.val,m.memo from
(
select * , px = row_number() over ( order by name , val) from tb
) m
where px = ( select min (px) from
(
select * , px = row_number() over ( order by name , val) from tb
) n
where n.name = m.name)

drop table tb

/*
name val memo
---------- ----------- --------------------
a 1 a1--a的第一个值
b 1 b1--b的第一个值

(2 行受影响)
*/

-----------------------------------------------------





合并列值
原著:邹建
改编:爱新觉罗.毓华(十八年风雨,守得冰山雪莲花开)  2007-12-16  广东深圳

表结构,数据如下:
id    value
----- ------
1    aa
1    bb
2    aaa
2    bbb
2    ccc

需要得到结果:
id    values
------ -----------
1      aa,bb
2      aaa,bbb,ccc
即:group by id, 求 value 的和(字符串相加)

1. 旧的解决方法(在sql server 2000中只能用函数解决。)
--1. 创建处理函数
create table tb(id int, value varchar(10))
insert into tb values(1, 'aa')
insert into tb values(1, 'bb')
insert into tb values(2, 'aaa')
insert into tb values(2, 'bbb')
insert into tb values(2, 'ccc')
go

CREATE FUNCTION dbo.f_str(@id int)
RETURNS varchar(8000)
AS
BEGIN
    DECLARE @r varchar(8000)
    SET @r = ''
    SELECT @r = @r + ',' + value FROM tb WHERE id=@id
    RETURN STUFF(@r, 1, 1, '')
END
GO

-- 调用函数
SELECt id, value = dbo.f_str(id) FROM tb GROUP BY id

drop table tb
drop function dbo.f_str

/*
id          value     
----------- -----------
1          aa,bb
2          aaa,bbb,ccc
(所影响的行数为 2 行)
*/

--2、另外一种函数.
create table tb(id int, value varchar(10))
insert into tb values(1, 'aa')
insert into tb values(1, 'bb')
insert into tb values(2, 'aaa')
insert into tb values(2, 'bbb')
insert into tb values(2, 'ccc')
go

--创建一个合并的函数
create function f_hb(@id int)
returns varchar(8000)
as
begin
  declare @str varchar(8000)
  set @str = ''
  select @str = @str + ',' + cast(value as varchar) from tb where id = @id
  set @str = right(@str , len(@str) - 1)
  return(@str)
End
go

--调用自定义函数得到结果:
select distinct id ,dbo.f_hb(id) as value from tb

drop table tb
drop function dbo.f_hb

/*
id          value     
----------- -----------
1          aa,bb
2          aaa,bbb,ccc
(所影响的行数为 2 行)
*/

2. 新的解决方法(在sql server 2005中用OUTER APPLY等解决。)
create table tb(id int, value varchar(10))
insert into tb values(1, 'aa')
insert into tb values(1, 'bb')
insert into tb values(2, 'aaa')
insert into tb values(2, 'bbb')
insert into tb values(2, 'ccc')
go
-- 查询处理
SELECT * FROM(SELECT DISTINCT id FROM tb)A OUTER APPLY(
        SELECT [values]= STUFF(REPLACE(REPLACE(
            (
                SELECT value FROM tb N
                WHERE id = A.id
                FOR XML AUTO
            ), ' <N value="', ','), '"/>', ''), 1, 1, '')
)N
drop table tb

/*
id          values
----------- -----------
1          aa,bb
2          aaa,bbb,ccc

(2 行受影响)
*/

--SQL2005中的方法2
create table tb(id int, value varchar(10))
insert into tb values(1, 'aa')
insert into tb values(1, 'bb')
insert into tb values(2, 'aaa')
insert into tb values(2, 'bbb')
insert into tb values(2, 'ccc')
go

select id, [values]=stuff((select ','+[value] from tb t where id=tb.id for xml path('')), 1, 1, '')
from tb
group by id

/*
id          values
----------- --------------------
1          aa,bb
2          aaa,bbb,ccc

(2 row(s) affected)

*/

drop table tb





-----------------------------------------------------

SQL code
   
   
-- 返程问题,找出雇员从本地出发后直接返回的情况
create table trav(name nvarchar ( 10 ),date datetime ,comefrom nvarchar ( 10 ),destin nvarchar ( 10 ),id int )
insert into trav select ' 张三 ' , ' 2007-01-01 ' , ' 上海 ' , ' 广州 ' , 1
insert into trav select ' 李四 ' , ' 2007-01-01 ' , ' 上海 ' , ' 广州 ' , 2
insert into trav select ' 李四 ' , ' 2007-02-01 ' , ' 上海 ' , ' 成都 ' , 3
insert into trav select ' 张三 ' , ' 2007-01-15 ' , ' 广州 ' , ' 上海 ' , 4
insert into trav select ' 张三 ' , ' 2007-02-06 ' , ' 上海 ' , ' 广州 ' , 5
insert into trav select ' 张三 ' , ' 2007-02-18 ' , ' 广州 ' , ' 上海 ' , 6
go
select a.name,a.date,a.comefrom,a.destin,b.date,b.comefrom,b.destin
from trav a inner join trav b on a.name = b.name and a.comefrom = b.destin and a.destin = b.comefrom where a.id < b.id
and not exists ( select 1 from trav where comefrom = b.comefrom and date < b.date and date > a.date)
go
drop table trav
/*
name date comefrom destin date comefrom destin
---------- ----------------------- ---------- ---------- ----------------------- ---------- ----------
张三 2007-01-01 00:00:00.000 上海 广州 2007-01-15 00:00:00.000 广州 上海
张三 2007-02-06 00:00:00.000 上海 广州 2007-02-18 00:00:00.000 广州 上海

(2 行受影响)
*/

-----------------------------------------------------

SQL code
   
   

-- 库存先进先出简单例子:

create table t(
id
int identity ( 1 , 1 ),
name
varchar ( 50 ), -- 商品名称
j int , -- 入库数量
c int , -- 出库数量
jdate datetime -- 入库时间
)
insert into t(name,j,c,jdate) select ' A ' , 100 , 0 , ' 2007-12-01 '
insert into t(name,j,c,jdate) select ' A ' , 200 , 0 , ' 2008-01-07 '
insert into t(name,j,c,jdate) select ' B ' , 320 , 0 , ' 2007-12-21 '
insert into t(name,j,c,jdate) select ' A ' , 100 , 0 , ' 2008-01-15 '
insert into t(name,j,c,jdate) select ' B ' , 90 , 0 , ' 2008-02-03 '
insert into t(name,j,c,jdate) select ' A ' , 460 , 0 , ' 2008-02-01 '
insert into t(name,j,c,jdate) select ' A ' , 510 , 0 , ' 2008-03-01 '
go



create proc wsp
@name varchar ( 50 ), -- 商品名称
@cost int -- 销售量
as
-- 先得出该货物的库存是否够
declare @spare float -- 剩余库存
select @spare = sum (j) - sum (c) from t where name = @name
if ( @spare >= @cost )
begin
-- 根据入库日期采用先进先出原则对货物的库存进行处理
update t set c =
case when ( select @cost - isnull ( sum (j), 0 ) + isnull ( sum (c), 0 ) from t where name = @name and jdate <= a.jdate and j != c) >= 0
then a.j
else
case when ( select @cost - isnull ( sum (j), 0 ) + isnull ( sum (c), 0 ) from t where name = @name and jdate < a.jdate and j != c) < 0 then 0
else ( select @cost - isnull ( sum (j), 0 ) + isnull ( sum (c), 0 ) + a.c from t where name = @name and jdate < a.jdate and j != c)
end
end
from t a where name = @name and j != c
end
else
raiserror ( ' 库存不足 ' , 16 , 1 )
return
go


-- 测试:

exec wsp @name = ' A ' , @cost = 180
select * from t

   
   
   
   



-- drop table t
--
drop proc wsp


-----------------------------------------------------

1. 查看数据库的版本
select @@version

2. 查看数据库所在机器操作系统参数
exec master..xp_msver

3. 查看数据库启动的参数
sp_configure

4. 查看数据库启动时间
select convert(varchar(30),login_time,120) from master..sysprocesses where spid=1

查看数据库服务器名和实例名
print 'Server Name...............: ' + convert(varchar(30),@@SERVERNAME)
print 'Instance..................: ' + convert(varchar(30),@@SERVICENAME)
5. 查看所有数据库名称及大小
sp_helpdb

重命名数据库用的SQL
sp_renamedb 'old_dbname', 'new_dbname'

6. 查看所有数据库用户登录信息
sp_helplogins

查看所有数据库用户所属的角色信息
sp_helpsrvrolemember

修复迁移服务器时孤立用户时,可以用的fix_orphan_user脚本或者LoneUser过程

更改某个数据对象的用户属主
sp_changeobjectowner [@objectname =] 'object', [@newowner =] 'owner'

注意: 更改对象名的任一部分都可能破坏脚本和存储过程。

把一台服务器上的数据库用户登录信息备份出来可以用add_login_to_aserver脚本

7. 查看链接服务器
sp_helplinkedsrvlogin

查看远端数据库用户登录信息
sp_helpremotelogin

8.查看某数据库下某个数据对象的大小
sp_spaceused @objname

还可以用sp_toptables过程看最大的N(默认为50)个表

查看某数据库下某个数据对象的索引信息
sp_helpindex @objname

还可以用SP_NChelpindex过程查看更详细的索引情况
SP_NChelpindex @objname

clustered索引是把记录按物理顺序排列的,索引占的空间比较少。
对键值DML操作十分频繁的表我建议用非clustered索引和约束,fillfactor参数都用默认值。
查看某数据库下某个数据对象的的约束信息
sp_helpconstraint @objname

9.查看数据库里所有的存储过程和函数
use @database_name
sp_stored_procedures
查看存储过程和函数的源代码
sp_helptext '@procedure_name'

查看包含某个字符串@str的数据对象名称
select distinct object_name(id) from syscomments where text like '%@str%'

创建加密的存储过程或函数在AS前面加WITH ENCRYPTION参数

解密加密过的存储过程和函数可以用sp_decrypt过程

10.查看数据库里用户和进程的信息
sp_who
查看SQL Server数据库里的活动用户和进程的信息
sp_who 'active'
查看SQL Server数据库里的锁的情况
sp_lock

进程号1--50是SQL Server系统内部用的,进程号大于50的才是用户的连接进程.
spid是进程编号,dbid是数据库编号,objid是数据对象编号
查看进程正在执行的SQL语句
dbcc inputbuffer ()

推荐大家用经过改进后的sp_who3过程可以直接看到进程运行的SQL语句
sp_who3

检查死锁用sp_who_lock过程
sp_who_lock

11.收缩数据库日志文件的方法
收缩简单恢复模式数据库日志,收缩后@database_name_log的大小单位为M
backup log @database_name with no_log
dbcc shrinkfile (@database_name_log, 5)
12.分析SQL Server SQL 语句的方法:

set statistics time {on | off}
set statistics io {on | off}
图形方式显示查询执行计划

在查询分析器->查询->显示估计的评估计划(D)-Ctrl-L 或者点击工具栏里的图形

文本方式显示查询执行计划
set showplan_all {on | off}

set showplan_text { on | off }
set statistics profile { on | off }

13.出现不一致错误时,NT事件查看器里出3624号错误,修复数据库的方法

先注释掉应用程序里引用的出现不一致性错误的表,然后在备份或其它机器上先恢复然后做修复操作

alter database [@error_database_name] set single_user

修复出现不一致错误的表

dbcc checktable('@error_table_name',repair_allow_data_loss)

或者可惜选择修复出现不一致错误的小型数据库名

dbcc checkdb('@error_database_name',repair_allow_data_loss)
alter database [@error_database_name] set multi_user
CHECKDB 有3个参数:
repair_allow_data_loss 包括对行和页进行分配和取消分配以改正分配错误、结构行或页的错误,
以及删除已损坏的文本对象,这些修复可能会导致一些数据丢失。
修复操作可以在用户事务下完成以允许用户回滚所做的更改。
如果回滚修复,则数据库仍会含有错误,应该从备份进行恢复。
如果由于所提供修复等级的缘故遗漏某个错误的修复,则将遗漏任何取决于该修复的修复。
修复完成后,请备份数据库。
repair_fast 进行小的、不耗时的修复操作,如修复非聚集索引中的附加键。
这些修复可以很快完成,并且不会有丢失数据的危险。
repair_rebuild 执行由 repair_fast 完成的所有修复,包括需要较长时间的修复(如重建索引)。
执行这些修复时不会有丢失数据的危险。




  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值