Delphi 生成指定范围和个数的不重复的随机整数


本文整理自网络,侵删。

 
生成指定范围内的随机数,有相应的随机函数(如RandomRange(x, y)可生成x <= d < y的随机整数),或者在基本的随机函数上稍加修改也可生成;生成 n 个随机数,只需调用 n 次随机函数即可;生成 n 个不重复的随机数,就会有一点点小麻烦。

常规思路
一般来说,要生成n个不重复的随机数,只需判断每次生成的随机数有没有和这前生成的随机数重复即可,若重复即抛弃,不重复则记录。

但是,这样要进行很多额外的判断,而且当生成的量变大时,这样的判断次数也几乎是呈指数级的增加(具体复杂度没有进行详细分析)。

另一种思路
如果每生成一个随机整数,就在一个整数序列上对应的位置做一个标记,那么只需要判断标记的个数有没有达到n即可,然后把有标记的整数取出就是 n 个不重复的随机整数。(其实该思路是借鉴了某个排序算法的思路,具体算法名称不记得了)

首先来考虑生成 n 个[0, m)的不重复的随机数方法,n < m。

var
  i: Integer;
  tmpMark: array of Integer;
begin
  SetLength(tmpMark, m);
  repeat
    Randomize;
    i := RandomRange(0, m); //[0,m)半开半闭区间
    tmpMark[i] := 1;
  until SumInt(tmpMark) = n;
  for i := 0 to m - 1 do
    if tmpMark[i] = 1 then
      i; //i即为随机出的不重复的整数
end;
本例中借助长度为m的数组tmpMark来进行标记,同时使用delphi自带的SumInt(在Math单元)函数来计算标记的个数,有标记的tmpMark下标即为随机出的整数。

对于[x, y)区间,可以在[0, m)的基础上进行平移来实现,n < y - x。

var
  l, i: Integer;
  tmpMark: array of Integer;
begin
  l := y - x;
  SetLength(tmpMark, l);
  repeat
    Randomize;
    i := RandomRange(x, y); //[x,y)半开半闭区间
    tmpMark[i - x] := 1;
  until SumInt(tmpMark) = n;
  for i := 0 to l - 1 do
    if tmpMark[i] = 1 then
      i + x; //i+x即为随机出的不重复的整数
end;

来源:https://my.oschina.net/afrusrsc/blog/3001060

相关阅读 >>

Delphi 动态加载删除字体

Delphi 用空格把 s 凑够 n 的长度

Delphi 打开android应用信息

Delphi nethttpclient1 数据库查询

Delphi实现背景音乐播放

Delphi 提高进程自身权限

Delphi输入字符与tlistbox项目匹配

Delphi json字符串转义

Delphi 压缩图片算法

Delphi winsock 域名获取远程服务器ip

更多相关阅读请进入《Delphi》频道 >>



打赏

取消

感谢您的支持,我会继续努力的!

扫码支持
扫码打赏,您说多少就多少

打开支付宝扫一扫,即可进行扫码打赏哦

分享从这里开始,精彩与您同在

评论

管理员已关闭评论功能...