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