尽管php是用C语言开发的,不过令我不解的是php没有提供对结构体struct的直接支持。
不过php提供了pack和unpack函数,用来进行二进制数据(binary data)和php内部数据的互转:
string pack ( string $format [, mixed $args [, mixed $...]] ) //Pack given arguments into binary string according to format. array unpack ( string $format, string $data ) //Unpacks from a binary string into an array according to the given format.
其中,$format跟perl里的pack格式类似,有如下一些(中文是我加的,有不准确的欢迎提出):
a NUL-padded string,即“\0”作为“空字符”的表示形式
A SPACE-padded string,空格作为“空字符”的表示形式
h Hex string, low nibble first,升序位顺序
H Hex string, high nibble first,降序位顺序
c signed char,有符号单字节
C unsigned char,无符号单字节
s signed short (always 16 bit, machine byte order)
S unsigned short (always 16 bit, machine byte order)
n unsigned short (always 16 bit, big endian byte order)
v unsigned short (always 16 bit, little endian byte order)
i signed integer (machine dependent size and byte order)
I unsigned integer (machine dependent size and byte order)
l signed long (always 32 bit, machine byte order)
L unsigned long (always 32 bit, machine byte order)
N unsigned long (always 32 bit, big endian byte order)
V unsigned long (always 32 bit, little endian byte order)
f float (machine dependent size and representation)
d double (machine dependent size and representation)
x NUL byte,实际使用的时候作为跳过多少字节用,很有用
X Back up one byte,后退1字节
@ NUL-fill to absolute position,实际使用的时候作为从开头跳到某字节用,很有用
实际使用发现:C里的“\0”(即字符串终止符)在php里并不是终止符,而是作为了字符串的一部分。因此,必须对“\0”进行特殊处理,才能进行
struct和php内部数据的完美互转。比如 char name[10]; 如果实际数据是“62 69 61 6E 00 62 69 61
6E
00”,在C语言里第5个位置有终止符,name应该是“bian”;而用了unpack转换以后在php里的name却是“bian\0bian
\0”。
一开始我用了strpos函数找到“\0”的位置,然后进行substr截取.
不过很Faint的事情发生了,不知道是strpos的bug还是substr的bug(其实测试一下就知道,懒得试),有些字符串没问题,有些字符串却只能得到空值(即$name == ”)。很是郁闷,后来找了个strtok函数,这下没有问题了.
难为大家看了那么多,下面写个完整的php读取二进制数据流(C语言结构体struct数据)文件的示例代码:
首先是C的struct定义示例,为了演示,我就写个简单点的,实际对照上面那个$format格式表应该没有问题:
struct TEST{ char name[10]; char pass[33]; int age; unsigned char flag; };
比如有个“file.dat”文件,内容就是上面的N个BIANBIAN结构体构成的。读取的php代码:
<?php //下面根据struct确定$format,注意int类型跟机器环境有关,我的32位Linux是4个长度 $format = 'a10name/a33pass/iage/Cflag'; //确定一个struct占用多少长度字节,如果只是读取单个结构体这是不需要的 $length = 10 + 33 + 4 + 1; //也可以用fopen + fread + fclose,不过file_get_contents因为可以mmap,效率更高 $data = file_get_contents('file.dat', 'r'); for ($i = 0, $c = strlen($data); $i < $c; $i += $length) { $bianbian = unpack("$format", $data); //reference传递是php 5才支持的,如果用php4,得用其他办法 foreach ($bianbian as &$value) { if (is_string($value)) { $value = strtok($value, "\0"); } } print_r($bianbian); } ?>
pack应该跟unpack相反。
顺便附上生成结构体文件的C语言代码:
#include <stdio.h> #include <string.h> struct example { char name[10]; char pass[33]; int age; unsigned char flag; }; int main() { example test; example read; FILE *fp; test.age = 111; test.flag = 10; strcpy(test.name, "Hello World!"); strcpy(test.pass, "zbl110119"); fp = fopen("file.dat", "w+"); if (!fp) { printf("open file error!"); return -1; } rewind(fp); fwrite(&test, sizeof(example), 1, fp); rewind(fp); fread(&read, sizeof(example), 1, fp); printf("%d, %s\n", read.age, read.name); fclose(fp); return 0; }
转自:http://bianbian.org http://www.zbl110119.cn/archives/130/comment-page-1
相关推荐
本篇文章是对php读取二进制流(C语言结构体struct数据文件)进行了详细的分析介绍,需要的朋友参考下
数据用二进制方式保存无疑是能减小数据文件体积!此类就是鉴此思想编写 类包括数据写入 读写(文件头(结构体)部分 记录数据基本参数和文件的数据块数量 正文部分为存数据(结构体))高效实现块写 块读 !
C语言结构体STRUCT用法.pdf
C语言结构体struct常见使用方法.doc
C语言结构体与共用体,C语言结构体与共用体的使用方法介绍
将结构体数据写入二进制文件,从二进制文件中读取结构体数据
详细的讲解了c语言的机构体,文件的相关内容,言简意赅。是PPT形式的。
详解C语言Struct(结构体)系列的第一部分的全部文章合集,具体分析文章见: http://blog.csdn.net/column/details/structure.html 第二季近期会很快提供
C语言全套资料 C语言程序设计 C语言算法 C语言课件 C语言顺序程序设计,C语言数组,C语言循环控制,C语言预处理命令,C语言文件操作指针,C语言选择结构程序设计,C语言结构体与共用体,C语言文件操作,C语言函数
主要功能:能友好的用java处理任何发送的C结构体对象,并且能发送java对象转换成C结构体接收的二进制。 功能说明 1、基于spring框架开发 2、对于结构体定义采用可视化XML配置,便于维护与调试,防止协议多次修改带来...
C语言教学课件:c语言结构体.ppt
文章主要对C语言结构体的位域知识进行详细介绍。
简单的C语言结构体范例代码,需要的可以拿走参考一下
C语言结构体部分内容详解包括结构体定义及使用,结构体函数的定义及使用,指向结构体的指针
如何简单的实现结构体方法,另外规范的写代码方式
就是结构体的用法,C语言结构体的用法.ppt
深入解析JNA—模拟C语言结构体
详细介绍c语言结构体与公用体, ppt课件