将使用按位的c方法转换为php
我正在尝试将 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 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(1)
更改这两行,它应该可以工作我相信
$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);