第十八章 面向对象的特性

学习要点:

1.OOP 的封装

2.OOP 的继承

3.OOP 的多态

面向对象的三个主要特性是封装、继承和多态。

一.OOP的封装

隐藏对象的字段和实现细节,仅对外公开接口,控制在程序中字段的读和修改的访问级

别;将抽象得到的数据和行为(或功能)相结合,形成一个有机的整体,也就是将数据与

操作数据的源代码进行有机的结合,形成“类”,其中数据和函数都是类的成员。

字段的作用域

1.public 公共的(类外可以访问)

2.private 私有的(类内可以访问)

3.protected 受保护的(类内和子类可以访问,类外不可访问)

创建使用了私有的字段,这样外部就无法访问了

  1. class Computer { //类的字段(成员)
  2. private $_name = '联想120'; private $_model = 'LX';
  3. }

通过一个公共方法作为入口,访问私有字段,而必须使用$this关键字。

  1. class Computer { //类的字段(成员)
  2. private $_name = '联想120'; private $_model = 'LX'; //通过公共方法来访问私有字段
  3. function run() { echo $this->_name;
  4. }
  5. } $computer->run ();

属性操作(私有字段的赋值与取值)

可以设计两个公共方法,一个方法为setName(),用于赋值;一个方法为getName(),

用于取值。

  1. class Computer { //类的字段(成员)
  2. private $_name; private $_model; //赋值
  3. function setName($_name) { $this->_name = $_name;
  4. } //取值
  5. function getName() { return $this->_name;
  6. }
  7. } $computer = new Computer (); $computer->setName ( 'IBM' ); echo $computer->getName ();

如果有十个字段那么就必须要二十个方法才能够赋值和取值,那么有没有更简便的方法

呢?PHP内置两个方法(拦截器)专门用于取值与赋值:set(),get()。

  1. class Computer { //类的字段(成员)
  2. private $_name; private $_model; //所有字段的赋值都在这里进行
  3. function __set($_key, $_value) { $this->$_key = $_value;
  4. } //所有字段的取值都在这里进行
  5. function __get($_key) { return $this->$_key;
  6. }
  7. } $computer = new Computer (); $computer->_model = 'LX'; echo $computer->_model;

方法私有:有些使用类里面的方法并不需要对外公开,只是里面运作的一部分,这个时

候可以将方法也封装起来。

  1. class Computer { //类的字段(成员)
  2. private $_name; private $_model; //私有方法
  3. private function getEcho() { echo '我是私有化的方法';
  4. } //公共方法一般是对外的入口
  5. public function run() { $this->getEcho ();
  6. }
  7. } $computer = new Computer (); $computer->run ();

建议:方法前面如果没有修饰符,那么就是外部可访问的公共方法,但为了让程序更加

的清晰,建议在前面加上public。

常量(constant)

在类中可以定义常量,用来表示不会改变的值。对于从该类实例化的任何对象来说,常

量值在这些对象的整个生命周期中都保持不变。

  1. class Computer { const PI = 3.1415926;
  2. } echo Computer::PI;

静态类成员

有时候,可能需要创建供所有类实例共享的字段和方法,这些字段和方法与所有的类实

例有关,但不能由任何特定对象调用。

  1. class Computer { public static $_count = 0;
  2. } echo Computer::$_count;

一般来说,必须将字段做成私有化。所以可能需要这么做:

  1. class Computer { private static $_count = 0; public static function setRun() {
  2. self::$_count ++;
  3. } public static function getRun() { return self::$_count;
  4. }
  5. }
  6. Computer::setRun (); echo Computer::getRun ();

Instanceof关键字

PHP5有一个instanceof关键字,使用这个关键字可以确定一个对象是类的实例、类的

子类,还是实现了某个特定接口,并进行相应的操作。

  1. class Computer {
  2. } $computer = new Computer (); echo ($computer instanceof Computer);

二.OOP继承

继承是从一个基类得到一个或多个类的机制。

继承自另一个类的类被称为该类的子类。这种关系通常用父类和孩子来比喻。子类将继

承父类的特性。这些特性由属性和方法组成。子类可以增加父类之外的新功能,因此子类也

被称为父类的“扩展”。

在PHP中,类继承通过extends关键字实现。继承自其他类的类成为子类或派生类,子

