背景:想在表中随机取10条记录,让取出来的数据不重复(表中必须是有个递增列,且递增从1开始间隔为1)。 

数据表:

CREATE TABLE testable
(
id INT  IDENTITY(1,1),
myname  NVARCHAR(1000),
insertedTime DATETIME DEFAULT SYSDATETIME()
)

 

表中共有100条数据,如下

省略……

 

 

 

 

1.     首先想到的是MSSQL自带的newid()

 

采用这种方法时,需要将表中所有记录与newid()生成的值进行比较从而进行排序。因此,如果表中的记录较多,操作会非常缓慢。

 

 

 


USE  Gift163DB
go

SELECT  TOP  14  * FROM  dbo.testable   ORDER BY NEWID()


 

缺点:1. 取出的10条数据会出现重复      2.当数据表数据很多的时候,速度将很慢 (每次重新计算newid)

 

 

 

2. 自定义函数返回一个表,表中记录的是随机生成的N个id值。

 

1)rand()生成随机数        rand()*count,CEILING,floor用法 

2)如果临时表中无此数据,则放入,否则重新生成

3)直到N条记录已经生成

 

注意:标量函数function内不能出现rand()方法,变通下生成个view  v_random,然后在函数内调用 v_random获取随机数

create view v_random
AS

   select CEILING(rand()*51) as random    --注意51,生成的是1到51之间的数字,因为事先知道数据库中有51条记录
go

 

 

自定义函数代码如下:

 

ALTER FUNCTION  randomIntStringWithCommaSplit(@counts int) –counts 表明返回的个数
          
RETURNS @t TABLE (filed1 int)  --返回表@t,有个int类型的 field列
AS
BEGIN

  DECLARE @randomInt INT
  DECLARE @i INT
  SET @i=0
        WHILE @i<@counts
        BEGIN

                 select @randomInt= random FROM v_random

          --不能是 SET @randomInt=SELECT random FROM v_random


          IF NOT EXISTS(SELECT TOP 1 * FROM @t WHERE filed1=@randomInt)
           BEGIN
             INSERT INTO @t VALUES (@randomInt)
             SET @i=@i+1
           END
        END
  RETURN


END

 

上面函数返回的是一个表类型,表中有个int字段,存放要查找的N个不同的keyId  (keyId为要查找表的递增列,且递增为1,从1开始递增)

所以返回的表中存放的数据是 dbo.Articles中的id列的值。

 

 

使用:调用上面的自定义函数返回10个不重复的id

 



SELECT * FROM randomIntStringWithCommaSplit(10)



 

下面是几次的执行结果,可以看到每个结果中都不存在重复的值(fidled1为临时表的唯一列)

 

 

最后通过  select * from table  where id in  randomIntStringWithCommaSplit(10) ,这样就可以从table中随机取出10条不重复的数据来了。

 

 

 

 

3.存储过程取不重复的数据

 

--dbo.getRandomDataFromTable

--输入参数  @tableName  nvarchar(100),--表名

---@dataCount  nvarchar(100)--取N条数据

 

输出结果集:列id,存放N条要查询的数据



USE Gift163DB
GO
IF OBJECT_ID ( 'dbo.getRandomDataFromTable', 'P' ) IS NOT NULL
    DROP PROCEDURE dbo.getRandomDataFromTable;
GO

CREATE PROC  [dbo].[getRandomDataFromTable]
@tableName  nvarchar(100),
@dataCount  nvarchar(100)
AS
BEGIN
  --SET NOCOUNT ON;

    DECLARE @t TABLE (id INT)   --临时表
    DECLARE @i  INT             --临时变量
    DECLARE @randomInt INT      --每次随机生成的整数
    DECLARE @tableCount INT     --表的行数

    --先获取表中最大数据的id
   EXEC(  'SELECT '+@tableCount+'=COUNT(*) FROM  '+@tableName+'')
    SET @i=0
    WHILE @i<@dataCount
    BEGIN
      SELECT @randomInt=CEILING(RAND()*@tableCount)

      IF NOT EXISTS(SELECT TOP 1 * FROM @t)
      BEGIN

          INSERT INTO @t VALUES (@randomInt)
          SET @i=@i+1
      END

    END


    --打印出取出的表的id
    SELECT * FROM @t



