序列化可通过php扩展访问的c++类的问题


Problem serializing c++ class accessible through a php extension

我编写了一个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扩展开发人员,所以我在这方面没有专业知识。