2

我有一个自定义 PHP 扩展,它比较二进制数据字符串中的每个字节。如果文件中同一位置的两个字节都打开(或大于 1),则同一位置的输出将打开。否则它关闭。扩展如下...

PHP_FUNCTION(compare_memory)
{
    char *memory1, *memory2;
    int memory1_length, memory2_length, return_length;
    int length;
    char *output;
    int x;

    zval *param;

    // MAKE SURE WE HAVE
    // string, (string_len,), string2, (string2_len) and length
    if(ZEND_NUM_ARGS() != 2) WRONG_PARAM_COUNT;

    // GET ARGUMENTS FROM PHP 
    if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ss", &memory1, &memory1_length, &memory2, &memory2_length) == FAILURE)
    {
        return;
    }

    if (memory1_length < memory2_length)
    {
        length = memory1_length;
        return_length = memory2_length;
    } else
    {
        length = memory2_length;
        return_length = memory1_length;
    }

    if ((output = emalloc(return_length * sizeof(int))) == NULL)
    {
        printf("Error on malloc\n");
        return;
    }

    // START COMPARING
    for(x=1;x<length;x++)
    {
        if (memory1[x] > 0 && memory2[x] == 1)
        {
            output[x] = memory1[x];
        }
    }

    RETVAL_STRINGL(output, return_length, 1);
    efree(output);
}   

我也有一个PHP实现如下......

function compare_memory($data1, $data2)
{
    $pack_one = pack("c", 1);
    $pack_zero = pack("c", 0);
    $strlen_data1 = strlen($data1);
    $strlen_data2 = strlen($data2);
    if ($strlen_data1 > $strlen_data2)
    {
        $strlen = $strlen_data1;
    } else
    {
        $strlen = $strlen_data2;
    }
    $output = "";

    for($x=0;$x<$strlen;$x++)
    {
        if ($strlen_data1 > $x)
        {
            $arr = unpack("cc", $data1[$x]);
            $d1 = $arr["c"];
        } else
        {
            $d1 = 0;
        }

        if ($strlen_data2 > $x)
        {
            $arr = unpack("cc", $data2[$x]);
            $d2 = $arr["c"];
        } else
        {
            $d2 = 0;
        }

        if ($d1 > 0 && $d2 == 1)
        {
            $output .= pack("c", $d1);
        } else
        {
            $output .= $pack_zero;
        }
    }

    return $output;
}

这个扩展在 PHP 5.0 上运行良好(我知道它已经很老了)。我们最近升级到 5.3,这不再有效。但是 PHP 实现确实如此。

我想不同版本之间的 PHP 内部 API 有很多变化,包括“参数解析 API 统一将导致某些函数在类型杂耍方面表现得或多或少严格”。

您能在 PHP 扩展中看到任何明显的可能导致它在较新版本的 PHP 中无法工作的内容吗?我对 C(和 PHP API)的了解很少。

4

1 回答 1

0

我不确定您是否可以保证emalloc会返回用零初始化的内存。您可以尝试使用ecalloc代替。或者自己做:

if ((output = emalloc(return_length * sizeof(int))) == NULL)
{
    printf("Error on malloc\n");
    return;
}
memset(output, 0, return_length * sizeof(int) );


另外,我认为emalloc 永远不会回来NULL

(我也不确定你为什么for(x=1;x<length;x++)从 1 而不是 0 开始,但也许是有原因的)

于 2012-04-27T13:16:05.810 回答