SQLServer 编程总结

case的用途

 case语句有四个关键字,缺一不可,分别是:case when then end,另外还有一个else。case 后面跟字段名(当在when后面出现字段名时,case里不能写出来),when后面跟判断语句,then后面跟显示结果, end表示case语句结束。

--第一种用法 用于简单的是非判断(字段名在case后面)
select sname,case ssex when 1 then '男' else '女' end as sex from student

--第二种 进行稍微复杂一些的判断(字段名在when后面)
select  sname,case when ssex = 1 then '男' when ssex =0 then '女' end as sex from student

  

for xml path

for xml path是将结果集以XML的格式返回。用的比较多的地方,就是group by之后,拼接字符串返回。

示例:

--创建一张表subjects,有三个人,分别选修了不同的科目
create table subjects(username varchar(20),subjects_name varchar(20))

insert into subjects (username,subjects_name) values
( '谭' ,'高数' ),
( '谭','音乐' ),
( '谭谭','体育' ),
( '谭谭','武术' ),
( '谭谭','化学' ),
( '谭谭谭','生物' )

  

根据上面的内容,我们就可以查询每个人学习的科目,并把这些科目放在一起(用逗号分隔)。

--使用stuff去除首位分隔符
select s.username,stuff(( select ','+b.subjects_name from subjects as b where b.username=s.username for xml path('')),1,1,'') as subjects
from subjects as s
group by username

  

 

除了上面的用法,也可是 FOR XML PATH可以用来拼接JSON字符串或字符串。例如:

-- STUFF结合FOR XML PATH可以用来拼接JSON字符串
select '['+ stuff((select ',{"name": "' + username + '","subjects": "' + subjects_name + '"}'
from subjects for xml path('')),1,1,'') +']'

  

while循环与游标

while语句使用示例

declare @i int
set @i = 1
while @i<5
begin
	--insert into student (sname,sage,ssex) values ('tanyongjun',12,1)
	print @i 
	set @i = @i +1
end

游标示例  

--1.声明游标
declare test_cursor cursor scroll
for select sname,sage from student where ssex = 1
--2.打开游标
open test_cursor
--3.声明游标提取数据所要存放的变量
declare @sname varchar(50), @sage int
--4.定位游标到哪一行
fetch first from test_cursor into @sname,@sage  --into的变量数量必须与游标查询结果集的列数相同
while @@fetch_status=0  --提取成功,进行下一条数据的提取操作 
 begin
   --进来业务处理
   --例如:把 sname为 tanyongjun  的数据中 sage 字段 加+1,进行修改
   if @sname='tanyongjun'
	begin
		update student set sage = sage+1 where current of test_cursor
		-- 如果是修改当前行,就可以用上面这中写法
	end

   -- 当然也可以根据提取的数据作为条件进行处理 
   fetch next from test_cursor into @sname,@sage  --移动游标
 end  
 --5.关闭游标
 close test_cursor
 --6.删除游标,释放资源
 deallocate test_cursor

总结:
1. while循环和游标都可以实现循环的目的,while用法比较简洁明了;
2. 游标是在while的基础上实现循环的功能,但是语法较复杂,且效率没有while高,所以一般不推荐使用;
3. while循环体内支持再嵌套一个循环,但是一定要注意不能出现死循环,否则sql服务就会卡死、崩溃。

exists 

exists是用来判断是否存在的,当exists查询中的查询存在结果时则返回真,否则返回假。not exists则相反。
exists做为where 条件时,是先对where 前的主查询询进行查询,然后用主查询的结果一个一个的代入exists的查询进行判断,如果为真则输出当前这一条主查询的结果,否则不输出。

