PHP Hessian协议研究学习

Crazyshark 2011-06-13

服务是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个字节再进行传输,所以性能上不会有大的影响。只是需要一次把整个文件全部读取出来,内存消耗是个问题。

相关推荐