4

这是使用的“C++”结构。

struct gross
    {
        char date[11];
        char ac[128];
        char type[5];
        float mvalue;
        float netraw;
        float netfer;
        char stat[128];
        float firr;
        float acb;
    };

这是使用 GCC 编译器编译时生成的二进制文件的内容。

12/12/1995 D US 12.23 34.12 90.12费用12 56.12 01/01/1998 A US 52.23 54.12 10.12费用92 16.12 31/12/1999 A US 52.23 54.12 10.12费用92 16.12 31/12/1999 D US 12.23 34.12 90.12费用12 56.12 01/01/01/01/01/01/01/01/01/01/01/01/01/01/01/01/01/211.12费用92 16.12 01/01/2000 Z US 12.23 34.12 90.12费用12 56.12 31/12/2010 A US 52.23 54.12 10.12费用92 16.12 31/12/2010 D US 12.23 34.12 90.12 费用 12 56.12

读取上述二进制文件内容的 PHP 编码。

echo "<table>";
while (!feof($f)) {

  if ($s = fread($f, 292)) {

        $nn = unpack('a11date/a128ac/a5type/fmvalue/fnetraw/fnetfer/a128stat/ffirr/facb', $s);

        echo "<td>" . $nn[date] ."</td>";
        echo "<td>" . $nn[ac] . "</td>";
        echo "<td>" . $nn[type] . "</td>";
        echo "<td>" . $nn[mvalue] . "</td>";
        echo "<td>" . $nn[netraw] . "</td>";
        echo "<td>" . $nn[netfer] . "</td>";
        echo "<td>" . $nn[stat] . "</td>";
        echo "<td>" . $nn[firr] . "</td>";
        echo "<td>" . $nn[acb] . "</td>";
        echo "</tr>";
 }
}
        echo "</table>";
fclose($f);
?>

这是我从上面的代码中得到的。我得到的是第二个和第七个字段中有很多垃圾值。并且精度超过了浮点字段。如何解决这个问题?

