3

出于测试和学习的目的,我想修改https://github.com/php/php-src :ext/standard/rand.c中的 php randmt_rand函数。

我想在每次调用 rand 函数时给出一个固定的输出,为此我修改了代码

PHPAPI long php_rand(TSRMLS_D)
{
    long ret;

    if (!BG(rand_is_seeded)) {
        php_srand(GENERATE_SEED() TSRMLS_CC);
    }

#ifdef ZTS
    ret = php_rand_r(&BG(rand_seed));
#else
# if defined(HAVE_RANDOM)
    ret = random();
# elif defined(HAVE_LRAND48)
    ret = lrand48();
# else
    ret = rand();
# endif
#endif

    // ignoring the results coming from the calls above and 
    // returning a constant value 
    ret = 3264;

    return ret;
}

编译

./configure
make
make install

最后调用 rand 函数echo rand(3000,4000);,它总是返回3000

修改此功能的方法是什么?为什么有TSRMLS_D但没有范围参数?

4

2 回答 2

2

从 PHP 调用的实际函数是声明PHP_FUNCTION(rand)的函数,它调用php_rand()您修改的函数。它执行以下操作:

PHP_FUNCTION(rand)
{
    long min;
    long max;
    long number;
    int  argc = ZEND_NUM_ARGS();

    if (argc != 0 && zend_parse_parameters(argc TSRMLS_CC, "ll", &min, &max) == FAILURE)
        return;

    number = php_rand(TSRMLS_C);
    if (argc == 2) {
        RAND_RANGE(number, min, max, PHP_RAND_MAX);
    }

    RETURN_LONG(number);
}

如您所见,如果您传入 2 个参数,它会调用RAND_RANGE宏,将结果php_rand()转换为给定的范围。

#define RAND_RANGE(__n, __min, __max, __tmax) \
    (__n) = (__min) + (long) ((double) ( (double) (__max) - (__min) + 1.0) * ((__n) / ((__tmax) + 1.0)))

如果您希望它始终输出给定值,我建议修改PHP_FUNCTION(rand),而不是php_rand().

尽管如果您要寻找的只是稳定的随机数输出,以便每次都能得到相同的结果,但如果您不关心确切的结果,则可以srand(0)在程序开始时调用,以播种随机数具有恒定值的数字生成器。这意味着随机数生成器将始终为您提供相同的随机数序列,这对于测试目的很有用,并且可以在不修补 PHP 本身的情况下完成。

于 2012-11-19T19:04:11.323 回答
0

查看 php_rand.h 中定义的 RAND_RANGE 宏。它会对输入值进行一些额外的排列/加扰,以使其适合您指定的范围。

如果您真的希望 rand() 函数返回 3264,即使它不适合传递给 rand() 的参数,请在 rand.c 中删除此块:

    if (argc == 2) {
    RAND_RANGE(number, min, max, PHP_RAND_MAX);
}

如果您仍想考虑 min/max,那么如果您指定的 min/max 不包括硬编码的数字,您将需要弄清楚您想要实际执行的操作。

于 2012-11-19T18:57:14.810 回答