2

私は、グローバル変数が複数の場所でさまざまなファイルで使用されているプロジェクトに取り組んでいますが、グローバル変数の値が期待したものではないという状況に遭遇しました。存在する場合、グローバル変数の変更の履歴を取得できる関数について知っておくと便利です。

debug_backtrace()この関数を使用すると、渡された変数の名前を引数として取り、行番号とファイル名に関する情報を含む結果を返すことを除いて、使用時に取得するのと同じ種類の出力を取得できると期待しています。グローバル変数に変更が加えられた場所。

そのようなものが今のところ存在しない場合は、取り組むのに良いハックになるでしょう。前もって感謝します。

4

1 に答える 1

3

すでに概説したように、これは簡単には実行できません。xdebugを選択して、リモート/ステップデバッガーとして使用することをお勧めします。そうは言って、少なくともどういうわけか自分で追跡することはできますが、それには代償が伴います。

最初に出力(ここではCLIスクリプト)から始めて例を示します。

GLOBAL $argv[0] changed to: /path/to/global.php
GLOBAL $argc changed to: 1
GLOBAL $test changed to: 1
GLOBAL $test changed to: 2
GLOBAL $test changed to: 3 

これは次の人によって生成されました:

<?php
/**
 * @link http://stackoverflow.com/questions/12895358/how-to-check-history-of-a-global-variable-in-php
 */

declare(ticks=1);

function array_diff_recursive(array $array1, array $array2) {

    $diff = array();

    foreach ($array1 as $key => $value) {
        if (array_key_exists($key, $array2)) {
            if (is_array($value) && is_array($array2[$key])) {
                if ($result = array_diff_recursive($value, $array2[$key])) {
                    $diff[$key] = $result;
                }
            } else {
                if ($value !== $array2[$key]) {
                    $diff[$key] = $value;
                }
            }
        } else {
            $diff[$key] = $value;
        }
    }

    return $diff;
}

function globals_diff() {
    static $last = array();

    if (!isset($GLOBALS)) {
        return;
    }

    $current = array();

    $ignore = array_flip(array('_GET', '_POST', '_COOKIE', '_FILES', '_ENV', '_REQUEST', '_SERVER', 'GLOBALS'));

    foreach($GLOBALS as $key => $value) {
        isset($ignore[$key]) || $current[$key] = $value;
    }

    $diff = array_diff_recursive($current, $last);

    foreach($diff as $var => $value) {
        if (is_array($value) && $value) {
            foreach($value as $array_key => $array_value) {
                printf("GLOBAL $%s[%s] changed to: %s\n", $var, $array_key, print_r($array_value, 1));
            }
        } else {
            printf("GLOBAL $%s changed to: %s\n", $var, print_r($value, 1));
        }
    }

    $last = $current;
}

register_tick_function('globals_diff', true);

$test = 1;
$test = 2;
$test = 3;

これには、ティックを有効にする必要があります。それ以外の場合は、グローバル関数を使用してスナップショットを作成できます。

時間の経過に伴う変更を追跡する場合は、他の(グローバル)データ構造を追加して、比較から除外しながら、これらの変更を保存する必要があります(注意、これは大きくなる可能性があります)(ingoreリストを参照)。

于 2012-10-15T12:48:04.180 に答える