/**
* 生成商家交易单号
* <br />特点:不重复
* <br />示例:
* <br />普通付款:array('shop_id'=>1,'product_id'=>array(1,2,3), 'user_id'=>1, 'ip'=>'127.0.0.1', 'amount'=>0.01, 'timestamp'=>'2017-06-22 18:02:33', 'sign_key'=>'signkey!@#123_') 结果为: ib1bd7s9bc50c787114b195e7
* <br />合并付款:generate_trade_no(array('shop_id'=>1,'product_id'=>array(1,2,3), 'user_id'=>1, 'ip'=>'127.0.0.1', 'amount'=>0.01, 'timestamp'=>'2017-06-22 18:02:33', 'sign_key'=>'signkey!@#123_')) 结果为:ib1bd7rs5c50c787114b195e7
* @param array $data
* @return string 返回30或25位位字符串,格式为: 时间{年月日时分秒,14位}+md5{16位}=30位, 例如: 20170622180940c50c787114b195e7 或 ib1bd7s9bc50c787114b195e7
*/
function generate_trade_no(array $data, $ziptime = true) {
ksort($data); // 根据数组的键值对数组重新排序
$hashtext = md5(json_encode($data)); // 把数组格式化为JSON字符串并生成MD5签名
$time = date('YmdHis');
if ($ziptime) {
$time = base_convert($time, 10, 32); // 使用32进制将14位时间戳压缩到9位32进制值
}
$trade_no = $time . substr($hashtext, 8, 16); // 把MD5签名截取16位, 并在签名加上14位或9位日期时间戳, 组成30位或25位字符串
return $trade_no;
} /**
* php生成32进制8位字符,减轻数据库唯一字段
* <br />背景:
* <br />一般在数据库里唯一字段我们用AUTO_ID或者CRC32或者MD5
* <br />AUTO_ID 10进制根据数据列长度,一般是初学者用,应为在高级应用里不灵活
* <br />CRC32 16进制8位,很直观,但是重复几率要比 MD5高,
* <br />MD5 16进制32位,不直观,太占数据库唯一字段,数据选择压力大。
* <br />最近2天一直在研究如何PHP做一个既有MD5的唯一能力又减少位数用在数据库开发里
* <br />想到压缩MD5为32进制8位,下面是函数,希望大家测试或者给点意见,或者有更灵活的方法
* <br />see: http://www.myexception.cn/database/832012.html
* <br />示例:only('a') 结果是: H1N29LQ1
* @param string $instr
* @return string 返回8位字符串
*/
function only($instr) {
$instr = md5($instr, true);
$dict = '0123456789ABCDEFGHIJKLMNOPQRSTUV';
$outstr = '';
for ($i = 0; $i < 8; $i++) {
$ord = ord($instr[$i]);
$outstr .= $dict[( $ord ^ ord($instr[$i + 8]) ) - $ord & 0x1F];
}
return $outstr;
} /**
* 把12位时间数值压缩成8-9位字母+数字组合字符串
* <br />重点是加上用户编号后将永不重复!
* <br />用途:订单号、交易号、唯一号
* <br />如:
* <br />timeserialize(date('y-m-d-H-i-s')), 结果是:RGXO3128
* <br />timeserialize('17-12-31-23-59-59'), 结果是:RM31X5959
* <br />timeserialize('50-12-31-23-59-59'), 结果是:2M31X5959
* <br />timeserialize('0-0-0-0-0-0'), 结果是:AAAAAA
* <br />timeserialize('99-99-99-99-99-99'), 结果是:99999999999999
* @return string 返回8-9位字母+数字组合字符串
* @since VER:1.0; DATE:2016-1-21; AUTHOR:SoChishun; EMAIL:14507247@qq.com; DESC:Added.
*/
function timeserialize($time = '') {
if (!$time) {
$time = date('y-m-d-H-i-s');
}
echo '<div>', $time, '</div>';
$atime = explode('-', $time);
$outstr = '';
// ASCII表:48(0)-57(9), 65(A)-90(Z), 97(a)-122(z)
// 有效压缩范围:0-25, 48-57
foreach ($atime as $stime) {
$itime = $stime * 1;
echo $itime, ',';
// 如果数值小于26,则转换成ASCII表的字母,这样可以把两位的数值变成一位字母
if ($itime < 26) {
// 65(A)-90(Z)
$outstr .= chr(65 + $itime);
continue;
}
// 48(0)-57(9)
// 如果数值在48-57之间,则转换成ASCII表的数字,把两位数转变为1数字
if ($itime >= 48 && $itime <= 57) {
$outstr .= chr($stime);
continue;
}
$outstr .= $stime;
}
return $outstr;
}

