公司当前的一个项目,主体上是用的J2EE架构,在一些与第三方的接入环节使用到了php,二者之间一部分数据的同步是使用的memcached作为中间件的。目前的问题是:Java写入,php读取,没有问题,但php写入的字符串,Java读出来却是一个Long类型的数据。多番尝试以及百度(实在是用不了google)之后,找到原因所在:
memcached中,存储键值时,有一个flags标志,用于接入程序自己来对值类型做识别,Java中String类型该flags值为32(据查询.net中是2),而php底层写入是并没有根据类型写入flags值,我想应该是php的变量本身是范类型的,所以它不需要设置。这就造成了用Java读取时,按照flags=0去构造对象,引起取得的值不一致。
到此,解决问题的点就是能不能把这个flags值设置得一致,那么就皆大欢喜了。
重点来了,解决办法:找到memcached扩展的源码,把其中调用memcached方法的地方,设置flags为32 ,编译安装,重启web即可。我用的是memcached-2.2.0.tgz
步骤:
1、进入memcached-2.2.0 ,打开php_memcache.c
2、找到 static void php_memc_store_impl 方法,在 switch (op) { 语句上方,加入 flags = 32; 注意,这样做是一个权益之计,把php设置的变量值统统按string类型存储,本身不影响php的存取,而在Java端读取的时候,就要按字符串的方式处理,如果是对象,就要反序列化操作一次
3、退出vi,执行
1) /usr/local/php/bin/phpize
2) ./configure --with-php-config=/usr/local/php/bin/php-config
3) make && make install
至此,重新安装php的memcached扩展完成
在memcached的命令行模式下,get key返回的数据和之前的不一样了:
VALUE key_abc 0 17
Jdhqycng783789269
现在是:
VALUE key_abc 32 17
Jdhqycng783789269
区别就在这个32上,是数据的flags
再赠送一个php的简单工具类:
<?php
class MemcachedUtils {
private $m;
public function __construct($ip="localhost",$port=11211) {
$this->m = new Memcached('mc');
$this->m->setOption(Memcached::OPT_LIBKETAMA_COMPATIBLE, true);
if (!count($this->m->getServerList())) { //在服务器池中查询,如果已有,则不再添加server
$this->m->addServers(array(
array($ip,$port)
));
}
}
public function get($key) {
return $this->m->get($key);
}
/*
* 注意:此处 $time 单位为秒
*/
public function set($key,$val,$time) {
$this->m->set($key,$val,$time);
}
}
?>
使用方法很简单:
$memcache = new MemcachedUtils();
$memcache->set($key,$val,2*60*60); //保存2小时