Why can't a PDO object be serialized?

前端 未结 2 1362
-上瘾入骨i
-上瘾入骨i 2021-01-14 02:17

I am making a multi-threaded CLI-PHP application and need to serialize PDO object to pass it between work inside the thread, and wake it up from a sleeping thre

相关标签:
2条回答
  • 2021-01-14 02:49

    A PDO object contains state that cannot be represented in the serialization format. For example, the PDO object contains an open connection to a database server.

    If you were to try to deserialize a serialized PDO object, the __wakeup() method would have to reconnect to the database server. This would require that authentication credentials be stored in a readable manner in the serialized PDO object, which is a security no-no.

    I worked on the Zend Framework's Zend_Db component a long time ago, and I deliberately designed the Zend_Db_Adapter object to not be serializable for this reason. Instances of Zend_Db_Table, Zend_Db_Table_Row, etc. could be serializable, but could not be "live" after deserialization until you assigned it a freshly connected Zend_Db_Adapter instance.

    Also, there's no guarantee that the database server would be reachable at the time you deserialize the PDO object. It's unclear whether this means the deserialization would be considered "failed."

    The same restrictions on serialization apply to other resources such as sockets or file handles.

    See also Why isn't every type of object serializable?

    0 讨论(0)
  • 2021-01-14 02:49

    What's being done at http://php.net/manual/en/language.oop5.magic.php is creating a wrapper that can be serialized since the PDO link itself cannot be.

    <?php
    class Connection
    {
        protected $link;
        private $dsn, $username, $password;
    
        public function __construct($dsn, $username, $password)
        {
            $this->dsn = $dsn;
            $this->username = $username;
            $this->password = $password;
            $this->connect();
        }
    
        private function connect()
        {
            $this->link = new PDO($this->dsn, $this->username, $this->password);
        }
    
        public function __sleep()
        {
            return array('dsn', 'username', 'password');
        }
    
        public function __wakeup()
        {
            $this->connect();
        }
    }?>
    

    The PDO object apparently does not keep the dsn, user, pwd after connecting, and so cannot be directly serialized. But if you created a wrapper like in the example above, where you stored this information, you could serialize the wrapper. Then when you unserialize, it will create a new PDO object and reconnect by passing the credentials in from the wrapper to PDO.

    0 讨论(0)
提交回复
热议问题