查询时,一般情况下,子查询会分成两种情况:
1.子查询与外表的字段有关系时
例如: select col1 , col2 from t1 where exists (select col1 , col2 from t2 where t2.col1 = t1.col1)
上面的sql语句相当于一个关联查询。它先执行t1的查询,然后把t1中的每一条记录放到t2的条件中去查询,如果存在,则显示此条记录。
2.子查询与外表的字段没有任何关联
例如: select col1 , col2 from t1 where exists ( select * from t2 where t2.col1 = '条件')
在这种情况下,只要子查询的条件成立,就会查询出t1中的所有记录,反之,如果子查询中没有查询到记录,则t1不会查询出任何的记录。
当子查询与主表不存在关联关系时,简单认为只要exists为一个条件判断,如果为true,就输出所有记录。如果为false则不输出任何的记录。

  

json操作

主要介绍5个函数:
1. openJson:打开Json字符串
2. IsJson:判断一个字符串是不是合法的Json格式。是返回1,否返回0,null返回null。
3. Json_Value:从Json字符串中提取值。
4. Json_Query:Json字符串中提取对象或数组。
5. Json_Modify:更新Json字符串中的属性值,并返回更新的Json字符串。

--IsJson:判断一个字符串是不是合法的Json格式。是返回1,否返回0,null返回null。
select isjson('{"id":"1","name":"tanyongjun","sex":1}')
select isjson('tan')

--从Json字段中提取值
select json_value('{"id":"1","name":"tanyongjun","sex":1}','$.name')

  

函数和存储过程

一、自定义函数

自定义函数分为:标量值函数或表值函数两种。
标量值函数:如果 RETURNS 子句指定一种标量数据类型,则函数为标量值函数。(返回字段)
表值函数:如果 RETURNS 子句指定 TABLE,则函数为表值函数。(返回table)
表值函数又分为两种:内嵌表值函数(行内函数)或多语句函数
如果 RETURNS 子句指定的 TABLE 不附带列的列表,则该函数为内嵌表值函数。
如果 RETURNS 子句指定的 TABLE 类型带有列及其数据类型,则该函数是多语句表值函数

--标量值函数
create function fun_add_num(@age int)
returns int
as
begin
	declare @num int
	set @num = 2
		return @age + @num
end

go

-- 调用  *注:必须加上dbo.否则会报错
select dbo.fun_add_num(11)

go

--内嵌表值函数
create function fun_test(@name varchar(50))
returns table
as
return (select sname,sage,ssex from student where sname = @name)
go

--调用  在调用 表值函数 时 dbo. 可以省略
select * from  fun_test('tanyongjun')

go

-- 多语句表值函数   ****分割字符串,然后返回
create function fun_split_str
 (
     @str_val      varchar(2000), --要分割的字符串
     @split varchar(10)  --分隔符号
 )
returns @temp table (result varchar(10))
as
 begin
     declare @result as varchar(10);   --定义变量用于接收单个结果
     set @str_val = @str_val + @split ;
     while (@str_val <> '')
     begin
         set @result = left(@str_val, charindex(@split, @str_val) -1) ;
         insert @temp values(@result) ;
         --STUFF()函数用于删除指定长度的字符,并可以在指定的起点处插入另一组字符。
         set @str_val = stuff(@str_val, 1, charindex(@split, @str_val), '');
     end
     return
 end
 go

 --调用 
 select * from fun_split_str('tan,tanyong,tanyongjun',',')

注意事项:

标量函数:
1. 所有的入参前都必须加@
2. create后的返回,单词是returns,而不是return
3. returns后面的跟的不是变量,而是返回值的类型,如:int,char等。
4. 在begin/end语句块中,是return。

内嵌表值函数:
1. 只能返回table,所以returns后面一定是TABLE
2. AS后没有begin/end,只有一个return语句来返回特定的记录。

多语句表值函数:
1. returns后面直接定义返回的表类型,首先是定义表名,表明前面要加@,然后是关键字TABLE,最后是表的结构。
2. 在begin/end语句块中,直接将需要返回的结果insert到returns定义的表中就可以了,在最后return时,会将结果返回。
3. 最后只需要return,return后面不跟任何变量。

  

