我想从一小部分整数(小于200)中随机选择一个值。作为替代
SELECT RAND()
我正在尝试使用
CAST(CAST(CRYPT_GEN_RANDOM(2) AS INTEGER) AS FLOAT) / 65535
但是我得到了一些奇怪的效果。
例如:
WITH Numbers (num)
AS
(
SELECT num
FROM (
VALUES (1), (2), (3), (4),
(5), (6), (7), (8),
(9), (10)
) AS Numbers (num)
),
RandomNumber (num)
AS
(
SELECT CAST(
(CAST(CAST(CRYPT_GEN_RANDOM(2) AS INTEGER) AS FLOAT) / 65535)
* (SELECT COUNT(*) FROM Numbers) + 1
AS INTEGER
)
)
SELECT T1.num, R1.num
FROM Numbers AS T1
INNER JOIN RandomNumber AS R1
ON T1.num = R1.num;
我希望这将返回精确的一行,并且两列值相等。
但是,它返回零,一或多个行,而列值偶尔只是相等的。
知道这里发生了什么吗?
最佳答案
试试这个:
select abs(checksum(newid()))%200
更新
这里发生了一些可疑的事情:
select 'Not in the range! This is impossible!' [Message]
where cast(CAST(CAST(CRYPT_GEN_RANDOM(2) AS int) AS float) / 65535 * 10 as int)
not in (0,1,2,3,4,5,6,7,8,9)
点击
F5
直到收到消息。更新2
此查询具有相同的效果:
select 'Not in the range! This is impossible!' [Message]
where abs(checksum(newid()))%10
not in (0,1,2,3,4,5,6,7,8,9)
更新3
select N'WTF? Schrödinger''s cat!' [Message], *
from (select 0 union select 1) t(n)
join (select abs(checksum(newid()))%2 rnd) r(n) on t.n = r.n
而且,最后一个查询有时可能返回2行,这实际上意味着
abs(checksum(newid()))%2
同时返回0和1,这是不可能的,因为它包含在“仅选择一行”语句中,因此该值在进行连接后稍后会更新。 :|然后再次发生联接,这是令人吃惊的。更新4
在Microsoft Connect上为a known bug。这是an insightful article关于正在发生的事情。