1.4 对象和函数
我们现在已经创建了一些类以代表各种各样的快递公司,也知道了如何从类定义中实例化对象。现在我们来了解如何标识对象并将其传送到对象方法里面。
首先,需要一个目标对象,让我们先创建一个Parcel类:
这是一个非常简单的类,正如所料,包裹本身就是相对单调乏味的!
1.4.1 类型提示
修改ship()方法以便只接受此种参数,它是将对象名放在参数之前的Parcel对象中:
这叫做类型提示(type hinting),其中可以指定哪种参数适合于这种方法—对于函数也是一样。可以类型提示对象名和数组。自从PHP放宽了数据类型之后(它是一个动态和弱类型的语言),对于字符串和数值这样的简单类型就不再使用类型提示了。
使用类型提示后,可以确定传入到函数内的对象种类,而且可以假定代码中会用到哪些属性和方法,以及会得到哪些结果。
1.4.2 多态性
设想一下,我们允许一个用户在其首选供应商名单上添加快递公司,按照这种思路,可以写出如下代码:
这看上去正常运行,但是如果我们需要保存一个PigeonPost对象呢?
事实上,如果传入一个PigeonPost对象到这个函数里,PHP识别出这是Courier对象的一个子类,因此该函数会接受它。这使我们可以将父类对象作为子类、孙子类的类型提示,甚至对象更遥远的后代都可以传入该函数。
这种同时识别出PigeonPost对象和Courier对象的能力称为多态性(polymorphism),它的字面意思就是“多种形式”。PigeonPost对象会同时识别出自己的类以及从哪个类中继承而来,而且不仅仅在类型提示时会识别。下面的示例使用instanceOf操作符来检查对象的类型:
正因为这样,当使用类型提示的时候,PigeonPost对象可以声明为PigeonPost和Courier两个对象。如果不行,还可以声明为Parcel。
1.4.3 对象和引用
当使用对象时,需要警惕在这个问题上犯错,那就是对象和简单变量类型表现大相径庭。很多数据类型都可以写时复制(copy-on-write),即当写代码$a=$b时,两个变量因赋予同样的值而告终。
然而对于对象而言,这就完全不同了,从下面的代码中你期望得到什么呢?
先动动脑筋想一下。
事实上,输出的结果是:
我们可以更进一步区分它们是否引用同一个原始对象,可用同样的方式使用= = =操作符进行比较:
当两个变量指向相同的值时,= = =比较操作符才会返回true。如果对象是完全相同的,但存储在不同的位置,此操作将返回false。这对于我们识别某个对象是否链接到另一个对象有很大的帮助。
1.4.4 作为函数参数传递的对象
从中断的地方继续关于引用的话题,我们必须牢记,对象总是通过引用传递。即当你传递一个对象到一个函数中,这个函数会作用于相同的对象,如果这个对象在函数内部发生改变,这种变化会反映到函数外部。这是将一个对象赋值给一个新变量的行为延伸。
对象总是以这样的方式表现,即它们提供一个对原始对象的引用,而不是创建自己的一个副本,这可能会导致意外的结果!
来看看下面的示例代码:
使用clone关键字会从同一个类中重新创建一个对象,这个对象和原始对象一样具有所有相同的属性。这两个对象之间没有链接,你可以安全地改变其中一个或另一个以使它们隔离开。
浅谈对象副本
当复制一个对象时,存储在其属性中的任何对象都将是引用而不是副本。因此,在处理复杂的面向对象的应用程序时必须非常小心。
PHP有一个神奇的方法,即如果声明了一个对象,当复制这个对象时,会调用这个对象,这就是_clone()方法,你可以声明而且以此来决定当复制对象时会做些什么,甚至不接受复制。
1.4.5 流畅的接口
我们知道,对象总是通过引用传递,这表明无需从一个方法中返回一个对象来观察它的变化。然而,如果从一个方法中返回$this,可以在应用程序内建立一个流畅的接口(fluent interface),可让你将方法链接在一起。其工作原理如下:
1)创建对象;
2)调用对象的方法;
3)得到从方法中返回的修正对象;
4)选择返回步骤2)。
下面是一个表述得更清楚的示例,在这里使用了一个Parcle类:
这里的关键是可以在一行代码中调用多个方法(可以加一些换行符以增加代码的可读性),并可按任意顺序调用。由于每个方法都返回生成的对象,因此可以通过返回对象再调用下一个方法。在很多设置中你可以看到这种模式,在适当的时候,你也可以使用这种模式来构建自己的应用程序。