版权声明:本文采用署名-非商业性使用-相同方式共享(CC BY-NC-SA 3.0 CN)国际许可协议进行许可,转载请注明作者及出处。
本文标题:代码收藏系列--php技巧--生成简短唯一订单号
本文链接:http://www.cnblogs.com/sochishun/p/7070365.html
本文作者:SoChishun (邮箱:14507247#qq.com | 博客:http://www.cnblogs.com/sochishun/)
发表日期:2017年6月23日

代码收藏系列--php--生成简短唯一订单号的更多相关文章

  1. 代码收藏系列--php--生成简短唯一订单号(转载)

    代码收藏系列--php--生成简短唯一订单号 /** * 生成商家交易单号 * <br />特点:不重复 * <br />示例: * <br />普通付款:arra ...

  2. Java订单号生成,唯一订单号(日均千万级别不重复)

    Java订单号生成,唯一订单号 相信大家都可以搜索到很多的订单的生成方式,不懂的直接百度.. 1.订单号需要具备以下几个特点. 1.1 全站唯一性. 1.2 最好可读性. 1.3 随机性,不能重复,同 ...

  3. 全局唯一订单号生成方法(参考snowflake)

    backgroud Snowflake is a network service for generating unique ID numbers at high scale with some si ...

  4. 代码收藏系列--javascript--日期函数

    /** * 获取当前时间的简短函数 * @returns {String} * @@example getTimeStamp() 结果是:2017-07-12 09:21:30 */ function ...

  5. 代码收藏系列--mysql--创建数据库、数据表、函数、存储过程命令

    创建mysql数据库 CREATE DATABASE IF NOT EXISTS `database_name` DEFAULT CHARSET utf8 COLLATE utf8_general_c ...

  6. php生成唯一订单号

    支持更改长度/** * 生成唯一订单号 * */ function build_order_no(){ return date('Ymd').substr(implode(NULL, array_ma ...

  7. php生成唯一id/唯一标识符/唯一订单号

    /** * php 生成唯一id * https://blog.csdn.net/hzqghost/article/details/18914681 */ function guid($factor= ...

  8. 代码收藏系列--jquery--筛选器、事件绑定技巧

    Jquery筛选器的一些常用技巧,比如过滤属性等 /* 过滤获取没有含data-xsui-grid-colspan的节点 */$(this).find('.xsui-grid-cell:not([da ...

  9. 代码收藏系列--javascript--移动端技巧

    JS判断是否是手机端访问: var is_mobi = navigator.userAgent.toLowerCase().match(/(ipod|iphone|android|coolpad|mm ...

随机推荐

  1. Docker for Windows(四)实践搭建&删除MySQL服务

    我们已经下载安装好了Docker for Windows:Docker for Windows(一)下载与安装,也简单了解了Docker常用命令:Docker for Windows(三)Docker ...

  2. js时间与毫秒数互相转换(转)

    [1]js毫秒时间转换成日期时间   var oldTime = (new Date("2017/04/25 19:44:11")).getTime(); //得到毫秒数    / ...

  3. PHP中cookie和session的区别

    1.cookie数据存放在客户的浏览器上,session数据放在服务器上. 2.cookie不是很安全,别人可以分析存放在本地的COOKIE并进行COOKIE欺骗考虑到安全应当使用session. 3 ...

  4. 通过代码动态创建IIS站点

    对WebApi进行单元测试时,一般需要一个IIS站点,一般的做法,是通过写一个批处理的bat脚本来实现,其实通过编码,也能实现该功能. 主要有关注三点:应用程序池.Web站点.绑定(协议类型:http ...

  5. WiFi 干扰器,有时间可以去试试呦!

    转自社区: 0X01 引言 想不想搞个WIFI干扰器?网上搜集了一下资料,发现用esp8266可以实现简单的干扰功能,包括断网.复制.欺骗等等.刚好手上有块Tpyboard V202(30元),也是e ...

  6. Oracle EBS 复制用户职责

    DECLARE l_new_user_name ) := 'HAND_HYL'; l_new_password ) :'; l_new_user_id NUMBER; l_user_name_copy ...

  7. SqlServer触发器实现表的级联插入、级联更新

    首先建立两张表,分别为test1与test2,期望在更改test1的时候,test2的相关记录能够同时做出更改.假定test1与test2的表结构相同,如下表所示 name age     触发器实现 ...

  8. Visual Studio 下nuget命令的使用

    从Visual Studio 2012版本开始默认集成了Nuget扩展,在Visual Studio 2010或以下的版本需要单独安装,安装方法如下: 1. “工具”→“扩展和更新...”,弹出扩展管 ...

  9. ORA-06512 问题解决

    SQL> select * from dba_role_privs where grantee='SUK';     GRANTEE GRANTED_ROLE ADMIN_OPTION DEFA ...

  10. Redis学习---基础学习[all]

    什么是NoSQL型数据库 NoSQL数据库---NoSQL数据库的分类 Redis学习---NoSQL和SQL的区别及使用场景 Redis学习---负载均衡的原理.分类.实现架构,以及使用场景 什么是 ...