12/12/1995 d¸³M·g·ew·¤dw·àew·Öúr·ÿÿÿÿ,w·ø;w·Ìª¹¿ô{·{·‹ây·Ð{·SpP·ip·¤dw·ew· €5w·ôOg· X«¹¿»qP·äSg us 12.229999542236 34.119998931885 90.120002746582 费用·5rP· #Ä äSg· rP·ô›Å #Ä 5rP·0#o·¸xw·x«¹¿·ÅÿÿýŽäSg·¸x ·ÆD¬¹¿L¬¹¿5rP·pòy· 12 56.119998931885

01/01/1998 a¸³M·g·ew·¤dw·àew·Öúr·ÿÿÿÿ,w·ø;w·Ìª¹¿ô{·{·‹ây·Ð{·SpP·ip·¤dw·ew· €5w·ôOg· X«¹¿»qP·äSg us 52.229999542236 54.119998931885 10.119999885559 费用·5rP· #Ä äSg· rP·ô›Å #Ä 5rP·0#o·¸xw·x«¹¿·ÅÿÿýŽäSg·¸x ·ÆD¬¹¿L¬¹¿5rP·pòy· 92 16.120000839233

1999 年 12 月 31 日 a¸³M·g·ew·¤dw·àew·Öúr·ÿÿÿÿ,w·ø;w·Ìª¹¿ô{·{·‹ây·Ð{·SpP·ip·¤dw·ew· €5w·ôOg· X«¹¿»qP·äSg us 52.229999542236 54.119998931885 10.119999885559 费用·5rP· #Ä äSg· rP·ô›Å #Ä 5rP·0#o·¸xw·x«¹¿·ÅÿÿýŽäSg·¸x ·ÆD¬¹¿L¬¹¿5rP·pòy· 92 16.120000839233

31/12/1999 d¸³M·g·ew·¤dw·àew·Öúr·ÿÿÿÿ,w·ø;w·Ìª¹¿ô{·{·‹ây·Ð{·SpP·ip·¤dw·ew· €5w·ôOg· X«¹¿»qP·äSg us 12.229999542236 34.119998931885 90.120002746582 费用·5rP· #Ä äSg· rP·ô›Å #Ä 5rP·0#o·¸xw·x«¹¿·ÅÿÿýŽäSg·¸x ·ÆD¬¹¿L¬¹¿5rP·pòy· 12 56.119998931885

01/01/2000 a¸³M·g·ew·¤dw·àew·Öúr·ÿÿÿÿ,w·ø;w·Ìª¹¿ô{·{·‹ây·Ð{·SpP·ip·¤dw·ew· €5w·ôOg· X«¹¿»qP·äSg us 52.229999542236 54.119998931885 10.119999885559 费用·5rP· #Ä äSg· rP·ô›Å #Ä 5rP·0#o·¸xw·x«¹¿·ÅÿÿýŽäSg·¸x ·ÆD¬¹¿L¬¹¿5rP·pòy· 92 16.120000839233

01/01/2000 z¸³M·g·ew·¤dw·àew·Öúr·ÿÿÿÿ,w·ø;w·Ìª¹¿ô{·{·‹ây·Ð{·SpP·ip·¤dw·ew· €5w·ôOg· X«¹¿»qP·äSg us 12.229999542236 34.119998931885 90.120002746582 费用·5rP· #Ä äSg· rP·ô›Å #Ä 5rP·0#o·¸xw·x«¹¿·ÅÿÿýŽäSg·¸x ·ÆD¬¹¿L¬¹¿5rP·pòy· 12 56.119998931885

31/12/2010 a¸³M·g·ew·¤dw·àew·Öúr·ÿÿÿÿ,w·ø;w·Ìª¹¿ô{·{·‹ây·Ð{·SpP·ip·¤dw·ew· €5w·ôOg· X«¹¿»qP·äSg us 52.229999542236 54.119998931885 10.119999885559 费用·5rP· #Ä äSg· rP·ô›Å #Ä 5rP·0#o·¸xw·x«¹¿·ÅÿÿýŽäSg·¸x ·ÆD¬¹¿L¬¹¿5rP·pòy· 92 16.120000839233

31/12/2010 d¸³M·g·ew·¤dw·àew·Öúr·ÿÿÿÿ,w·ø;w·Ìª¹¿ô{·{·‹ây·Ð{·SpP·ip·¤dw·ew· €5w·ôOg· X«¹¿»qP·äSg us 12.229999542236 34.119998931885 90.120002746582 费用·5rP· #Ä äSg· rP·ô›Å #Ä 5rP·0#o·¸xw·x«¹¿·ÅÿÿýŽäSg·¸x ·ÆD¬¹¿L¬¹¿5rP·pòy· 12 56.119998931885

4

2 回答 2

0

在 PHP 的packandunpack中,格式化代码a代表 NUL填充字符串。在这种情况下,看起来第二个和第七个字段没有填充 NUL;他们只有 NUL 来指示字符串的结束位置,然后是随机数据。

要在 NUL 之前获取字符串的一部分,您可以使用substrand strpos

$input = "a\000b"; // string with embedded NUL 
$output = substr($input, 0, strpos($input, "\000"));
var_dump($output); // string(1) "a"

浮点字段具有正确的值,您在 C 程序的输出中看到的值是四舍五入的。要在 PHP 中执行相同的操作,您可以使用sprintf例如:

$input = 16.120000839233;
$output = sprintf("%.2f", $input);
var_dump($output); // string(5) "16.12"
于 2012-06-08T08:10:01.443 回答
0

是的,因为您解压的第二个参数是“a128ac”,而手册(http://jp2.php.net/manual/en/function.pack.php)说“a”代表“NUL 填充字符串”,但您可能想要“A”是“空格填充的字符串”。你看到的是垃圾,因为没有正确找到终结器。

于 2012-06-08T08:10:32.333 回答