> 백엔드 개발 > PHP 튜토리얼 > !如何更快速的生成一个随机数!

!如何更快速的生成一个随机数!

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
풀어 주다: 2016-06-23 13:47:05
원래의
1087명이 탐색했습니다.

需求是这样的:
会员购物,会生成一个定单号,这个定单号必须是数字,而且是在一定的范围之内的数据,如 1 到 10万。

我现在用的方法是 用  rand(1,100000),生成一个随机数,得到这个随机数后,再去数据库里查,如果已经存在这个数了,就循环再查。
代码如下:

         set_time_limit(0);		$test = M('test');		$_run = true;		$_order_id = rand(0, 100000);		while ($_run) {			$orderInfo = $test->where("val = ".$_order_id)->find();			if(empty($orderInfo)){				$_run = false;				return $_order_id;			}else{				$_order_id = rand(0, 100000);			}		}
로그인 후 복사


这样做了,在定单号还很少的时候,很轻松的就取得了定单号,可假如当定单号已经达到了99990
这个时间,要生成其它的定单号就要很久了,几分钟甚至一个小时都有可能。

这种方法行不通了。

请朋友们帮想想办法,或提供下其它的思路,谢谢了!


回复讨论(解决方案)

$_order_id = date('YmdHis') . rand(100000,999999);//这样的订单号应该不会有重复的,又方便,前面还能看出时间来
로그인 후 복사
로그인 후 복사
로그인 후 복사
로그인 후 복사

$_order_id = date('YmdHis') . rand(100000,999999);//这样的订单号应该不会有重复的,又方便,前面还能看出时间来
로그인 후 복사
로그인 후 복사
로그인 후 복사
로그인 후 복사



需求要求,非要在1-10万之间的数


$_order_id = date('YmdHis') . rand(100000,999999);//这样的订单号应该不会有重复的,又方便,前面还能看出时间来
로그인 후 복사
로그인 후 복사
로그인 후 복사
로그인 후 복사



需求要求,非要在1-10万之间的数



那你还是用原子性的redis自增吧



$_order_id = date('YmdHis') . rand(100000,999999);//这样的订单号应该不会有重复的,又方便,前面还能看出时间来
로그인 후 복사
로그인 후 복사
로그인 후 복사
로그인 후 복사



需求要求,非要在1-10万之间的数



那你还是用原子性的redis自增吧



我没有描述清楚,补充下吧,定单号要在1-10万之间。 定单号不能 递增或递减的生成,一定要随机的,只要在1-10万之间随机,如果随机的过程中产生连号不用处理

可以自己写一个伪随机数发生器
先观察

$m = 100000;$c = 101;$b = 81;$n = 0;for($i=0; $i<100000; $i++) {  $n = ($n * $c + $b) % $m;  $r[] = $n;}print_r(array_count_values(array_count_values($r)));
로그인 후 복사
로그인 후 복사
Array(    [1] => 100000)
로그인 후 복사
로그인 후 복사
可知 $r 中保有 0 - 99999 随机排列的数字,且每个数字只会出现一次

令 $n 为最后一次入库的号码,则 ($n * $c + $b) % $m 一定就不在库中

希望你能理解

先把已有订单号查询出来放数组,没必要每次查库

先查询没有使用的订单号总量,然后随机数最大值就设它,随到多少,你就limit多少

会员ID+时间戳,位置不够在补上足够的随机数就可以。永远也不会重复

可以自己写一个伪随机数发生器
先观察

$m = 100000;$c = 101;$b = 81;$n = 0;for($i=0; $i<100000; $i++) {  $n = ($n * $c + $b) % $m;  $r[] = $n;}print_r(array_count_values(array_count_values($r)));
로그인 후 복사
로그인 후 복사
Array(    [1] => 100000)
로그인 후 복사
로그인 후 복사
可知 $r 中保有 0 - 99999 随机排列的数字,且每个数字只会出现一次

令 $n 为最后一次入库的号码,则 ($n * $c + $b) % $m 一定就不在库中

希望你能理解


能解释下为何是不重复的随机数吗,数学原理是什么啊,我找到不少伪随机数都是要用自然对数的

加时间生成订单号也是可以的

线性同余法  线性同余方法是目前应用广泛的伪随机数生成算法,其基本思想是通过对前一个数进行线性运算并取模从而得到下一个数。即:a(i+1)=(a(i)*b+c)mod(m)其中b称为乘数,c称为增量,m称为模数,它们均为常数。乘数、增量和模数的选取可以多种多样,只要保证产生的随机数有较好的均匀性和随机性即可。线性同余法的最大周期是m,但一般情况下会小于m。要使周期达到最大,应该满足以下条件:(1) c和m互质;(2) m的所有质因子的积能整除b-1;(3) 若m是4的倍数,则b-1也是;(4) b,c,a(0)(初值,一般即种子)都比m小;(5) b,c是正整数。
로그인 후 복사
经过耐心的筛选,可以找到使绝对均匀的b和c

array_count_values(array_count_values($r)) 中
array_count_values($r) 得到一个数组,保存有$r中每个值出现的次数
再次统计得
Array
(
    [1] => 100000
)
可知这十万个数是不重复的

$_order_id = date('YmdHis') . rand(1000,9999);

谢谢版主的热心帮助

원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