PHP随机数 C扩展随机数


Posted in PHP onMay 04, 2016

由于要用到固定长度的随机字符串。

首先是一段PHP代码

$str_md5=md5(uniqid());
 $rand = mt_rand(1, 28);
 $str1=substr($str_md5,$rand,6);
 $rand = mt_rand(1, 28);
 $str2=substr($str_md5,$rand,6);
 $rand = mt_rand(1, 28);
 $str3=substr($str_md5,$rand,6);
 $code=substr($str1.$str2.$str3,0,8);

PHP随机数 C扩展随机数

生成180000个随机字符串,图中是按照重复数量倒序排列,可以看到基本都有重复的。不过也是比较理想的。

由于想提升一下自己的C语言能力,所以用C重新写了一下随机生成字符串。

其中用到了随机数函数srand(),rand();

不过折腾一两个小时,随机数还是有问题。并发访问时时间可能几乎为同时,那么srand给的种子时间可以视为相同的。这样就导致了,产生的随机数也是一样的。从而产生的随机字符串也是一样的。循环输出随机字符串,几乎都是一模一样的。

后来想到了ukey,这个扩展可以实现唯一的ID,那么访问都产生唯一的ID,是不是可以将这个ID作为种子时间。答案是肯定的。

PHP随机数 C扩展随机数

上图是产生的随机字符串,可以自定义长度。也同样可以输出只有数字的字符串。相较PHP所产生的随机字符串重复率更低且速度更快。

PHP_FUNCTION(get_random__num_str)
{
  int length=8;
  
  if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &length) == FAILURE) 
  {
  length=8;
   
  }
  length++;
 int flag, i; 
 char* string; 
 __uint64_t timestamp = realtime();
 __uint64_t retval;
 int len;
 char buf[128];
 
 if (timestamp == 0ULL) {
  RETURN_FALSE;
 }
 
 spin_lock(lock, pid);
 
 if (context->last_timestamp == timestamp) {
  context->sequence = (context->sequence + 1) & context->sequence_mask;
  if (context->sequence == 0) {
   timestamp = skip_next_millis();
  }
 
 } else {
  context->sequence = 0; /* Back to zero */
 }
 
 context->last_timestamp = timestamp;
 
 retval = ((timestamp - context->twepoch) << context->timestamp_left_shift)
   | (context->datacenter_id << context->datacenter_id_shift)
   | (worker_id << context->worker_id_shift)
   | context->sequence;
 
 spin_unlock(lock, pid);
 //printf('%ld',retval);
 srand((unsigned)retval);
 //srand((unsigned) time(NULL )); 
 if ((string = (char*) emalloc(length)) == NULL ) 
 { 
  //myLog("Malloc failed!flag:14\n"); 
  RETURN_NULL() ; 
 } 
 
 for (i = 0; i < length - 1; i++) 
 { 
  flag = rand() % 3; 
   
  switch (flag) 
  { 
   case 0: 
    string[i] = '1' + rand() % 5; 
    break; 
   case 1: 
    string[i] = '2' + rand() % 7; 
    break; 
   case 2: 
    string[i] = '0' + rand() % 10; 
    break; 
   default: 
    string[i] = '9'; 
    break; 
  } 
   
   
   
 } 
 string[length - 1] = '\0'; 
 RETURN_STRINGL(string,length,0);
}
 PHP_FUNCTION(get_random_str)
{
  int length=8;
  
  if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &length) == FAILURE) 
  {
  length=8;
   
  }
  length++;
 int flag, i; 
 char* string; 
 __uint64_t timestamp = realtime();
 __uint64_t retval;
 int len;
 char buf[128];
 
 if (timestamp == 0ULL) {
  RETURN_FALSE;
 }
 
 spin_lock(lock, pid);
 
 if (context->last_timestamp == timestamp) {
  context->sequence = (context->sequence + 1) & context->sequence_mask;
  if (context->sequence == 0) {
   timestamp = skip_next_millis();
  }
 
 } else {
  context->sequence = 0; /* Back to zero */
 }
 
 context->last_timestamp = timestamp;
 
 retval = ((timestamp - context->twepoch) << context->timestamp_left_shift)
   | (context->datacenter_id << context->datacenter_id_shift)
   | (worker_id << context->worker_id_shift)
   | context->sequence;
 
 spin_unlock(lock, pid);
 //printf('%ld',retval);
 srand((unsigned)retval);
 //srand((unsigned) time(NULL )); 
 if ((string = (char*) emalloc(length)) == NULL ) 
 { 
  //myLog("Malloc failed!flag:14\n"); 
  RETURN_NULL() ; 
 } 
 
 for (i = 0; i < length - 1; i++) 
 { 
  flag = rand() % 3; 
   
  switch (flag) 
  { 
   case 0: 
    string[i] = 'A' + rand() % 26; 
    break; 
   case 1: 
    string[i] = 'a' + rand() % 26; 
    break; 
   case 2: 
    string[i] = '0' + rand() % 10; 
    break; 
   default: 
    string[i] = 'x'; 
    break; 
  } 
   
   
   
 } 
 string[length - 1] = '\0'; 
 RETURN_STRINGL(string,length,0);
}

PHP随机数 C扩展随机数

上图是PHP生成18W随机字符串所用的时间

PHP随机数 C扩展随机数

