每日学习笔记(6)

今天在做的事情是将一个as3项目的部分代码移植到PHP中来,记录下移植过程中遇到的两个问题。

1AS3String类的charCodeAt函数是返回字符的unicode编码,而在PHP中并没有直接与之对应的库函数,可以用下面这个函数做替代:

复制代码
代码
<?php
function uniord($str,$from_encoding=false){
        
$from_encoding=$from_encoding ? $from_encoding : 'UTF-8';

        
if(strlen($str)==1
                
return ord($str);
                
        
$str=mb_convert_encoding($str, 'UCS-4BE', $from_encoding);
        
$tmp=unpack('N',$str);
        
return $tmp[1];
}
$str="12345";
$result=array();
for($i=0,$l=mb_strlen($str,'utf-8');$i<$l;++$i){
        
$result[]=uniord(mb_substr($str,$i,1,'utf-8'));
}
echo join(",",$result);
?>
复制代码

2,PHP的二进制位移操作

PHP主要是设计于文本操作的,其实PHP不适合做数学运算,效率也不高,不过因为AS3项目中有大量的二进制位移操作,在移植到PHP来时遇到了麻烦。

因为PHP只有32位有符号整数,没有64位长整型,也没有无符号整数。其整型的范围是-231-1231,超出这个范围的,将被解释为浮点数。因此,0xFFFFFFFF,直接打印,显示的是4294967295,及232

>> 0xFFFFFFFFF
4294967295
>>  gettype(0xFFFFFFFF)
'double'

  而在32位有符号整型中,0xFFFFFFFF应表示-1

 >> (int)0xFFFFFFFFF
-
1

  而PHP不支持浮点数的二进制位移操作,如果要进行,会先转换为整型,最后的结果,也将按照整型来返回:

复制代码
>> 1 << 31
-
2147483648
>> 1 << 30
1073741824
>> 1 << 32
1
>> 0xFFFFFFFF >> 1
-
1
复制代码

  同时PHP的向右位移操作,高位会填充符号位,而且PHP没有提供类似Java>>>来强制填充0

复制代码
>> 1 << 32
1
>> 0xFFFFFFFF >> 1
-
1
>> 0xFFFFFFFF >> 2
-
1
>> 0xFFFFFFFF >> 3
-
1
>> 0xFFFFFFFF >> 31
-
1
复制代码

  如何解决这个问题呢,我考虑过使用BCMath数学函数库,直接处理字符串表示的整数,或者是GMP/BigInt扩展等。

不过我想既然使用字符串,那么我可以字符串地彻底一些,把数字转换成32个二进制的字符串,再手工填充0,最后转换回来。

逻辑移位的代码如下:

复制代码
代码
<?php  
/*
 * 无符号32位右移 
 * @param mixed $x 要进行操作的数字,如果是字符串,必须是十进制形式 
 * @param string $bits 右移位数 
 * @return mixed 结果,如果超出整型范围将返回浮点数 
 
*/  
function shr32($x, $bits){  
    
// 位移量超出范围的两种情况  
    if($bits <= 0){  
        
return $x;  
    }  
    
if($bits >= 32){  
        
return 0;  
    }  
    
//转换成代表二进制数字的字符串  
    $bin = decbin($x);  
    
$l = strlen($bin);  
    
//字符串长度超出则截取底32位,长度不够,则填充高位为0到32位  
    if($l > 32){  
        
$bin = substr($bin, $l - 32, 32);  
    }
elseif($l < 32){  
        
$bin = str_pad($bin, 32, '0', STR_PAD_LEFT);  
    }  
    
//取出要移动的位数,并在左边填充0  
    return bindec(str_pad(substr($bin, 0, 32 - $bits), 32, '0', STR_PAD_LEFT));  
}  
/*
 * 无符号32位左移 
 * @param mixed $x 要进行操作的数字,如果是字符串,必须是十进制形式 
 * @param string $bits 左移位数 
 * @return mixed 结果,如果超出整型范围将返回浮点数 
 
*/  
function shl32 ($x, $bits){  
    
// 位移量超出范围的两种情况  
    if($bits <= 0){  
        
return $x;  
    }  
    
if($bits >= 32){  
        
return 0;  
    }  
    
//转换成代表二进制数字的字符串  
    $bin = decbin($x);  
    
$l = strlen($bin);  
    
//字符串长度超出则截取底32位,长度不够,则填充高位为0到32位  
    if($l > 32){  
        
$bin = substr($bin, $l - 32, 32);  
    }
elseif($l < 32){  
        
$bin = str_pad($bin, 32, '0', STR_PAD_LEFT);  
    }  
    
//取出要移动的位数,并在右边填充0  
    return bindec(str_pad(substr($bin, $bits), 32, '0', STR_PAD_RIGHT));  
}  
?>  
复制代码

 

 

posted on   Phinecos(洞庭散人)  阅读(931)  评论(0编辑  收藏  举报

编辑推荐:
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
阅读排行:
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· .NET周刊【3月第1期 2025-03-02】
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· [AI/GPT/综述] AI Agent的设计模式综述
历史上的今天:
2008-07-08 基于MFC的Direct3D程序设计
2008-07-08 一个动态数组类
2008-07-08 Windows下搭建Subversion 服务器

导航

统计

点击右上角即可分享
微信分享提示