《PHP教程:PHP的偽隨機數與真隨機數詳解》要點:
本文介紹了PHP教程:PHP的偽隨機數與真隨機數詳解,希望對您有用。如果有疑問,可以聯系我們。
PHP編程首先需要聲明的是,計算機不會產生絕對隨機的隨機數,計算機只能產生“偽隨機數”.其實絕對隨機的隨機數只是一種抱負的隨機數,即使計算機怎樣發展,它也不會產生一串絕對隨機的隨機數.計算機只能生成相對的隨機數,即偽隨機數.
PHP編程偽隨機數并不是假隨機數,這里的“偽”是有規律的意思,就是計算機產生的偽隨機數既是隨機的又是有規律的.怎樣理解呢?產生的偽隨機數有時遵守一定的規律,有時不遵守任何規律;偽隨機數有一部分遵守一定的規律;另一部分不遵守任何規律.好比“世上沒有兩片形狀完全相同的樹葉”,這正是點到了事物的特性,即隨機性,但是每種樹的葉子都有近似的形狀,這正是事物的共性,即規律性.從這個角度講,你大概就會接受這樣的事實了:計算機只能產生偽隨機數而不能產生絕對隨機的隨機數.
PHP編程首先來了解一下真隨機數和偽隨機數的概念.
PHP編程真隨機數發生器:英文為:true random number generators ,簡稱為:TRNGs,是利用弗成預知的物理方式來產生的隨機數.
PHP編程偽隨機數發生器:英文為:pseudo-random number generators ,簡稱為:PRNGs,是計算機利用必定的算法來產生的.
PHP編程對比一下兩種方法產生的隨機數的圖片.
PHP編程Random.org(利用大氣噪音來生成隨機數,而大氣噪音是空氣中的雷暴所發生的 )生成的隨機位圖:
PHP編程
PHP編程Windows下PHP的rand()函數發生的隨機圖片:
PHP編程
PHP編程很顯然,后者偽隨機數發生器發生的圖片有這明顯的條紋.
PHP編程利用php的rand隨機函數發生這張圖片的代碼為:
PHP編程另外,Linux內核(1.3.30以上)包括了一個隨機數發生器/dev/random ,對于很多平安目的是足夠的.
PHP編程下面是關于Linux的隨機數產生器的原理介紹 :
PHP編程Linux 操作系統提供本色上隨機(或者至少具有強烈隨機性的部件)的庫數據.這些數據通常來自于設備驅動程序.例如,鍵盤驅動程序收集兩個按鍵之間時間的信息,然后將這個環境噪聲填入隨機數發生器庫.
PHP編程隨機數據存儲在 熵池 ( linux內核維護了一個熵池用來收集來自設備驅動程序和其它來源的環境噪音.理論上,熵池中的數據是完全隨機的,可以實現產生真隨機數序列.為跟蹤熵池中數據的隨 機性,內核在將數據加入池的時候將估算數據的隨機性,這個過程稱作熵估算.熵估算值描述池中包含的隨機數位數,其值越大表示池中數據的隨機性越好. ) 中,它在每次有新數據進入時進行“攪拌”.這種攪拌實際上是一種數學轉換,贊助提高隨機性.當數據添加到熵池中 后,系統估計獲得了多少真正隨機位.
PHP編程測定隨機性的總量是很重要的.問題是某些量往往比起先考慮時看上去的隨機性小.例如,添加表現自從上次按鍵盤以來秒數的 32 位數實際上并沒有提供新的 32 位隨機信息,因為大多數按鍵都是很接近的.
PHP編程從 /dev/random 中讀取字節后,熵池就使用 MD5 算法進行暗碼散列,該散列中的各個字節被轉換成數字,然后返回.
PHP編程如果在熵池中沒有可用的隨機性位, /dev/random 在池中有足夠的隨機性之前等待,不返回結果.這意味著如果使用 /dev/random 來發生許多隨機數,就會發現它太慢了,不夠實用.我們經常看到 /dev/random 生成幾十字節的數據,然后在許多秒內都不發生結果.
PHP編程幸運的是有熵池的另一個接口可以繞過這個限制:/dev/urandom.即使熵池中沒有隨機性可用,這個替代設備也總是返回隨機數.如果您取出許 多數而不給熵池足夠的時間重新充滿,就再也不能獲得各種來源的合用熵的好處了;但您仍可以從熵池的 MD5 散列中獲得非常好的隨機數!這種方式的問題是,如果有任何人破解了 MD5 算法,并通過查看輸出了解到有關散列輸入的信息,那么您的數就會立刻變得完全可預料.大多數專家都認為這種分析從計算角度來講是不可行的.然而,仍然認為 /dev/urandom 比 /dev/random 要“不平安一些”(并通常值得懷疑).
PHP編程Windows下沒有/dev/random可用,但可以使用微軟的“capicom.dll”所提供的CAPICOM.Utilities 對象.
PHP編程以下是使用PHP時比用mt_rand()函數發生更好的偽隨機數的一段例子代碼:
PHP編程$pr_bits = '';
PHP編程// Unix/Linux platform?
$fp = @fopen('/dev/urandom','rb');
if ($fp !== FALSE) {
$pr_bits .= @fread($fp,16);
@fclose($fp);
}
PHP編程// MS-Windows platform?
if (@class_exists('COM')) {
try {
$CAPI_Util = new COM('CAPICOM.Utilities.1');
$pr_bits .= $CAPI_Util->GetRandom(16,0);
PHP編程// if we ask for binary data PHP munges it, so we
// request base64 return value. We squeeze out the
// redundancy and useless ==CRLF by hashing...
if ($pr_bits) { $pr_bits = md5($pr_bits,TRUE); }
} catch (Exception $ex) {
// echo 'Exception: ' . $ex->getMessage();
}
}
PHP編程if (strlen($pr_bits) < 16) {
// do something to warn system owner that
// pseudorandom generator is missing
}
?>
PHP編程所以PHP要發生真隨機數 還是要調用外部元素來支持的!
維易PHP培訓學院每天發布《PHP教程:PHP的偽隨機數與真隨機數詳解》等實戰技能,PHP、MYSQL、LINUX、APP、JS,CSS全面培養人才。