PHP Hessian协议研究学习
服务是Hessianforjava4.0版实现,客户端用PHPHessianv2.0.2实现,用php发送rawbinary给服务器端,服务器端抛出异常:
com.caucho.hessian.io.HessianProtocolException:'2'isanunknownclassdefinition
为什么是'2'?
因为rawbinary是0x62打头(十进制98),读取后赋值给tag,然后有如下代码:
intref=tag-0x60;
intsize=_classDefs.size();
if(ref<0||size<=ref)
thrownewHessianProtocolException("'"+ref+"'isanunknownclassdefinition");
这样的话,ref=tag-0x60值就是2了,而size是0,故而抛出异常。
这样在用php给java服务发送hessian请求时,rawbinary的基本上都会接收不了,解决方法寻找中。
最终发现还是不行。Hessian2.0协议规范如是写:
60-x6f#objectwithdirecttype
而在HessianJava解析中有这样的判断:
public Object readObject(Class cl){ ... case 0x60: case 0x61: case 0x62: case 0x63: case 0x64: case 0x65: case 0x66: case 0x67: case 0x68: case 0x69: case 0x6a: case 0x6b: case 0x6c: case 0x6d: case 0x6e: case 0x6f: { int ref = tag - 0x60; int size = _classDefs.size(); if (ref < 0 || size <= ref) throw new HessianProtocolException("'" + ref + "' is an unknown class definition"); ObjectDefinition def = _classDefs.get(ref); return readObjectInstance(cl, def); } ... }
而在_classDefs中没有与之对应的directtype,从而必然失败。
解决办法也是有的,实际上通过php读取一个文件内容:
$data=fread($fh,$_FILES["filename"]["size"])
这个$data可以看作一个大字符串,通过HessianPHP的writeString方法发送给服务器端。但由于HessianPHP还有很多TODO,对于大字符串的传输没有实现bychunks,于是参考Hessian的协议规范填补了这一块:
Hessian2Writer.php
functionwriteString($value){
$len=HessianUtils::stringLength($value);
if($len<32){
returnpack('C',$len)
.$this->writeStringData($value);
}else
if($len<1024){
$b0=0x30+($len>>;
$stream=pack('C',$b0);
$stream.=pack('C',$len);
return$stream.$this->writeStringData($value);
}else{
//TODO:chunks
$total=$len;
//zhuyiboadded.chunkstransferoflargestring
$offset=0;
$stream='';
while($total>0x8000){//eachchunkhas2^1516-bitcharacters
$subLen=0x8000;
$tag='R';//x52('R')representsanynon-finalchunk
$stream.=$tag.pack('n',$subLen);//x52b1b0<utf8-data>string
$data=HessianUtils::subString($value,$offset,$subLen);//utf8-datapart
$stream.=$this->writeStringData($data);
$total-=$subLen;
$offset+=$subLen;
}
$tag='S';
$stream.=$tag.pack('n',$total);
$data=HessianUtils::subString($value,$offset,$total);
$stream.=$this->writeStringData($data);
return$stream;
}
}
其中HessianUtils::subString等方法也是自己实现的。
这样就可以将一个二进制块当作字符串传输给服务器端了。在服务器端用字符串接收到后,可以采用如下方式得到字节数组:
byte[] dataBytes = dataStr.getBytes("ISO-8859-1");
从而可以开始后续处理。
这种方式相对于rawbinary方式而言,因为也是采用chunks传输,每个chunk大小是2^15次方个双字节字符,而采用rawbinary是每次读取32768个字节再进行传输,所以性能上不会有大的影响。只是需要一次把整个文件全部读取出来,内存消耗是个问题。