二、存储过程

-- 无参存储过程
create proc proc_test1 
as -- as 不能省略
begin
 select * from tb_test
end

go

exec proc_test1

go

--有参存储过程
create proc proc_test2 
@username varchar(50)
as 
begin
 select * from tb_test where username = @username
end

exec proc_test2 'tanyongjun'

--有参 有返回值的存储过程
create proc proc_test3 
@username varchar(50),
@row_number int output
as 
begin
 select @row_number = count(*) from tb_test where username = @username
end
go

declare @row_num int
exec proc_test3 'tanyongjun11', @row_num output -- 不写output ,@row_num 不输出任何信息
print @row_num

存储过程和函数的使用:
1. 存储过程和函数都可以将一段SQL语句进行封装,这样大大的方便了实际使用时候的调用步骤。
2. 存储过程里面可以嵌套存储过程,也经常会调用各种函数。
3. 函数里面可以调用其他函数,但是一般不嵌套存储过程。
4. 上面例子中的函数,也可以改写成存储过程,但是函数使用起来要方便的多,因为函数可以直接用select联用,但是存储过程只能用exec执行。

  

常用函数(字符串、时间)

一、字符串

--计算字符串长度
select len('tanyongjun')

--字符串转换为大、小写
select lower('TAN')  --将字符串转换为小写
select upper('tan')  --将字符串转换为大写

--去字符串前后空格   
select trim( '   tanyongjun   ')

--删除指定长度的字符,并在指定的起点处插入另一组字符
--stuff(character_expression , start , length ,character_expression)
--start 一个整数值,指定删除和插入的开始位置。
--length 一个整数,指定要删除的字符数。
select stuff('abcdefg',1,6,'Hello ')

--从左侧开始取子字符串
select left('tanyongjun',3)
--从右侧开始取子字符串
select right('tanyongjun',7)

--字符串替换
select replace('tanyongjun','yong','YONG')

--字符串拼接
select 'tan'+'yogjun'  --用+号实现字符串拼接
select concat('tan','yongjun')   --用concat()内置函数实现字符串拼接

  

二、时间

select convert(varchar(50),getdate(),23)  -- 获取当前日期 
select convert(varchar(50),getdate(),108)  -- 获取当前时间

  

1. DATEADD (datepart , number, date )函数
用于进行日期时间的加法运算。其中datepart参数是表示日期部分的参数(比如是以日期还是以月份相加等),number参数是具体的加数,正数代表向未来日期方向加,负数代表向过去日期方向减,date参数为待计算的日期时间类型数据。
datepart参数可以取的单位类型如下:

例如:当前时间加上 3 天 :   select dateadd(dd,3,getdate())

2. DATEDIFF ( datepart , startdate , enddate )函数

用于计算两个日期时间之间的差额。其中datepart参数表示日期部分的参数(比如只比较年还是只比较月等),startdate参数为起始日期时间类型数据;enddate参数为结束日期时间类型数据。

示例: select datediff(dd, '2023/1/17 ', '2023/1/19')

3. DATENAME(datepart,date)函数 

用来获取一个日期的特定部分,比方只获取年份或者是只获取月份等。其中datepart参数是表示要返回的日期部分的参数(即如果是Year的话,则返回的是具体的年份),date参数为待计算日期。

示例: select datename(dd,getdate())  -- 获取当前时间是 “几号”

 

三、其他

-- cast  类型转换
declare @temp1 int = 1
select cast(@temp1 as varchar(50) )


-- isnull 如果表达式为 NULL,则返回指定值,否则返回表达式
declare @temp2 varchar(50)
select isnull(@temp2,'没有内容')
set @temp2 = 'tanyongjun'
select isnull(@temp2,'没有内容')

 

posted @ 2023-01-17 23:53  #谭  阅读(232)  评论(0编辑  收藏  举报