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 相关文章推荐
ob_start(),ob_start('ob_gzhandler')使用
Dec 25 PHP
发布一个用PHP fsockopen写的HTTP下载的类
Feb 22 PHP
mysql4.1以上版本连接时出现Client does not support authentication protocol问题解决办法
Mar 15 PHP
dedecms中显示数字验证码的修改方法
Mar 21 PHP
将数组写入txt文件 var_export
Apr 21 PHP
PHP 程序授权验证开发思路
Jul 09 PHP
php编写简单的文章发布程序
Jun 18 PHP
PHP CURL采集百度搜寻结果图片不显示问题的解决方法
Feb 03 PHP
PHP和MYSQL实现分页导航思路详解
Apr 11 PHP
详解PHP5.6.30与Apache2.4.x配置
Jun 02 PHP
PHP使用标准库spl实现的观察者模式示例
Aug 04 PHP
PHP里的$_GET数组介绍
Mar 22 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
Http 1.1 Etag 与 Last-Modified提高php效率
2008/01/10 PHP
php disk_free_space 返回目录可用空间
2010/05/10 PHP
php查询及多条件查询
2017/02/26 PHP
Windows 下安装 swoole 图文教程(php)
2017/06/05 PHP
laravel 去掉index.php伪静态的操作方法
2019/10/12 PHP
PHP xpath提取网页数据内容代码解析
2020/07/16 PHP
Nigma vs Liquid BO3 第一场2.14
2021/03/10 DOTA
JavaScript 滚轮事件使用说明
2010/03/07 Javascript
ajax异步刷新实现更新数据库
2012/12/03 Javascript
jQuery中json对象的复制方式介绍(数组及对象)
2013/06/08 Javascript
使用GruntJS链接与压缩多个JavaScript文件过程详解
2013/08/02 Javascript
js实现精美的图片跟随鼠标效果实例
2015/05/16 Javascript
常用jQuery代码分享
2015/07/14 Javascript
简单实现异步编程promise模式
2015/07/31 Javascript
Javascript 基础---Ajax入门必看
2016/07/06 Javascript
Vue.js 父子组件通讯开发实例
2016/09/06 Javascript
谈谈第三方App接入微信登录 解读
2016/12/27 Javascript
ES6下子组件调用父组件的方法(推荐)
2018/02/23 Javascript
解决在Vue中使用axios用form表单出现的问题
2019/10/30 Javascript
[01:02:48]2018DOTA2亚洲邀请赛小组赛 A组加赛 Newbee vs Liquid
2018/04/03 DOTA
[43:35]EG vs Winstrike 2018国际邀请赛小组赛BO2 第一场 8.18
2018/08/19 DOTA
python 队列详解及实例代码
2016/10/18 Python
Python中字符串与编码示例代码
2019/05/20 Python
Python 旋转打印各种矩形的方法
2019/07/09 Python
基于sklearn实现Bagging算法(python)
2019/07/11 Python
浅谈pytorch池化maxpool2D注意事项
2020/02/18 Python
Python ckeditor富文本编辑器代码实例解析
2020/06/22 Python
Pytorch mask-rcnn 实现细节分享
2020/06/24 Python
Python如何获取文件路径/目录
2020/09/22 Python
Nayomi官网:沙特阿拉伯王国睡衣和内衣品牌
2020/12/19 全球购物
什么是Deployment descriptors;都有什么类型的部署描述符
2015/07/28 面试题
大专应届生个人的自我评价
2013/11/21 职场文书
一年级学生评语
2014/04/23 职场文书
青年文明号创建口号大全
2015/12/25 职场文书
Python办公自动化解决world文件批量转换
2021/09/15 Python
【海涛解说】暗牧也疯狂,牛蛙成配角
2022/04/01 DOTA