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 相关文章推荐
基于mysql的论坛(1)
Oct 09 PHP
用PHP和ACCESS写聊天室(六)
Oct 09 PHP
php strstr查找字符串中是否包含某些字符的查找函数
Jun 03 PHP
在windows平台上构建自己的PHP实现方法(仅适用于php5.2)
Jul 05 PHP
使用openssl实现rsa非对称加密算法示例
Jan 24 PHP
thinkphp3查询mssql数据库乱码解决方法分享
Feb 11 PHP
将PHP从5.3.28升级到5.3.29时Nginx出现502错误
May 09 PHP
PHP实现过滤掉非汉字字符只保留中文字符
Jun 04 PHP
基于laravel制作APP接口(API)
Mar 15 PHP
thinkphp关于简单的权限判定方法
Apr 03 PHP
php报错502badgateway解决方法
Oct 11 PHP
laravel与thinkphp之间的区别与优缺点
Mar 02 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
附件名前加网站名
2008/03/23 PHP
php class中self,parent,this的区别以及实例介绍
2013/04/24 PHP
php之可变变量的实例详解
2017/09/12 PHP
彻底搞懂PHP 变量结构体
2017/10/11 PHP
laravel框架select2多选插件初始化默认选中项操作示例
2020/02/18 PHP
不使用浏览器运行javascript代码的方法
2013/07/24 Javascript
jquery 漂亮的删除确认和提交无刷新删除示例
2013/11/13 Javascript
javascript中字符串拼接详解
2014/09/26 Javascript
jQuery简单实现上下,左右滑动的方法
2016/06/01 Javascript
对jQuary选择器的全面总结
2016/06/20 Javascript
BootStrap CSS全局样式和表格样式源码解析
2017/01/20 Javascript
Vue Spa切换页面时更改标题的实例代码
2017/07/15 Javascript
Vue的Class与Style绑定的方法
2017/09/01 Javascript
webpack项目轻松混用css module的方法
2018/06/12 Javascript
JavaScript中的回调函数实例讲解
2019/01/27 Javascript
webpack-url-loader 解决项目中图片打包路径问题
2019/02/15 Javascript
python实现封装得到virustotal扫描结果
2014/10/05 Python
Python中用format函数格式化字符串的用法
2015/04/08 Python
Python使用turtule画五角星的方法
2015/07/09 Python
python 对key为时间的dict排序方法
2018/10/17 Python
使用Python实现在Windows下安装Django
2018/10/17 Python
Python 分发包中添加额外文件的方法
2019/08/16 Python
利用Tensorflow的队列多线程读取数据方式
2020/02/05 Python
canvas像素画板的实现代码
2018/11/21 HTML / CSS
Smashbox官网:美国知名彩妆品牌
2017/01/05 全球购物
国际奢侈品品牌童装购物网站:Designer Childrenswear
2019/05/08 全球购物
UML设计模式笔试题
2014/06/07 面试题
学生干部学习的自我评价
2014/02/18 职场文书
工作鉴定评语
2014/05/04 职场文书
小学生我的梦想演讲稿
2014/08/21 职场文书
高三英语教学反思
2016/03/03 职场文书
《和时间赛跑》读后感3篇
2019/12/16 职场文书
微信小程序用户授权最佳实践指南
2021/05/08 Javascript
Java Optional<Foo>转换成List<Bar>的实例方法
2021/06/20 Java/Android
CSS中实现动画效果-附案例
2022/02/28 HTML / CSS
Java中API的使用方法详情
2022/04/06 Java/Android