PHP Hessian协议研究学习

服务是Hessian for java 4.0版实现,客户端用PHP Hessianv2.0.2实现,用php发送raw binary给服务器端,服务器端抛出异常:
com.caucho.hessian.io.HessianProtocolException: '2' is an unknown class definition

为什么是'2'?
因为raw binary是0x62打头(十进制98),读取后赋值给tag,然后有如下代码:
int ref = tag - 0x60;
int size = _classDefs.size();

if (ref < 0 || size <= ref)
  throw new HessianProtocolException("'" + ref + "' is an unknown class definition");

这样的话,ref=tag-0x60值就是2了,而size是0,故而抛出异常。

这样在用php给java服务发送hessian请求时,raw binary的基本上都会接收不了,解决方法寻找中。

最终发现还是不行。Hessian2.0协议规范如是写:
60 - x6f    # object with direct type
而在Hessian Java解析中有这样的判断:
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中没有与之对应的direct type,从而必然失败。

解决办法也是有的,实际上通过php读取一个文件内容:
$data = fread($fh, $_FILES["filename"]["size"])
这个$data可以看作一个大字符串,通过HessianPHP的writeString方法发送给服务器端。但由于HessianPHP还有很多TODO,对于大字符串的传输没有实现by chunks,于是参考Hessian的协议规范填补了这一块:
Hessian2Writer.php

function writeString($value){
$len = HessianUtils::stringLength($value);
if($len < 32){
return pack('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;
                          
            //zhuyibo added. chunks transfer of large string
            $offset = 0;
            $stream = '';
            while($total > 0x8000) { //each chunk has 2^15 16-bit characters
            $subLen = 0x8000;
    $tag = 'R';    //x52 ('R') represents any non-final chunk
    $stream .= $tag . pack('n', $subLen); //x52 b1 b0 <utf8-data> string
                $data = HessianUtils::subString($value, $offset, $subLen); //utf8-data part
    $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");

从而可以开始后续处理。

这种方式相对于raw binary方式而言,因为也是采用chunks传输,每个chunk大小是2^15次方个双字节字符,而采用raw binary是每次读取32768个字节再进行传输,所以性能上不会有大的影响。只是需要一次把整个文件全部读取出来,内存消耗是个问题。


猜你喜欢

转载自quietflow.iteye.com/blog/1084302