类所继承的类成为父类或基类。(PHP只支持单继承,PHP不支持方法重载)。

  1. class Computer { private $_name = '联想120'; private function __get($_key) { return $this->$_key;
  2. } public function run() { echo '我是父类';
  3. }
  4. } class NoteBookComputer extends Computer {
  5. } $notebookcomputer = new NoteBookComputer (); $notebookcomputer->run (); echo $notebookcomputer->_name;

字段和方法的重写(覆盖)

有些时候,并不是特别需要父类的字段和方法,那么可以通过子类的重写来修改父类的

字段和方法。

  1. class Computer { public $_name = '联想120'; protected function run() { echo '我是父类';
  2. }
  3. } class NoteBookComputer extends Computer { public $_name = 'IBM'; public function run() { echo '我是子类';
  4. }
  5. }

子类调用父类的字段或方法

为了安全,我们一般将父类的方法封装了起来,这样,外部就无法调用,只能被继承它

的子类所看到。这个时候,就需要通过子类操作来调用父类了。

  1. class Computer { protected $_name = '联想120'; protected function run() { echo '我是父类';
  2. }
  3. } class NoteBookComputer extends Computer { public function getName() { echo $this->_name;
  4. } public function getRun() { echo $this->run ();
  5. }
  6. }

通过重写调用父类的方法

有的时候,我们需要通过重写的方法里能够调用父类的方法内容,这个时候就必须使用

语法:父类名::方法() 或者parent::方法()即可调用。

  1. class Computer { protected function run() { echo '我是父类';
  2. }
  3. } class NoteBookComputer extends Computer { public function run() { echo Computer::run ();
  4. }
  5. }

final关键字可以防止类被继承,有些时候只想做个独立的类,不想被其他类继承使用,

那么就必须使用这个关键字。建议只要是单独的类,都加上这个关键字。

  1. final class Computer { //无法继承的类
  2. final public function run() {
  3. } //无法被继承的方法
  4. } class NoteBookComputer extends Computer { //会报错
  5. }

抽象类和方法(abstract)

抽象方法很特殊,只在父类中声明,但在子类中实现。只有声明为abstract的类可以声

明抽象方法。

规则:

1.抽象类不能被实例化,只能被继承。

2.抽象方法必须被子类方法重写。

  1. abstract class Computer { abstract function run();
  2. } final class NotebookComputer extends Computer { public function run() { echo '我实现了';
  3. }
  4. }

接口(interface)

接口定义了实现某种服务的一般规范,声明了所需的函数和常量,但不指定如何实现。

之所以不给出实现的细节,是因为不同的实体可能需要用不同的方式来实现公共的方法定

义。关键是要建立必须实现的一组一般原则,只要满足了这些原则才能说实现了这个接口。

规则:

1.类全部为抽象方法(不需要声明abstract)

2.接口抽象方法必须是public

3.成员(字段)必须是常量

  1. interface Computer { const NAME = '联想120'; public function run();
  2. } final class NotebookComputer implements Computer { public function run() { echo '实现了接口的方法';
  3. }
  4. } $notebookcomputer = new NoteBookComputer (); $notebookcomputer->run (); echo Computer::NAME;

子类可以实现多个接口

  1. interface Computer { const NAME = '联想120'; public function run();
  2. } interface Notebook { public function book();
  3. } final class NotebookComputer implements Computer, Notebook { public function run() { echo '实现了接口的方法';
  4. } public function book() { echo '实现了接口的方法';
  5. }
  6. }

三.多态

多态是指OOP 能够根据使用类的上下文来重新定义或改变类的性质或行为,或者说接

口的多种不同的实现方式即为多态。把不同的子类对象都当作父类来看,可以屏蔽不同子类

对象之间的差异,写出通用的代码,做出通用的编程,以适应需求的不断变化。

  1. interface Computer { public function version(); public function work();
  2. } class NotebookComputer implements Computer { public function version() { echo '联想120';
  3. } public function work() { echo '笔记本正在随时携带运行!';
  4. }
  5. } class desktopComputer implements Computer { public function version() { echo 'IBM';
  6. } public function work() { echo '台式电脑正在工作站运行!';
  7. }
  8. } class Person { public function run($type) { $type->version (); $type->work ();
  9. }
  10. } $person = new Person (); $desktopcomputer = new desktopComputer (); $notebookcomputer = new NoteBookComputer (); $person->run ( $notebookcomputer );

注:文章出自李炎恢PHP视频教程,本文仅限交流使用,不得用于商业用途,否则后果自负。

原文: https://wizardforcel.gitbooks.io/liyanhui-tutorials/content/99.html