我编写了一个c++库,它公开了许多类。我还使用zend编写了一个php扩展,作为c++库的包装器。
我有一个正确序列化我的对象的问题,例如,当我试图将它们存储在$_SESSION中时。
下面是一个例子:
struct spider_object
{
zend_object m_std;
Spider::QGramTokenizer* m_pObject;
};
...
zend_class_entry *spider_QGramTokenizer_ce;
PHP_METHOD(spider_QGramTokenizer, __construct)
{
long lQGramSize;
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &lQGramSize) == FAILURE)
{
WRONG_PARAM_COUNT;
}
Spider::QGramTokenizer* pQT = new Spider::QGramTokenizer(static_cast<uint16_t>(lQGramSize));
spider_object* pO = static_cast<spider_object*>(zend_object_store_get_object(getThis() TSRMLS_CC));
pO->m_pObject = pQT;
}
PHP_METHOD(spider_QGramTokenizer, getQGramSize)
{
spider_object *pO = static_cast<spider_object*>(zend_object_store_get_object(getThis() TSRMLS_CC));
Spider::QGramTokenizer* pQT = static_cast<Spider::QGramTokenizer*>(pO->m_pObject);
if (pQT != NULL)
{
RETURN_LONG(pQT->getQGramSize());
}
else
zend_error(E_ERROR, "Null QGramTokenizer instance.'n");
}
我在我的脚本中创建了一个spider_QGramTokenizer对象,它工作得很好。一旦我将其存储为会话变量并尝试从不同的脚本获取它,我就可以很好地获得对象,但如果我调用getQGramSize,那么我就会得到一个"Null QGramTokenizer实例"错误。
换句话说,serialize()不知道如何__sleep或__wakeup spider_object::m_pObject实例,它只是将其设置为null。或者,实际上是存储在zend_object_store中的整个spider_object实例根本不序列化。
我想重写__sleep和__wake - up,但我不知道zend在下面做什么引擎盖和__sleep的返回数组需要是什么,以及我如何使用我自己的m_object相关的数据在那里?这可能吗?
当PHP中的对象被反序列化时,构造函数不会(再次)被调用,因为实例已经被构造(然后被序列化,然后最后被反序列化)。这至少可以解释你看到的行为(比较:原则2如何在不调用实体构造函数的情况下检索实体?)。
我不知道PHP扩展API为反序列化提供了哪些例程,但是如果您能够将c++对象的实例存储在PHP私有成员中,那么它将在序列化后可用。
我不能说如果__sleep
和__wakeup
可以在C代码中定义,可能这是有效的。但我不是PHP扩展开发人员,所以我在这方面没有专业知识。