如果要扩展一个实现 Traversable 的内置类,是否可以使用 debug_backtrace 获取对根对象的引用?
例如,如果我有
$foo->bar->baz->biz->bon->bop->bob();
在 bop 的 bob 方法中我使用 debug_backtrace 有没有办法获取对 $foo 的引用? (那是什么方式?)
如果是这样,这是否是最优雅或最有效的方法?
我尝试查看 debug_backtrace php.net 页面,但我仍然不清楚如何真正使用此功能,但从其他材料和实验来看,我有疑问(请参阅更新)。
更新#1:
对于 debug_backtrace 是否应该留在生产代码中似乎确实存在一些争论。 生产代码中的PHP debug_backtrace 获取有关调用方法的信息?
当然,这部分是耦合问题,被调用的对象是否应该知道调用它的对象?
更新#2
在找到找出哪个类调用了另一个类中的方法后,我尝试使用我所学到的使用debug_backtrace进行反向遍历,并发现这是不可能的。
<?php
class mother extends SimpleXMLElement {
protected $test_a = 'foo';
public $test_b = 'bar';
public function shout(){
echo "test_a '" , $this->test_a , "' while test_b '" , $this->test_b , "'.\n";
}
public function tell_mother($message,$limit=42){
$trace = debug_backtrace();
--$limit;
if($limit<1){
echo "Enough of that pointlessness\n";
var_dump($trace);
return 0;
}
if ( isset($trace[1]) && isset($trace[1]['object']) ){
// $trace[1] is the caller class
echo "I will pass the message on\n";
$trace[1]['object']->tell_mother($message,$limit);
}else{
echo "I am a " , get_class($this) , "\n";
echo "I have been told \"{$message}\".\n";
var_dump($trace);
}
}
}
echo "<pre>";
$xml = <<<XML
<?xml version='1.0'?>
<a>lob
<b>tob
<c>bob
<d>boo
<e>bin
<f>baz
<g>bar
<h>foo</h>
</g>
</f>
</e>
</d>
</c>
</b>
</a>
XML;
$obj = simplexml_load_string($xml,'mother');
$obj->b->c->d->e->f->g->h->tell_mother("I love her");
$obj->shout();
$obj->b->c->d->e->f->shout();
据我所知,debug_backtrace 无法反向遍历,并且无法访问对象范围中保存的任何值。
上面的输出给出
I am a mother
I have been told "I love her".
array(1) {
[0]=>
array(7) {
["file"]=>
string(58) "/home/[[snip]]_test.php"
["line"]=>
int(64)
["function"]=>
string(11) "tell_mother"
["class"]=>
string(6) "mother"
["object"]=>
object(mother)#2 (1) {
[0]=>
string(3) "foo"
}
["type"]=>
string(2) "->"
["args"]=>
array(1) {
[0]=>
&string(10) "I love her"
}
}
}
test_a '' while test_b ''.
test_a '' while test_b ''.
我的结论是,PHP 在内部并不认为子元素是由父元素调用的。因此,我编辑了我的问题,只是询问是否可以反向遍历 Traversable 类。
通过问题中详细介绍的反复试验(主要是错误),我了解到至少有一些实现 Traversable 的 PHP 类只是看起来是类,但实际上是系统资源。
所以debug_backtrace(我的第一个想法)将无法反转遍历。这很可能是不可能的,但是只要有一点聪明才智,就可以创建一个委托包装器来提供实际对象的两种方式遍历。
此类类的示例可以在我对我提出的另一个问题的回答中找到:如何在 PHP 中获取对根对象的迭代对象引用?
此外,debug_backtrace 并不是在生产中执行除其设计目的的错误日志记录和调试之外的任何操作的最佳选择。
TD;DR:可能不是,但我们可以解决这个问题。