ホームページ >バックエンド開発 >PHPチュートリアル >php_PHP チュートリアルの __destruct と register_shutdown_function の実行順序

php_PHP チュートリアルの __destruct と register_shutdown_function の実行順序

WBOY
WBOYオリジナル
2016-07-13 10:16:46810ブラウズ

php

の__destructとregister_shutdown_functionの実行順序は、phpマニュアルの分析に基づいています。

__destruct は

デストラクターは、オブジェクトへのすべての参照が削除されるか、オブジェクトが明示的に破棄されるときに実行されます。

そして register_shutdown_function は

スクリプトの実行が終了するか exit() が呼び出された後に実行されるコールバックを登録します。

文字通り理解すると、__destruct はオブジェクト レベルですが、register_shutdown_function はスクリプト全体のレベルである必要があり、登録された関数も最後に実行される必要があります。私たちの推測を確認するために、スクリプトを書きます:

コードは次のとおりです:


register_shutdown_function(function(){echo 'global';});
クラスA{
パブリック関数 __construct(){
}
パブリック関数 __destruct()
{
echo __class__,'::',__function__,'
';
}
}
新しいA;

実行結果:

コードは次のとおりです:


A::__ を破壊する
グローバル

私たちの推測を完全に裏付けました。オブジェクト -> スクリプトの順序で実行されました。

しかし、オブジェクトに register_shutdown_function を登録したらどうなるでしょうか?まだ同じ順番ですか? !

コードは次のとおりです:


クラスA{
パブリック関数 __construct(){
Register_shutdown_function(function(){echo 'local', '
';});
}
パブリック関数 __destruct()
{
echo __class__,'::',__function__,'
';
}
}
新しいA;

結果:

コードをコピーします。コードは次のとおりです:


地元
A::__破壊

register_shutdown_function が最初に呼び出され、最後に実行オブジェクトの __destruct が呼び出されていることがわかります。これは register_shutdown_function で登録した関数がクラス内のメソッドとして扱われることを示しています。 !わかりませんが、これを解析するには PHP ソース コードを表示する必要があるかもしれません。

範囲を拡大して状況を確認できます:

コードは次のとおりです:


register_shutdown_function(function(){echo 'global', '
';});
    クラスA{
        パブリック関数 __construct(){
            register_shutdown_function(array($this, 'op'));
        }
        パブリック関数 __destruct()
        {
            echo __class__,'::',__function__,'
';
        }
        パブリック関数 op()
        {
            echo __class__,'::',__function__,'
';
        }
    }
    クラスB{
        パブリック関数 __construct()
        {
            register_shutdown_function(array($this, 'op'));
            $obj = 新しい A;
        }
        パブリック関数 __destruct()
        {
            echo __class__,'::',__function__,'
';
        }
        パブリック関数 op()
        {
            echo __class__,'::',__function__,'
';
        }
    }
    $b = 新しい B;

私は、register_shutdown_function 関数の 1 つを全局で登録し、クラス AB でさらに 1 つを登録し、クラス内でそれぞれ別のメソッドを実行しました。

代码如下:

グローバル
B::おっ
あ::おっと
A::__ を破壊する
B::__破壊

結果は私たちの想像を完全に覆し、register_shutdown_function関数はクラス内での登録は完全な登録ではなく、すべてが先に実行され、クラス内で実行される順序は、それらが登録された後の順序になります。シャットダウン機能関数は前も後もこの結果であり、同様に結果があり、register_shutdown_function より __destruct が実行され、全体の register_shutdown_function 関数がさらにクラス内の注釈の register_shutdown_function が実行されます。

そして遅い、私はこの結果を受け入れることができません、このような結果を照会して、道说が完了した後、__destructを再実行してもいいですか? shutdown_function,したがって全局register_shutdown_functionを保持:

代码如下:

クラスA{

        パブリック関数 __destruct()
        {
            echo __class__,'::',__function__,'
';
        }
    }
    クラスB{
        パブリック関数 __construct()
        {
            $obj = 新しい A;
        }
        パブリック関数 __destruct()
        {
            echo __class__,'::',__function__,'
';
        }
    }
    register_shutdown_function(function(){echo 'global', '
';});


出所:

代码如下:

A::__ を破壊する

グローバル
B::__破壊


結果、リーダーは確かに、A、B の 2 つの種類の分析関数関数の実行順序は疑う余地はありません、B が承認したため、A は B 先销毁を肯定しましたが、全体のローカルの register_shutdown_function関数また怎么夹が実行されました?!费解。

手動の解析に従って、構造関数の解析は、終了時に実行することもできます。

exit() を使用してスクリプトが終了した場合でも、デストラクターが呼び出されます。デストラクターで exit() を呼び出すと、残りのシャットダウン操作が中止されます。

exit が関数内で呼び出される場合、どのように呼び出されますか?

コードは次のとおりです:


クラスA{
パブリック関数 __construct(){
register_shutdown_function(array($this, 'op'));
終了します;
}
パブリック関数 __destruct()
{
echo __class__,'::',__function__,'
';
}
パブリック関数 op()
{
echo __class__,'::',__function__,'
';
}
}
クラスB{
パブリック関数 __construct()
{
register_shutdown_function(array($this, 'op'));
$obj = 新しい A;
}
パブリック関数 __destruct()
{
echo __class__,'::',__function__,'
';
}
パブリック関数 op()
{
echo __class__,'::',__function__,'
';
}
}
Register_shutdown_function(function(){echo 'global', '
';});
$b = 新しい B;

出力:

コードは次のとおりです:


グローバル
B::おっ
あ::おっと
B::__ を破壊する
A::__破壊

このシーケンスは上記の 3 番目の例と似ていますが、異なる点と驚くべき点は、クラス B のデストラクターがクラス A の前に実行されることです。クラス A へのすべての参照は B が破棄された後にのみ破棄されるというのは本当ですか? !未知。

結論:
1. register_shutdown_function と __destruct をスクリプト内で混合しないようにしてください。これらの動作はまったく予測できません。
1. オブジェクトは相互に参照するため、コンテンツを順番に出力する必要がある場合、コンテンツをデストラクター __destruct に配置すべきではありません。 2. register_shutdown_function の順序を予測するのは難しく (関数はこのオブジェクトが呼び出されたときにのみ登録されます)、__destruct は register_shutdown_function を完全に置き換えることができるため、クラスには register_shutdown_function を登録しないようにしてください。 3. スクリプトの終了時に関連するアクションを実行する必要がある場合は、スクリプトの先頭で register_shutdown_function を登録し、すべてのアクションを関数に入れるのが最善です。

http://www.bkjia.com/PHPjc/896773.html

tru​​ehttp://www.bkjia.com/PHPjc/896773.html技術記事 php の __destruct と register_shutdown_function の実行順序は、php マニュアルの分析に基づいています。 __destruct は、オブジェクトへのすべての参照が削除されたとき、または...
ときに発生するデストラクターです。
声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。