Неполный класс PHP при сериализации объекта в сеансах


Я работаю над корзиной покупок (модель корзины). Одним из его защищенных свойств является "_items", который содержит массив объектов продукта. Все они (продукты) сохраняются в БД для заполнения сеанса (с помощью ZF, Zend_Session_SaveHandler_DbTable() и т.д.).

public function addItem(Model_Product $product, $qty)
{
    $qty = (int) $qty;
    $pId = $product->getId();

    if ($qty > 0) {
        $this->_items[$pId] = array('product' => $product, 'qty' => $qty);
    } else {
        // if the quantity is zero (or less), remove item from stack
        unset($this->_items[$pId]);
    }

    // add new info to session
    $this->persist();
}

В контроллере я беру объект продукта из базы данных с помощью ProductMapper и предоставляю его "Additem()":

    $product1 = $prodMapper->getProductByName('cap');
    $this->_cart->addItem($product1, 2);

getProductByName() возвращает новый заполненный объект Model_Product.


Обычно я получаю

Please ensure that the class definition "Model_Product" of the object you are trying to operate on was loaded _before_ ...

Сообщение об ошибке, дамп сеанса явно показывает

['__PHP_Incomplete_Class_Name'] => 'Model_Product'


Я знаю о "объявлении класса перед его сериализацией". Моя проблема заключается в следующем: как я могу объявить класс продукта в addItem(), если он введен (первый параметр) в первую очередь? Не будет ли новое объявление (например, new Model_Product()) перезаписывать параметр (исходный объект) в addItem()? Должен ли я снова объявить это в модели корзины?

Кроме того, я наверняка получу Cannot redeclare class Model_Product, если я... повторно объявлю это в корзине.

Author: nevvermind, 2011-05-05

1 answers

В начальной загрузке ZF сеанс был запущен перед автоматической загрузкой.

    /**
     * Make XXX_* classes available
     */
    protected function _initAutoloaders()
    {
        $loader = new Zend_Application_Module_Autoloader(array(
                    'namespace' => 'XXX',
                    'basePath' => APPLICATION_PATH
                ));
    }

    public function _initSession()
    {
        $config = $this->_config->custom->session;

        /**
         * For other settings, see the link below:
         * http://framework.zend.com/manual/en/zend.session.global_session_management.html
         */
        $sessionOptions = array(
            'name'             => $config->name,
            'gc_maxlifetime'   => $config->ttl,
            'use_only_cookies' => $config->onlyCookies,
//            'strict'           => true,
//            'path'             => '/',
        );

        // store session info in DB
        $sessDbConfig = array(
            'name'           => 'xxx_session',
            'primary'        => 'id',
            'modifiedColumn' => 'modified',
            'dataColumn'     => 'data',
            'lifetimeColumn' => 'lifetime'
        );

        Zend_Session::setOptions($sessionOptions);
        Zend_Session::setSaveHandler(new Zend_Session_SaveHandler_DbTable($sessDbConfig));
        Zend_Session::start();
    }

Когда я получал ошибки, о которых я говорил, объявление метода было наоборот: _initSession() было первым, затем _initAutoloaders() - и это был точный порядок, в котором ZF их обрабатывал.

Я проверю еще немного, но это, кажется, работает (и логично). Спасибо за все ваши предложения.

 4
Author: nevvermind, 2011-05-05 16:46:52