上图是C扩展生成18W随机字符串所用的时间

所用的服务器都是1G内存 双核的阿里云服务器。

只要在ukey中加入上如代码就可以生产随机字符串和随机长度数字字符串,PHP唯一ID生成扩展ukey。

php.ini的配置项:

[ukey]
ukey.datacenter = integer
ukey.worker = integer
ukey.twepoch = uint64

datacenter配置项是一个整数, 用于设置数据中心;
worker配置项是一个整数, 用于设置数据中心的机器序号;
twepoch配置项是一个64位的整数, 用于设置时间戳基数, 此值越大, 生成的ID越小;

安装:

$ cd ./ukey
$ phpize
$ ./configure
$ make
$ sudo make install

Ukey提供3个有用的函数:

ukey_next_id() -- 用于生成唯一ID
ukey_to_timestamp(ID) -- 用于将ID转换成时间戳
ukey_to_machine(ID) -- 用于将ID转换成机器信息

使用实例:

<?php
$id = ukey_next_id();
echo $id;
 
$timestamp = ukey_to_timestamp($id);
echo date('Y-m-d H:i:s', $timestamp);
 
$info = ukey_to_machine($id)
var_dump($info);
?>

以上就是本文的全部内容,希望对大家的学习有所帮助。

PHP 相关文章推荐
php2html php生成静态页函数
Dec 08 PHP
基于empty函数的判断详解
Jun 17 PHP
关于PHP堆栈与列队的学习
Jun 21 PHP
php 判断字符串中是否包含html标签
Feb 17 PHP
php写的AES加密解密类分享
Jun 20 PHP
PHP图片处理之图片背景、画布操作
Nov 19 PHP
php通过淘宝API查询IP地址归属等信息
Dec 25 PHP
PHP判断一个数组是另一个数组子集的方法详解
Jul 31 PHP
利用php + Laravel如何实现部署自动化详解
Oct 11 PHP
搜索附近的人PHP实现代码
Feb 11 PHP
php中isset与empty函数的困惑与用法分析
Jul 05 PHP
PHP设计模式之数据访问对象模式(DAO)原理与用法实例分析
Dec 12 PHP
PHP正则表达式过滤html标签属性(DEMO)
May 04 #PHP
Joomla使用Apache重写模式的方法
May 04 #PHP
Joomla开启SEF的方法
May 04 #PHP
Joomla简单判断用户是否登录的方法
May 04 #PHP
Joomla实现组件中弹出一个模式(modal)窗口的方法
May 04 #PHP
joomla组件开发入门教程
May 04 #PHP
Yii2前后台分离及migrate使用(七)
May 04 #PHP
You might like
PHP命名空间(namespace)的使用基础及示例
2014/08/18 PHP
PHP simplexml_load_file()函数讲解
2019/02/03 PHP
JavaScript的parseInt 进制问题
2009/05/07 Javascript
动态加载图片路径 保持JavaScript控件的相对独立性
2010/09/06 Javascript
javascript循环变量注册dom事件 之强大的闭包
2010/09/08 Javascript
jQuery操作表格(table)的常用方法、技巧汇总
2014/04/12 Javascript
node.js中的定时器nextTick()和setImmediate()区别分析
2014/11/26 Javascript
javascript常用代码段搜集
2014/12/04 Javascript
js获取上传文件的绝对路径实现方法
2016/08/02 Javascript
浅谈AngularJs指令之scope属性详解
2016/10/24 Javascript
js实现符合国情的日期插件详解
2017/01/19 Javascript
js轮播图无缝滚动效果
2017/06/17 Javascript
javascript 作用于作用域链的详解
2017/09/27 Javascript
webpack构建的详细流程探底
2018/01/08 Javascript
修改vue+webpack run build的路径方法
2018/09/01 Javascript
在vue中给列表中的奇数行添加class的实现方法
2018/09/05 Javascript
Element Card 卡片的具体使用
2020/07/26 Javascript
[41:11]完美世界DOTA2联赛PWL S2 Inki vs Magma 第一场 11.22
2020/11/24 DOTA
python模拟新浪微博登陆功能(新浪微博爬虫)
2013/12/24 Python
python使用beautifulsoup从爱奇艺网抓取视频播放
2014/01/23 Python
python实现给微信指定好友定时发送消息
2019/04/29 Python
Python读取yaml文件的详细教程
2020/07/21 Python
HTML5之SVG 2D入门6—视窗坐标系与用户坐标系及变换概述
2013/01/30 HTML / CSS
美国购买体育赛事门票网站:TicketCity
2019/03/06 全球购物
Unineed旗下时尚轻奢网站:FABHunt
2019/05/13 全球购物
迅雷Cued工作心得体会
2014/01/27 职场文书
跟单业务员岗位职责
2014/03/08 职场文书
求职简历自我评价范例
2014/03/12 职场文书
聚美优品的广告词
2014/03/14 职场文书
三八节主持词
2014/03/17 职场文书
公司员工活动策划方案
2014/08/20 职场文书
学校百日安全活动总结
2015/05/07 职场文书
毕业欢送会致辞
2015/07/29 职场文书
MySQL的join buffer原理
2021/04/29 MySQL
Java方法重载和方法重写的区别到底在哪?
2021/06/11 Java/Android
python 单机五子棋对战游戏
2022/04/28 Python