将使用按位的c方法转换为php

发布于 2025-01-01 11:02:50 字数 3924 浏览 2 评论 0原文

我正在尝试将 APR(apache 运行时)方法转换为 PHP,但我不太明白。

该方法创建一个作为目录路径和文件的字符串 - 因此示例输出将为 A/B/C/[a-fA-F0-9_@]。它对从 md5($string,true) 返回的二进制字符串使用按位运算;当我运行下面的 php 时,我得到:

array (
  0 => 'A',
  1 => '/',
  2 => 'A',
  3 => '/',
  4 => 'A',
  5 => '/',
  6 => 'AAAAAAAAAAAAAAAAAAA',
)

我做错了什么?由于 PHP 内部如何表示 char 数组,这是否不可能?

这是 c 方法

static void cache_hash(const char *it, char *val, int ndepth, int nlength)
{
    apr_md5_ctx_t context;
    unsigned char digest[16];
    char tmp[22];
    int i, k, d;
    unsigned int x;
    static const char enc_table[64] =
    "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_@";

    apr_md5_init(&context);
    apr_md5_update(&context, (const unsigned char *) it, strlen(it));
    apr_md5_final(digest, &context);

    /* encode 128 bits as 22 characters, using a modified uuencoding
     * the encoding is 3 bytes -> 4 characters* i.e. 128 bits is
     * 5 x 3 bytes + 1 byte -> 5 * 4 characters + 2 characters
     */
    for (i = 0, k = 0; i < 15; i += 3) {
        x = (digest[i] << 16) | (digest[i + 1] << 8) | digest[i + 2];
        tmp[k++] = enc_table[x >> 18];
        tmp[k++] = enc_table[(x >> 12) & 0x3f];
        tmp[k++] = enc_table[(x >> 6) & 0x3f];
        tmp[k++] = enc_table[x & 0x3f];
    }

    /* one byte left */
    x = digest[15];
    tmp[k++] = enc_table[x >> 2];    /* use up 6 bits */
    tmp[k++] = enc_table[(x << 4) & 0x3f];

    /* now split into directory levels */
    for (i = k = d = 0; d < ndepth; ++d) {
        memcpy(&val[i], &tmp[k], nlength);
        k += nlength;
        val[i + nlength] = '/';
        i += nlength + 1;
    }
    memcpy(&val[i], &tmp[k], 22 - k);
    val[i + 22 - k] = '\0';
}

这是我转换为 PHP 的尝试:

$URL = 'thumbnail.php?u=http%3A%2F%2Fwww.google.com%2Flogos%2Fclassicplus.png&w=200';

cache_hash($URL, $theFile);

function cache_hash($URL, &$theFile, $dirlevels = 3, $dirlength = 1)
{
    $enc_table = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_@";
    $md5URL = md5($URL,true);

//  echo "MD5URL: ".var_export($md5URL,true);
    //$md5URL = digest

    $tmp = '0000000000000000000000';    //22 chars
    for ($i = 0; $i < 22; $i++) {
        $tmp[$i] = 0x0;
    }

    var_export(unpack('S',$md5URL));
    p(ord($md5URL[0]));
    p(ord($md5URL[1]));

    /* encode 128 bits as 22 characters, using a modified uuencoding
     * the encoding is 3 bytes -> 4 characters* i.e. 128 bits is
    * 5 x 3 bytes + 1 byte -> 5 * 4 characters + 2 characters
    */
    for ($i = 0, $k = 0; $i < 15; $i += 3) {
//      p($md5URL[$i] << 16);
//      p($md5URL[$i + 1] << 8);
//      p($md5URL[$i + 2]);


        $x = ($md5URL[$i] << 16) | ($md5URL[$i + 1] << 8) | $md5URL[$i + 2];
//      p($x);
        p($x >> 18);
        $tmp[$k++] = $enc_table[$x >> 18];
        $tmp[$k++] = $enc_table[($x >> 12) & 0x3f];
        $tmp[$k++] = $enc_table[($x >> 6) & 0x3f];
        $tmp[$k++] = $enc_table[$x & 0x3f];
    }

    /* one byte left */
    $x = $md5URL[15];
    $tmp[$k++] = $enc_table[$x >> 2];    /* use up 6 bits */
    $tmp[$k++] = $enc_table[($x << 4) & 0x3f];

    /* now split into directory levels */
    for ($i = $k = $d = 0; $d < $dirlevels; ++$d) {
var_export($tmp);
        $theFile[$i] = substr($tmp,$k,$dirlength);
        $k += $dirlength;
        $theFile[$i + $dirlength] = '/';
        $i += $dirlength + 1;
    }
    $theFile[$i] = substr($tmp,$k,22 - $k);

    var_export($theFile);
}

function p($val) {
    echo " Dec:\n";
    printf('  val= %d', ord($val));

    echo "\nBinary:\n";
    printf('  val= %b', ord($val));

echo "\n";
}

I'm trying to convert an APR (apache runtime) method to PHP, but I can't quite figure it out.

The method creates a string that is a directory path and file - so example output would be A/B/C/[a-fA-F0-9_@]. It's using bitwise operations on a binary string returned from md5($string,true); When I run the php below, I get:

