如何在PHP中检查全局变量的历史记录?

时间:2012-10-15 12:08:47

标签: php debugging global-variables

我正在开发一个项目,其中全局变量用于多个地方和各种文件,并且遇到全局变量的值不是我所期望的情况。很高兴知道一个函数,如果存在,允许我获取全局变量的变化历史。

我希望该函数允许我获得与使用debug_backtrace()时相同的输出,除了它将传递的变量的名称作为参数并将返回包含信息的结果关于对全局变量进行了更改的文件的行号和名称。

如果现在不存在这样的事情,那将是一个很好的黑客。提前谢谢。

1 个答案:

答案 0 :(得分: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;

这需要您拥有ticks enabled。否则,您可以使用全局功能进行快照。

如果你想跟踪一段时间内的变化,你需要添加一些其他(全局)数据结构来存储这些变化(注意,这可能会变大)同时保持它不在比较之内(参见ingore列表)。