PHP头条
热点:

Problems with objects in PHP 4 PHP4中使用对象的问题

1    class MyFoo { 
2        function MyFoo() 
3        { 
4            $this->me = &$this; 
5            $this->value = 5; 
6        } 
7 
8        function setValue($val) 
9        { 
10            $this->value = $val; 
11        } 
12 
13        function getValue() 
14        { 
15            return $this->value; 
16        } 
17 
18        function getValueFromMe() 
19        { 
20            return $this->me->value; 
21        } 
22    } 
23 
24        function CreateObject($class_type) 
25        { 
26            switch ($class_type) { 
27                case "foo": 
28                    $obj = new MyFoo(); 
29                    break; 
30                case "bar": 
31                    $obj = new MyBar(); 
32                    break; 
33            } 
34            return $obj; 
35        } 
36 
37        $global_obj = CreateObject ("foo"); 
38        $global_obj->setValue(7); 
39 
40        print "Value is " . $global_obj->getValue() . "\n"; 
41        print "Value is " . $global_obj->getValueFromMe() . "\n"; 

让我们一步步来讨论. 首先,有一个MyFoo类.在构造函数里,我们给$this->me一个引用,并设定

我们有其它三个成员函数: 一个设定this->value的值;一个返回this->value的值;另一个返回this->value->me的值. 但是--$this不是相同的东西吗? MyFoo::getValue()和MyFoo::getValueFromMe()返回的值不是一样的吗?

首先,我们调用CreateObject("foo"),这会返回一个MyFoo类型的对象. 然后我们调用MyFoo::setValue(7). 最后,我们调用MyFoo::getValue() 和MyFoo::getValueFromMe(), 期望得到返回值7.当然,如果我们在任何情况下都得到7, 以上这个例子将不是本书中最没有意义的例子. 所以我相信你已经猜到?我们得不到两个7这样的结果.

但是我们将得到什么结果,并且更重要地,为什么呢?

我们将得到的结果分别是7和5. 至于为什么?--有三个很好的理由.

首先,看构造函数. 当在构造函数内部,我们在this和this->me间建立引用. 换句话说,this和this->me是同个东西. 但是我们是在构造函数内. 当构造函数结束,PHP要重新建立对象(new MyFoo的结果,第28行)分配给$obj. 因为对象没有特殊化对待,就像其它任何数据类型一样,赋值X给Y意味着Y是X的一个副本. 也就是说,obj将是new MyFoo的一个副本,而new MyFoo是一个存在于构造函数的对象. Obj->me怎么样呢? 因为它是一个引用,它原封不动仍然指向原来的对象?this. Voila-obj和obj->me不再是同个东西了?改变其中一个另一个不变.

以上是第一条理由. 还有其它类似于第一条的理由. 奇迹般地我们打算克服实例化对象这个问题(第28行). 一旦我们把CreateObject返回的值赋给global_object,我们仍然要撞上相同的问题?global_object将变成返回值的一个副本,并且再次地,global_object和global_object->me将不再相同. 这就是第二条理由.

但是,事实上我们还走不了那么远? 一旦CreateObject返回$obj,我们将破坏引用(第34行) . 这就是第三条理由.

那么,我们如何改正这些? 有两个选择. 一是在所有地方增加&符号,就像例图5那样(第24, 28, 31, 37行). 二.如果你幸运地使用上了PHP5,你可以忘了以上这一切,PHP5会自动为你考虑这些. 如果你想知道PHP5是如何考虑这些问题的,继续阅读下去.

WTMA syndrome in PHP 4 PHP4中的WTMA综合症

1    class MyFoo { 
2        function MyFoo() 
3        { 
4            $this->me = &$this; 
5            $this->value = 2; 
6        } 
7 
8        function setValue($val) 
9        { 
10            $this->value = $val; 
11        } 
12 
13        function getValue() 
14        { 
15            return $this->value; 
16        } 
17 
18        function getValueFromMe() 
19        { 
20            return $this->me->value; 
21        } 
22    }; 
23 
24        function &CreateObject($class_type) 
25        { 
26            switch ($class_type) { 
27                case "foo": 
28                    $obj =& new MyFoo(); 
29                    break; 
30                case "bar": 
31                    $obj =& new MyBar(); 
32                    break; 
33            } 
34            return $obj; 
35        } 
36 
37        $global_obj =& CreateObject ("foo"); 
38        $global_obj->setValue(7); 
39 
40        print "Value is " . $global_obj->getValue() . "\n"; 
41        print "Value is " . $global_obj->getValueFromMe() . "\n"; 

图5

PHP5是第一个把对象看成与其它类型数据不同的PHP版本. 从用户的角度看,这证明它非常明白的方式?在PHP5中,对象总是通过引用来传递,而其它类型数据(如integer,string,array)都是通过值来传递. 最显著地,没有必要再用&符号来表示通过引用来传递对象了.

面向对象编程广泛利用了对象网络和对象间的复杂关系,这些都需要用到引用. 在PHP的前些版本中,需要显示地指明引用. 因此, 现在默认用引用来移动对象,并且只有在明确要求复制时才复制对象,这样比以前更好.

它是如何实现的呢?

在PHP5之前,所有值都存在一个名为zval(Zend Value)的特殊结构里. 这些值可以存入简单的值,如数字和字符串,或复杂的值如数组和对象. 当值传给函数或从函数返回时,这些值会被复制,在内存的另一个地址建立一个带有相同内容的结构.

在PHP5中,值仍存为zval结构中,但对象除外. 对象存在一个叫做Object Store的结构里,并且每个对象有一个不同的ID. Zval中,不储存对象本身,而是存着对象的指针. 当复制一个持有对象的zval结构,例如我们把一个对象当成参数传给某个函数,我们不再复制任何数据. 我们仅仅保持相同的对象指针并由另一个zval通知现在这个特定的对象指向的Object Store. 因为对象本身位于Object Store,我们对它所作的任何改变将影响到所有持有该对象指针的zval结构.这种附加的间接作用使PHP对象看起来就像总是通过引用来传递,用透明和有效率的方式.

使用PHP5,我们现在可以回到示例图4,除去所有的&符号, 一切代码都仍然可以正常工作.当我们在构造函数(第4行)中持有一个引用时一个&符号都不用. 

  1. 对象的串行化
  2. PHP面向对象编程入门


www.phpzy.comtrue/php/18717.htmlTechArticleProblems with objects in PHP 4 PHP4中使用对象的问题 1 class MyFoo { 2 function MyFoo() 3 { 4 $this-me = $this; 5 $this-value = 5; 6 } 7 8 function setValue($val) 9 { 10 $this-value = $val; 11...

相关文章

PHP之友评论

今天推荐