array (
  0 => 'A',
  1 => '/',
  2 => 'A',
  3 => '/',
  4 => 'A',
  5 => '/',
  6 => 'AAAAAAAAAAAAAAAAAAA',
)

What am I doing wrong? Is this not possible due to how PHP internally represents char arrays?

Here is the c method:

static void cache_hash(const char *it, char *val, int ndepth, int nlength)
{
    apr_md5_ctx_t context;
    unsigned char digest[16];
    char tmp[22];
    int i, k, d;
    unsigned int x;
    static const char enc_table[64] =
    "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_@";

    apr_md5_init(&context);
    apr_md5_update(&context, (const unsigned char *) it, strlen(it));
    apr_md5_final(digest, &context);

    /* encode 128 bits as 22 characters, using a modified uuencoding
     * the encoding is 3 bytes -> 4 characters* i.e. 128 bits is
     * 5 x 3 bytes + 1 byte -> 5 * 4 characters + 2 characters
     */
    for (i = 0, k = 0; i < 15; i += 3) {
        x = (digest[i] << 16) | (digest[i + 1] << 8) | digest[i + 2];
        tmp[k++] = enc_table[x >> 18];
        tmp[k++] = enc_table[(x >> 12) & 0x3f];
        tmp[k++] = enc_table[(x >> 6) & 0x3f];
        tmp[k++] = enc_table[x & 0x3f];
    }

    /* one byte left */
    x = digest[15];
    tmp[k++] = enc_table[x >> 2];    /* use up 6 bits */
    tmp[k++] = enc_table[(x << 4) & 0x3f];

    /* now split into directory levels */
    for (i = k = d = 0; d < ndepth; ++d) {
        memcpy(&val[i], &tmp[k], nlength);
        k += nlength;
        val[i + nlength] = '/';
        i += nlength + 1;
    }
    memcpy(&val[i], &tmp[k], 22 - k);
    val[i + 22 - k] = '\0';
}

Here is my attempt at converting to PHP:

$URL = 'thumbnail.php?u=http%3A%2F%2Fwww.google.com%2Flogos%2Fclassicplus.png&w=200';

cache_hash($URL, $theFile);

function cache_hash($URL, &$theFile, $dirlevels = 3, $dirlength = 1)
{
    $enc_table = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_@";
    $md5URL = md5($URL,true);

//  echo "MD5URL: ".var_export($md5URL,true);
    //$md5URL = digest

    $tmp = '0000000000000000000000';    //22 chars
    for ($i = 0; $i < 22; $i++) {
        $tmp[$i] = 0x0;
    }

    var_export(unpack('S',$md5URL));
    p(ord($md5URL[0]));
    p(ord($md5URL[1]));

    /* encode 128 bits as 22 characters, using a modified uuencoding
     * the encoding is 3 bytes -> 4 characters* i.e. 128 bits is
    * 5 x 3 bytes + 1 byte -> 5 * 4 characters + 2 characters
    */
    for ($i = 0, $k = 0; $i < 15; $i += 3) {
//      p($md5URL[$i] << 16);
//      p($md5URL[$i + 1] << 8);
//      p($md5URL[$i + 2]);


        $x = ($md5URL[$i] << 16) | ($md5URL[$i + 1] << 8) | $md5URL[$i + 2];
//      p($x);
        p($x >> 18);
        $tmp[$k++] = $enc_table[$x >> 18];
        $tmp[$k++] = $enc_table[($x >> 12) & 0x3f];
        $tmp[$k++] = $enc_table[($x >> 6) & 0x3f];
        $tmp[$k++] = $enc_table[$x & 0x3f];
    }

    /* one byte left */
    $x = $md5URL[15];
    $tmp[$k++] = $enc_table[$x >> 2];    /* use up 6 bits */
    $tmp[$k++] = $enc_table[($x << 4) & 0x3f];

    /* now split into directory levels */
    for ($i = $k = $d = 0; $d < $dirlevels; ++$d) {
var_export($tmp);
        $theFile[$i] = substr($tmp,$k,$dirlength);
        $k += $dirlength;
        $theFile[$i + $dirlength] = '/';
        $i += $dirlength + 1;
    }
    $theFile[$i] = substr($tmp,$k,22 - $k);

    var_export($theFile);
}

function p($val) {
    echo " Dec:\n";
    printf('  val= %d', ord($val));

    echo "\nBinary:\n";
    printf('  val= %b', ord($val));

echo "\n";
}

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(1

滥情哥ㄟ 2025-01-08 11:02:50

更改这两行,它应该可以工作我相信

$x = (ord($md5URL[$i]) << 16) | (ord($md5URL[$i + 1]) << 8) | ord($md5URL[$i + 2]);

$x = ord($md5URL[15]);

我认为你也必须改变这一行:
$md5URL = md5($URL, false);

change the two lines, it should work I believe

$x = (ord($md5URL[$i]) << 16) | (ord($md5URL[$i + 1]) << 8) | ord($md5URL[$i + 2]);

$x = ord($md5URL[15]);

I think you have to change this line too:
$md5URL = md5($URL, false);

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文