END
Go



 

 

生成测试数据100条

USE Gift163DB
GO
if exists (select 1
            from  sysobjects
           where  id = object_id('testable')
            and   type = 'U')
   drop table testable
go

CREATE TABLE testable
(
id INT  IDENTITY(1,1),
myname  NVARCHAR(1000),
insertedTime DATETIME DEFAULT SYSDATETIME()
)



//插入100条数据
DECLARE @i INT
SET @i=1
WHILE @i<100
BEGIN
  INSERT INTO tesTable (myname) VALUES ('我的名字是'+CONVERT(NVARCHAR, @i) )   --将 varchar 值 '我的名字是' 转换成数据类型 int 时失败。
  SET @i=@i+1


End


  

 

 

调用上面的存储过程从表testable取10条不重复id

 

 



USE Gift163DB
go
 exec    getRandomDataFromTable  'testable',10


 

 

 

自定义的存储过程不会出现重复的记录

 

 

 

 

 

4.改进的存储过程,最终存储过程

 

输入参数3个:表名,表的递增列名 , 要取的N条数据

 

USE Gift163DB
GO
IF OBJECT_ID ( 'dbo.getRandomDataFromTable', 'P' ) IS NOT NULL
    DROP PROCEDURE dbo.getRandomDataFromTable;
GO

