62

所以我不确定我要向你们展示什么,如果你需要更多代码,请不要犹豫,问:

所以这个方法将在我们的应用程序中为 Zend 设置 initMailer:

protected function _initMailer()
{
    if ('testing' !==  APPLICATION_ENV) {
        $this->bootstrap('Config');
        $options = $this->getOptions();
        $mail = new Zend_Application_Resource_Mail($options['mail']);
    }elseif ('testing'  ===  APPLICATION_ENV) {
        //change the mail transport only if dev or test
        if (APPLICATION_ENV <> 'production') {

            $callback = function()
            {
                return 'ZendMail_' . microtime(true) .'.tmp';
            };

            $mail = new Zend_Mail_Transport_File(
                array('path' => '/tmp/mail/',
                        'callback'=>$callback
                )
            );

            Zend_Mail::setDefaultTransport($mail);
        }
    }


    return $mail;
}

您可以看到其中的闭包。当我运行任何使用此代码的测试时,我得到:

Exception: Serialization of 'Closure' is not allowed 

因此与此“关闭”有关的所有测试都失败了。所以我在这里问你们我应该怎么做。

为了澄清上述内容,我们所做的只是说我们发送的任何电子邮件都希望将有关该电子邮件的信息存储在文件中 /tmp/mail/ 目录中的文件夹中。

4

5 回答 5

42

显然匿名函数不能被序列化。

例子

$function = function () {
    return "ABC";
};
serialize($function); // would throw error

从您的代码中,您正在使用 Closure:

$callback = function () // <---------------------- Issue
{
    return 'ZendMail_' . microtime(true) . '.tmp';
};

解决方案1:替换为正常功能

例子

function emailCallback() {
    return 'ZendMail_' . microtime(true) . '.tmp';
}
$callback = "emailCallback" ;

解决方案 2:通过数组变量间接调用方法

如果您查看 http://docs.mnkras.com/libraries_23rdparty_2_zend_2_mail_2_transport_2file_8php_source.html

   public function __construct($options = null)
   63     {
   64         if ($options instanceof Zend_Config) {
   65             $options = $options->toArray();
   66         } elseif (!is_array($options)) {
   67             $options = array();
   68         }
   69 
   70         // Making sure we have some defaults to work with
   71         if (!isset($options['path'])) {
   72             $options['path'] = sys_get_temp_dir();
   73         }
   74         if (!isset($options['callback'])) {
   75             $options['callback'] = array($this, 'defaultCallback'); <- here
   76         }
   77 
   78         $this->setOptions($options);
   79     }

您可以使用相同的方法发送回调

$callback = array($this,"aMethodInYourClass");
于 2012-12-05T23:21:53.813 回答
27

PHP 不允许直接闭包序列化。但是你可以使用像 PHP Super Closure 这样的强大类:https ://github.com/jeremeamia/super_closure

这个类使用起来非常简单,并且捆绑到队列管理器的 laravel 框架中。

来自 github 文档:

$helloWorld = new SerializableClosure(function ($name = 'World') use ($greeting) {
    echo "{$greeting}, {$name}!\n";
});

$serialized = serialize($helloWorld);
于 2013-11-01T15:47:22.697 回答
12

如前所述:开箱即用的闭包不能序列化。

然而,使用__sleep()魔法__wakeup()方法和反射你可以手动使闭包可序列化。有关更多详细信息,请参阅extended-php-5-3-closures-with-serialization-and-reflection

这利用了反射和 php 函数eval。请注意,这会打开代码注入的可能性,因此请注意您正在序列化的内容。

于 2013-02-11T10:09:43.333 回答
2

您必须禁用全局变量

 /**
 * @backupGlobals disabled
 */
于 2017-09-01T09:38:36.893 回答
-1

它是实验性的,不安全,它有几个风险,比如必须使用可以禁用的 eval ()。最好的办法是在 heredoc 示例中编写要序列化的脚本:

$code = <<<CODE
 <?php
class \$gen_class_{$user}{
 ...
}
CODE;

您可以通过序列化代码或使用扩展名“.class.php”编写代码来更轻松地使用代码。您将能够轻松调用您的脚本并使其持久化。

https://3v4l.org/jpHm9 更新 PHP 8

<?php

function closure_to_str($func)
{
    $refl = new \ReflectionFunction($func); // get reflection object
    $path = $refl->getFileName();  // absolute path of php file
    $begn = $refl->getStartLine(); // have to `-1` for array index
    $endn = $refl->getEndLine();
    $dlim = PHP_EOL;
    $list = explode($dlim, file_get_contents($path));         // lines of php-file source
    $list = array_slice($list, ($begn-1), ($endn-($begn-1))); // lines of closure definition
    $last = (count($list)-1); // last line number

    if((substr_count($list[0],'function')>1)|| (substr_count($list[0],'{')>1) || (substr_count($list[$last],'}')>1))
    { throw new \Exception("Too complex context definition in: `$path`. Check lines: $begn & $endn."); }

    $list[0] = ('function'.explode('function',$list[0])[1]);
    $list[$last] = (explode('}',$list[$last])[0].'}');


    return implode($dlim,$list);
}

$test = 10;
$dog2 = function($var=0) use ($test){ 
    $var = 10;
    echo $var . PHP_EOL;
    return $test . $var;
};

echo closure_to_str($dog2)."\n\n";

返回字符串

function($var=0) use ($test){ 
    $var = 10;
    echo $var . PHP_EOL;
    return $test . $var;
}
于 2021-11-11T21:28:32.777 回答