黄宽的blog

宁可十年不将军,不可一日不拱卒



对象的本质,延迟绑定

        php中的类和对象到底是怎样的一个关系?或者说对象到底由什么而组成?什么叫类的延迟绑定,怎么去解决?以前的认知算比较浅的,最近翻阅了一些文章和书籍,让我对类和对象的理解更深了一层。(可能是因为闲吧)。

        类是方法和属性的集合,那对象是什么?如果在以前问我,我会说对象是类的实例化,现在我会说对象是一堆属性组成。对象在底层到底是怎么实现的呢?对象在底层的实现是采取属性数组+方法数组来实现的。对象在zend中的定义是使用了一种zend_object_value结构体来存储的,这个结构体包含了三种玩意,第一种是一个指针,也就是说明这个对象由哪个类实现出来的,这个类在哪里。第二种玩意就是这个对象的属性。第三个东西是guards,阻止递归调用的。那对象里面只有这些东西,那对象的方法在哪里,对象的方法不会存在对象里面,要使用对象的方法,实际上是通过指针找到这个类,再用这个类里面的方法来执行的。要验证这种说法,可以通过序列化serialize一个对象,就可以进行观察了。前几天听见有人说数组的本质是对象,我想他搞错了,他认为一切都是对象这个没错,但是这只是可以以思维认为它是个对象,但本质是不会变的,我觉得他学岔道了。一切事物都可以抽象出来为一个类,当你为这个类赋予不同的属性时,就会生产对象,但是这仅仅只是可以将事物抽象出来,而不是本质。就如上面所说,对象是由一个类指针和属性数组加一个guards。那怎么证明类属性是个数组呢,其实也可以将一个数组serialize序列化一下,就能看出相同和不同了。

        延迟绑定是什么?其实理解以上说的对象到底是什么后,再来看延迟绑定,会理解得更深一些。设想一段场景,你和你老爸都有个习惯,就是吃完饭都得喝点东西来解渴,你老爸爱喝茶,你爱喝牛奶。下面通过一段小代码说明延迟绑定。

        class Father{

               public $type = 'father';

               public function eat(){

                       echo '饭吃完了';

                      echo self::drink();

                }

                public function drink(){

                      return '父亲想喝茶';

               }

        }

        class Son extends Father{//儿子不想喝茶,就想喝奶,于是重写了drink方法

                public $type = 'son';

                public function drink(){

                        return '儿子想喝奶';

                }

        }

        $son = new Son;

        $son->eat();

        到这里请问输出什么,按道理来讲是不是应该输出'饭吃完了儿子想喝奶啊',因为drink方法被重写了嘛,但实际上还是'父亲想喝茶',这是为什么呢,这个现象叫延迟绑定,解决办法很简单,就是不用self,改为static就行了,static在这里表示实际调用者(static关键字在不同的地方有不同的含义)。

        好了,我来试着探究一下为什么会出现这个问题,首先将子类对象和父类对象全部序列化,观察得到子类对象于父类对象的区别只在于指针和属性不同。当子类对象调用eat方法时,程序沿着指针找到了Son这个类寻找eat方法,但是并没有找到,发现这个类是一个子类,于是继续往上找,找到父类的时候找到了eat方法,调用,echo完吃饭后,此时程序还在父类中执行eat方法,下面的指令是调用本类的drink方法,然而现在指针还在父类,所以调用的还是父类的drink方法,而不是子类重写后的drink方法了。

  php

作者  :  黄宽

不耻最后,即使慢,驰而不息,纵会落后,纵令失败



About ME

about me

黄宽

我不想成为一个庸俗的人。十年百年后,当我们死去,质疑我们的人同样死去,后人看到的是裹足不前、原地打转的你,还是一直奔跑、走到远方的我?

友情链接