CREATE PROC  [dbo].[getRandomDataFromTable]
@tableName  nvarchar(100),
@identityKey NVARCHAR(100),
@dataCount  nvarchar(100)
AS
BEGIN
  --SET NOCOUNT ON;

    --DECLARE @t TABLE (id INT)   --临时表
    DECLARE @i  INT             --临时变量
    DECLARE @randomInt INT      --每次随机生成的整数
    DECLARE @tableCount INT     --表的行数



    --先获取表中最大数据的id
   DECLARE @str NVARCHAR(3000)
   SET @str='SELECT @tableCount=COUNT(*) FROM  '+@tableName
   exec sp_executesql @str, N'@tableCount int output', @tableCount output


   create TABLE #sdf (id int)



    SET @i=0
    WHILE @i<@dataCount AND @i<@tableCount
    BEGIN
      SELECT @randomInt=CEILING(RAND()*@tableCount)

      IF NOT EXISTS(SELECT TOP 1 * FROM #sdf WHERE id=@randomInt)
      BEGIN

          INSERT INTO #sdf VALUES (@randomInt)
          SET @i=@i+1
      END

    END


    --取出数据
    DECLARE @str2 NVARCHAR(2000)

    SET @str2=' SELECT  * FROM '+@tableName
        +' where '+@identitykey
        +' in '
        +' (select id from #sdf )'




    PRINT @str2
    --select id from  @t

    EXEC (@str2)
    --exec sp_executesql @str2,N'@t TABLE',@t OUTPUT





END

 

调用存储过程:随机取10条数据

 

最终的存储过程不管你随机取多少条数据(只要每次取的数据数目小于表中行数) 就可以保证每次取的数据不会重复。当然前提是,取的表必须有个递增列,而且以1开始,递增1.

sql 生成随机数 以及不重复随机数的更多相关文章

  1. 动态生成16位不重复随机数、随机创建2位ID

    /** 1. * 动态生成16位不重复随机数 * * @return */ public synchronized static String generate16() { StringBuffer ...

  2. js【生成规定数量不重复随机数】、【冒泡排序】、【鸡尾酒排序】、【选择排序】、【插入排序】、【未完工的二分插入排序】------【总结】

    [生成规定数量不重复随机数] function creatRandom( num ){ var randomLen = num, ranArr = [], thisRan = null, whileO ...

  3. sql 生成某个范围内的随机数

    从i-j的范围内的随机数,那么公式为FLOOR(i+RAND()*(j-i+1))

  4. SQL - 生成指定范围内的随机数

    今天按照公司需求,需要做一个sql作业来对数据库定时触发,其中有个难点,就是在sql中需要在1-n中随机出来一个结果. google了半天,找到一个比较好的方式. 写下这个sql: DECLARE @ ...

  5. 【转】 SQL - 生成指定范围内的随机数

    DECLARE @Result INT DECLARE @Upper INT DECLARE @Lower INT SET @Lower = 1 SET @Upper = 10 SELECT @Res ...

  6. Java数组实现随机生成N-M之间不重复的随机数

    接收一个整形数组,使用Math.Random每次在规定的数字范围内随机产生数字,然后嵌套for循环依次判断是否有重复值,如果有既外循环变量减一,直到把数组装满为止. /** * 随机生成 N--M的不 ...

  7. C#生成不重复随机数列表

    C#生成不重复(随机数 http://www.jbxue.com/tags/suijishu.html)列表实例的代码.例子: ;             Random rnd = ; i <  ...

  8. PHP函数:生成N个不重复的随机数

    思路:将生成的随机数存入数组,再在数组中去除重复的值,即可生成一定数量的不重复随机数. 程序: <?php /* * array unique_rand( int $min, int $max, ...

  9. [转载]C# Random 生成不重复随机数

    Random 类 命名空间:System 表示伪随机数生成器,一种能够产生满足某些随机性统计要求的数字序列的设备. 伪随机数是以相同的概率从一组有限的数字中选取的.所选数字并不具有完全的随机性,因为它 ...

随机推荐

  1. 原!mysql5.6 存储过程 批量建表

    由于业务需求,需要按天分表,因此写了个存储过程,根据时间生成表. 根据createTime 的时间,以及 while循环的变量设置范围,生成该指定日期及之后的多张表. BEGIN ); ); ; '; ...

  2. python用whl文件安装包

    直接用pip安装包的时候有时会因为许多奇怪的原因安装失败,这时就需要祭出大杀器——whl文件 python3 -m pip install whl文件路径名 whl库:https://www.lfd. ...

  3. 解决ajax post json数据 后端无法收到的问题

    如图,想把input框中的文字以json格式post出去,结果后端收不到 使用wireshark抓包,根本没有抓到发往服务器的包,说明错误在前端. 后来发现ajax post json数据的时候key ...

  4. python setup.py install 报错:error: [WinError 3] 系统找不到指定的路径。: 'C:\\Program Files (x86)\\Microsoft Visual Studio 14.0\\VC\\PlatformSDK\\lib

    Outline 在通过 setup.py 安装python模块时,遇到了以下报错: # 执行 python setup.py install # 报错: error: [WinError 3] 系统找 ...

  5. window安装redis数据库

    1.下载安装包 1.百度网盘链接:https://pan.baidu.com/s/1MrAK5Suc1xpzkbp1WQbP0A 提取码:u9uq 2.GitHub:https://github.co ...

  6. Mysql实现企业级日志管理、备份与恢复

    数据备份形式 文件备份: 通过Linux的备份命令把文件统一打个包存起来,可存在本地和远程服务器,等到要恢复时,再用这些文件恢复到指定位置. 数据库数据备份: 在一些对数据可靠性要求很高的行业如银行. ...

  7. php socket 处理只是来数据流,该怎样避免(好像是堵塞了)

    php socket 处理只是来数据流,该怎样处理(好像是堵塞了) 需求:php接受一个硬件往8888port上发送数据,假设收到后,应socket_send函数返回"\xFA\x01\x0 ...

  8. mysql以下c连接mysql数据库

    1.安装sudo yum install mysql-devel  安装组件和库 2. #include <stdio.h> #include <stdlib.h> #incl ...

  9. xpath中遇到[<Element a at 0x39a9a80>](转)

    Element是什么 回归正题,大家晕头转脑的看完繁杂的语法之后,已经迫不及待写点什么东西了,然后部分同学可能遇到了这个 <Element a at 0x39a9a80>或者类似 Elem ...

  10. vscode 编写c++

    参考了 配置文件之前的操作:https://www.cnblogs.com/lianshuiwuyi/p/8094388.html 配置文件具体内容:https://blog.csdn.net/qq5 ...