仅为个人学习笔记

ABS函数

在数据库获取数据时通常会遇到负数的情况,但客户要求的是不能有负数,那么可以使用abs函数进行处理

ABS 函数 返回一个数值的绝对值 |x|

select ABS(-3)

返回结果 3

随机函数

RAND随机函数

Rand()函数是获取随机数的函数,可直接运行select rand() 获取0~1之间float型的数字。
如果想要获取0~100之间的整数随机数,可以这样使用

select round(100*rand(),0)

为方便使用,下面延伸了一个方法:

DECLARE @NumBegin Int=60	--随机数的最小值
DECLARE @NumEnd Int=100		--随机数的最大值
DECLARE @Decimal Int=2		--保留小数点几位
SELECT @NumBegin+round((@NumEnd-@NumBegin)*rand(),@Decimal)

这个方法通过传入最大值和最小值以及返回数保留几位小数,来获取对应的随机值。上面这个例子执行结果是获取60和100之间的随机数保留两位小数,如果将2改为-1,则变成获取60~100之间的10的倍数的整数,不信可以尝试一下!

其实rand函数有个弊端,获取的随机数可能是重复的。这并不是我们想要的结果。
如果换成是newid试试,就不会出现重复。
为什么会出现这样的结果?
一样的sql语法,使用newid出来的结果是“真随机”,而用rand出来的结果却是“伪随机”。
而我们需要的是一串纯数字,用newid出来的结果并不方便处理。所以,针对这种情景,我们还有一个方式获得随机数,那就需要引入另外一个函数 checksum 了,checksum可以和newid结合使用产生随机数。



Checksum(Newid()获取随机数(区别于Rand)

前面有介绍rand获取随机数的方法,因为rand本身有一个无法避免的尴尬问题,所以需要引入checksum结合newid的方法来获取随机数!

在此之前,先简单了解一下checksum吧!
Checksum:总和检验码,校验和。在数据处理和数据通信领域中,用于校验目的的一组 数据项的和。这些数据项可以是数字或在计算检验总和过程中看作数字的其它字符串。它通常是以十六进制为数制表示的形式, 如果校验和的数值超过十六进制的FF,也就是255. 就要求其补码作为校验和.通常用来在通信中,尤其是远距离通信中保证数据的完整性和准确性。
在SQLServer中Checksum()需要传入1个参数,可以是任何类型,如下示例:

SELECT CHECKSUM(1)			--1
SELECT CHECKSUM('A')		--114
SELECT CHECKSUM('AA')		--34472462
SELECT CHECKSUM(GETDATE())	--18516390

可以看出,传入不同参数就有不同的返回值,每个参数的返回值都是固定的,而且还有可能会出现负数的,几乎看不出有什么规律~
因此,我们可以通过传入newid()来获得随机数,因为newid()每次获得的值都是唯一的随机的。
可以以此验证一下:

SELECT CHECKSUM(Newid()) union all
SELECT CHECKSUM(Newid()) union all
SELECT CHECKSUM(Newid()) union all
SELECT CHECKSUM(Newid())

返回值即为随机数。


稍加处理,就可以当作rand来使用了,而且还避免了在某些情景下rand获得一堆重复的随机数的尴尬局面:

SELECT CHECKSUM = CONVERT(BIGINT,RIGHT(ABS(CHECKSUM(Newid())),9))*0.1/100000000,RAND=RAND() union all
SELECT CHECKSUM = CONVERT(BIGINT,RIGHT(ABS(CHECKSUM(Newid())),9))*0.1/100000000,RAND=RAND() union all
SELECT CHECKSUM = CONVERT(BIGINT,RIGHT(ABS(CHECKSUM(Newid())),9))*0.1/100000000,RAND=RAND()

可以和Rand放在一起比较,看起来区别不大。

下面有一张临时表,只有id一个字段,我用checksum和rand两种方法分别给每个id附上一个随机数:

看出问题所在了吧!rand获得的随机数都是一样的,而checksum结合newid的方法获得的随机数才是我们想要的!

下面延伸一个方法,供将来有需要获得随机数的时候使用!新建一个标量值函数,通过传入参数最大值、最小值、保留几位小数、以及newid()来获得随机数:

Create FUNCTION Scalar_CheckSumNEWID
(
	@From int,
	@To int,
	@Keep int,
	@newid varchar(50)
)
RETURNS float
BEGIN
	DECLARE @ResultVar float
	SELECT @ResultVar=CONVERT(BIGINT,RIGHT(ABS(CHECKSUM(@newid)),9))*0.1/100000000
	RETURN @From+round((@To-@From)*@ResultVar,@Keep)
END
GO

注:newid之所以要放在传入参数中,是因为想newid和rand之类的不能放在函数的本身执行,只能通过传入参数带入

随机实例操作

向上取整随机整数

select cast (floor(rand()*80) AS datetime ) AS 大头儿子

向下取整随机整数

select  CEILING(rand()*100) as 小头爸爸

生成一段范围内的随机数字

declare@Result float DECLARE @Upper float DECLARE @Lower float SET @Lower = 50 SET @Upper = 100 SELECT @Result = ROUND(((@Upper - @Lower -1) * RAND() + @Lower), 3) SELECT @Result

生成一段时间内的随机时间

declare @Date_start datetime declare @Date_end datetime set @Date_start= '2021-01-01' set @Date_end=getdate() select 时间=dateadd(minute,abs(checksum(newid()))%(datediff(minute,@Date_start,@Date_end)+1),@Date_start)

这里我们引用到另一个时间差值函数

在下一篇笔记里讲