美文网首页程序员
使用 php-x 开发php唯一ID扩展

使用 php-x 开发php唯一ID扩展

作者: daydaygo | 来源:发表于2017-09-18 13:41 被阅读555次

    date: 2017-9-17 19:48:13
    title: 使用 php-x 开发php唯一ID扩展

    唯一ID 相关资源:

    php 扩展开发相关资源:

    整体体验下来, 韩老大的 php-x 在易用性上完胜, 毕竟官方 wiki 目前给出的 api 非常少, 稍微阅读一下, 配合源码中给出的 example, 就能轻松上手.

    唯一 id 设计归纳:

    • 唯一性
    • 时间相关: 添加时间戳, 常用秒级和毫秒级
    • 粗略有序: 数据库的自增 id 是一个参考思路, 但完全自增也会导致攻击者可以根据已有 id 构造其他 id
    • 可反解
    • 分布式: 可选, 在性能要求比较高的场景下, 可以增加 machine_id 或者 worker_id 来达到分布式的效果, 比如使用 48bit MAC 地址

    技术难点: sequence 在生成的时候, 考虑到并发, 需要加锁, 自旋锁 是可选的方案.

    show the code

    由于 c++ 已经荒废多年, 写起来确实不顺手, 所以实现了一版简单的, 权当抛砖引玉:

    采用 Simple flake: timestamp + sequence
    uid(64bit) = timestamp(秒级, 32bit) + sequence(随机数, 32bit)

    • 原版 c++ 代码
    #include <iostream>
    #include <ctime>
    #include <cstdlib>
    #include <cinttypes>
    using namespace std;
    
    int main()
    {
        uint64_t data;
        uint64_t timestamp = (unsigned)time(NULL); // 秒级时间戳, 32bit
        cout << time(NULL) << endl;
        srand(timestamp);
        uint64_t sequence = rand() % 4294967295; // 32bit
        data = timestamp << 32
            | sequence;
        cout << timestamp << '\t' << sequence << '\t' << data << endl;
    
        // 1505655347   31225   6466740474412562937
        data = 6466740474412562937;
        timestamp = data >> 32;
        sequence = data & 0xfffffff;
        cout << timestamp << '\t' << sequence << '\t' << data << endl;
        return 0;
    }
    
    • php-x

    最开始使用 uint64_t, 要和 Variant 进行强制类型转换, 下面是第二版
    字符串要使用双引号, 这个地方坑了好久

    // cpp_atom.cc
    #include "phpx.h"
    #include <iostream>
    #include <ctime>
    #include <cstdlib>
    
    using namespace php;
    using namespace std;
    
    PHPX_FUNCTION(cpp_atom_generate)
    {
        long data;
        long timestamp = (unsigned)time(NULL);
        srand(timestamp);
        int sequence = rand() % 4294967295; // 32bit
        data = timestamp << 32
            | sequence;
        // cout<< data << endl;
        retval = data;
    }
    
    PHPX_FUNCTION(cpp_atom_explain)
    {
        Variant key = args[0];
        long data = key.toInt();
        // sscanf(key.toCString(), "%llu", &data); // 另一个类型转换的方式
        int timestamp = data >> 32;
        int sequence = data & 0xfffffff;
        // cout << data << '\t' << timestamp << '\t' << sequence << endl;
        Array arr;
        arr.set("ts", timestamp);
        arr.set("seq", sequence);
    
        retval = arr;
    }
    
    PHPX_EXTENSION()
    {
        Extension *extension = new Extension("cpp_atom", "0.0.1");
        extension->registerFunction(PHPX_FN(cpp_atom_generate));
        extension->registerFunction(PHPX_FN(cpp_atom_explain));
    
        extension->info(
        {
            "cpp_atom support", "enabled"
        },
        {
            { "author", "daydaygo" },
            { "version", extension->version },
            { "date", "2017-9-17 21:24:46" },
        });
    
        return extension;
    }
    
    • Makefile
    PHP_INCLUDE = `php-config --includes`
    PHP_LIBS = `php-config --libs`
    PHP_LDFLAGS = `php-config --ldflags`
    PHP_INCLUDE_DIR = `php-config --include-dir`
    PHP_EXTENSION_DIR = `php-config --extension-dir`
    
    cpp_atom.so: cpp_atom.cc
        c++ -DHAVE_CONFIG_H -g -o cpp_atom.so -O0 -fPIC -shared cpp_atom.cc -std=c++11 -lphpx ${PHP_INCLUDE} -I${PHP_INCLUDE_DIR}
    install: cpp_atom.so
        cp cpp*.so ${PHP_EXTENSION_DIR}/
    clean:
        rm *.so
    
    • test case

    测试时建议将代码中的 cout 部分的调试代码加上, 用来对比, 第一版强制类型转换的时候, 就遇到 (int)data 这个将 64位整形转为 32位整形

    <?php
    var_dump(cpp_atom_generate());
    var_dump(cpp_atom_explain(6466751737141169189));
    

    无论是 php-x 还是 php-cpp 这样的项目, 都是在 zend-api 上面多封装了一层, 有兴趣还是可以去翻翻 zend-api 看看.
    当然, 真要写扩展, 还是要 c/c++ 功底的.

    相关文章

      网友评论

        本文标题:使用 php-x 开发php唯一ID扩展

        本文链接:https://www.haomeiwen.com/subject/jbfbsxtx.html