1

我已经看过几个关于 object destroy order 的答案,并且都指出不能保证顺序。由于我无法控制顺序,我想在所有对象都被销毁后调用一个函数。

register_shutdown_function 在对象销毁之前被调用,因此不是一个选项。我已经查看了使用对象的 set_error_handler 之类的技巧,因此它被“延迟”调用,但这还不够。

关于问题的一些背景,这是一个复杂的 CMS,其中包含数十个用于路由(视图)层的单独文件。有一个常见的引导包括,但不是一个常见的在关机时运行。我通过一个常见的继承基类使用 APCu 对象缓存,并且需要确保清除对象。对于在页面加载期间创建的同一对象的任何两个实例,一个可能想要清除自己,而另一个可能想要缓存自己。显然 purge 胜过所有其他事情,所以我需要在一组全局缓存键上调用 apc_delete 来清除所有 __destruct()'ion 完成。

4

2 回答 2

4

正如我在上面的评论中所说,多个对象实例的混合状态听起来像是一个设计缺陷。我认为您总是希望对象的所有实例都是最新的,或者如果不是,至少不要触及缓存。我认为您的缓存和/或对象加载层可以处理这个问题。


为了回答您的基本问题,我编写了一个脚本来测试 PHP 的关闭顺序

<?php  /* Adapted from https://evertpot.com/160/ */

    ob_start(
        function($buffer) {
            return $buffer . "output buffer flushed\n";
        }
    );

    $empty = function() {
        return true;
    };

    $close = function() {
        echo "session close\n";
        return true;
    };

    $write = function() {
        echo "session write\n";
        return true;
    };

    session_set_save_handler($empty, $close, $empty, $write, $empty, $empty);
    session_start();

    register_shutdown_function(
        function() {
            echo "register_shutdown_function\n";
        }
    );

    class MyClass {
        function __destruct() {
           echo "object destructor\n";
        }
    }
    $myObject = new MyClass;

输出:

register_shutdown_function
object destructor
output buffer flushed
session write
session close

似乎输出缓冲和会话写入/关闭的刷新是您知道所有对象已被破坏的两个地方。(这也假设您没有使用可以将自己注册为关闭函数的新形式的 session_set_save_handler。)

于 2017-01-20T16:38:14.247 回答
0

您是否尝试在函数中调用该__destruct()函数?PHP 在销毁所有对象后调用此函数,因此它可能会有所帮助。

于 2017-01-18T21:19:53.353 回答