第一章 面向对象思想的核心概念
代码:
class object
{
public $name;
public $gender;
public function say(){
echo $this->name."is".$this->gender;
}
}
$student = new object();
$student->name = 'Tom';
$student->gender = 'male';
$student->say();
//由于PHP 的对象是用数组来模拟的,因此把对象转换为数组,就能查看这个对象的所有属性了
var_dump($student);
var_dump((array)$student);//(array)对象类型转换成数组,打印属性值
var_dump(array($student));
print_r((array)$student);
echo '<br />';
//对象序列化
$str = serialize($student);
echo $str.'<br />';
file_put_contents('store.txt',$str);
//反序列化,取出这个对象
$str = file_get_contents('store.txt');
$student = unserialize($str);
$student->say();
结果:
Tomismale
object(object)[1]
public 'name' => string 'Tom' (length=3)
public 'gender' => string 'male' (length=4)
array (size=2)
'name' => string 'Tom' (length=3)
'gender' => string 'male' (length=4)
array (size=1)
0 =>
object(object)[1]
public 'name' => string 'Tom' (length=3)
public 'gender' => string 'male' (length=4)
Array ( [name] => Tom [gender] => male )
O:6:"object":2:{s:4:"name";s:3:"Tom";s:6:"gender";s:4:"male";}
Tomismale
1.3 继承与多态
面向对象的优势在于类的复用
。继承与多态都是对类进行复用,它们一个是类级别的复用
,一个是方法级别的复用
。提到继承必提组合,二者有何异同?PHP到底有没有多态?若没有,则为什么没有?有的话,和其他语言中
在继承中,用parent
指代父类,用self
指代自身。使用“::”
运算符(范围解析操作符)调用父类的方法。
通过一些总结,得出了继承是一种“是、像
”的关系,而组合是一种“需要
”的关系。利用这条规律,就可以很简单地判断出父亲与儿子应该是继承关系,父亲与家庭应该是组合关系。还可以从另外一个角度看,组合偏重整体与局部的关系,而继承偏重父与子的关系。
耦合是一个软件结构内不同模块之间互连程度的度量,也就是不同模块之间的依赖关系。
低耦合指模块与模块之间,尽可能地使模块间独立存在;模块与模块之间的接口尽量少而简单。现代的面向对象的思想不强调为真实世界建模,变得更加理性化一些,把目标放在解耦上。
解耦是要解除模块与模块之间的依赖。
按照这个思想,继承与组合二者语义上难于区分,在二者均可使用的情况下,更倾向于使用组合。为什么呢?继承存在什么问题呢?
1)继承破坏封装性。
比如,定义鸟类为父类,具有羽毛属性和飞翔方法,其子类天鹅、鸭子、鸵鸟等继承鸟这个类。显然,鸭子和鸵鸟不需要飞翔这个方法,但作为子类,它们却可以无区别地使用飞翔这个方法,显然破坏了类的封装性。而组合,从语义上来说,要优于继承。
2)继承是紧耦合的。
继承使得子类和父类捆绑在一起。组合仅通过唯一接口和外部进行通信,耦合度低于继承。
3)继承扩展复杂。
随着继承层数的增加和子类的增加,将涉及大量方法重写。使用组合,可以根据类型约束,实现动态组合,减少代码。
4)不恰当地使用继承可能违反现实世界中的逻辑。
1.3.2 各种语言中的多态
多态确切的含义是:同一类的对象收到相同消息时,会得到不同的结果。而这个消息是不可预测的。多态,顾名思义,就是多种状态,也就是多种结果。
区别是否是多态的关键在于看对象是否属于同一类型。如果把它们看做同一种类型,调用相同的函数,返回了不同的结果,那么它就是多态;否则,不能称其为多态。由此可见,弱类型的PHP里多态和传统强类型语言里的多态在实现和概念上是有一些区别的,而且弱类型语言实现起多态来会更简单,更灵活。
本节解决了什么是多态,什么不是多态的问题。至于多态是怎么实现的,各种语言的策略是不一样的。但是,最终的实现无非就是查表和判断。总结如下:
多态指同一类对象在运行时的具体化。
PHP语言是弱类型的,实现多态更简单、更灵活。
类型转换不是多态。
PHP中父类和子类看做“继父”和“继子”关系,它们存在继承关系,但不存在血缘关系。 因此子类无法向上转型为父类,从而失去多态最典型的特征。
多态的本质就是if…else,只不过实现的层级不同。
1.4 面向接口编程
狭义的接口,即interface关键字。广义的接口可以是任何一个对外提供服务的出口,比如提供数据传输的USB接口、淘宝网对其他网站开放的支付宝接口。
接口作为一种规范和契约存在。作为规范,接口应该保证可用性;作为契约,接口应该保证可控性。
接口只是一个声明,一旦使用interface关键字,就应该实现它。可以由程序员实现(外部接口),也可以由系统实现(内部接口)。接口本身什么都不做,但是它可以告诉我们它能做什么。
PHP中的接口存在两个不足,一是没有契约限制,二是缺少足够多的内部接口。
接口其实很简单,但是接口的各种应用很灵活,设计模式中也有很大一部分是围绕接口展开的。
1.5 反射
面向对象编程中对象被赋予了自省的能力,而这个自省的过程就是反射。
反射,直观理解就是根据到达地找到出发地和来源。比方说,我给你一个光秃秃的对象,我可以仅仅通过这个对象就能知道它所属的类、拥有哪些方法。
反射指在PHP运行状态中,扩展分析PHP程序,导出或提取出关于类、方法、属性、参数等的详细信息,包括注释。这种动态获取信息以及动态调用对象方法的功能称为反射API。
小结
本章主要介绍面向对象思想的程序的组成元素——类和对象。类是一个动作和属性的模板,对象是数据的集合。结合PHP自身实际情况,着重讲述PHP里面向对象的一些比较模糊的知识点,包括魔术方法、接口、多态、类的复用、反射、异常机制等。接口是一种类型,从接口的实现讲述接口是怎么实现“即插即用”的。
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。