当前位置:首页 > PHP教程 > php面向对象 > 列表

详解php中的类与对象(继承)

发布:smiling 来源: PHP粉丝网  添加日期:2021-08-23 10:27:13 浏览: 评论:0 

在php中,类型的继承使用extends关键字,而且最多只能继承一个父类,php不支持多继承,这篇文章主要介绍了php中的类与对象(继承),需要的朋友可以参考下

简介

在php中,类型的继承使用extends关键字,而且最多只能继承一个父类,php不支持多继承。

  1. class MyClass   
  2. {  
  3.  public $dat = 0;  
  4.  public function __construct($dat) {  
  5.   $this->dat = $dat;  
  6.  }  
  7.  public function getDat() {  
  8.   return "$this->dat\n";  
  9.  }  
  10. }  
  11. class MySubClass extends MyClass  
  12. {  
  13.  public function getDat() {  
  14.   return "dat: $this->dat\n";  
  15.  }  
  16. }  
  17. $a = new MyClass(3);  
  18. $b = new MySubClass(4);  
  19. echo $a->getDat();  // 3  
  20. echo $b->getDat();  // dat: 4 

方法覆盖

包括构造函数在内,子类可以重新定义同名的类方法以覆盖父类方法。覆盖时遵循以下规则:

1.除构造函数之外,其他函数在覆盖时,函数的参数列表必须相同

2.包括构造函数在内,方法被覆盖后,调用子类方法时并不会自动调用父类方法

3.如果父类要禁止方法被子类覆盖,可以使用final来声明方法,这时如果子类仍要覆盖父类方法,将会出错

  1. class MyClass   
  2. {  
  3.  private $name = "";  
  4.  public $num = 0;  
  5.  public $str = "";  
  6.  public function __construct($name) {  
  7.   $this->name = $name;  
  8.   $this->num = 100;  
  9.   $this->str = "none";  
  10.  }  
  11.  public function getName() {  
  12.   return $this->name;  
  13.  }  
  14. }  
  15. class MySubClass extends MyClass  
  16. {  
  17.  public function __construct($name$str) {  
  18.   parent::__construct($name);    // 调用父类方法  
  19.   $this->num = "0";  
  20.   $this->str = $str;  
  21.   echo parent::getName()."\n";    // 调用父类方法  
  22.  }  
  23.  public function getName() {  
  24.   return parent::getName()."$this->str\n"// 调用父类方法  
  25.  }  
  26. }  
  27. $b = new MySubClass("myName", true);  // myName  
  28. echo $b->getName();          // myName1  
  29. class MyClass   
  30. {  
  31.  final public function getName() {  
  32.  }  

属性重定义

在子类中,可以访问父类中的public和protected属性成员,除非重定义了同名的自有属性,这时,父类中的属性将无法访问。

方法则不同,子类对方法进行覆盖后,仍然可以访问到父类方法。

  1. class MyClass   
  2. {  
  3.  public $a = 1;  
  4.  protected $b = 2;  
  5.  private $c = 3;  
  6.  public function f1() {  
  7.   echo "MyClass f1\n";  
  8.   echo "\$a:$this->a; \$b:$this->b; \$c:$this->c;\n";  
  9.  }  
  10.  protected function f2() {  
  11.   echo "MyClass f2\n";  
  12.   echo "\$a:$this->a; \$b:$this->b; \$c:$this->c;\n";  
  13.  }  
  14.  private function f3() {  
  15.   echo "MyClass f3\n";  
  16.  }  
  17. }  
  18. class MySubClass extends MyClass   
  19. {  
  20.  public $b = 22;  
  21.  public $c = 33;  
  22.  public function f1() {  
  23.   echo "MySubClass f1\n";  
  24.   // 继承到父类中的$a属性,直接使用  
  25.   echo "\$a:$this->a; \$b:$this->b; \$c:$this->c;\n";  
  26.   // 调用父类中的同名方法  
  27.   parent::f1();  
  28.   // 继承到父类中的f2()方法,直接使用  
  29.   $this->f2();  
  30.  }  
  31.  // 父类的f3()是私有的,这里的定义与父类无关  
  32.  public function f3() {  
  33.   echo "MySubClass f3\n";  
  34.  }  
  35. }  
  36. $b = new MySubClass;  
  37. $b->f1();echo "\n";  
  38. /*  
  39. MySubClass f1  
  40. $a:1; $b:22; $c:33;  
  41. MyClass f1  
  42. $a:1; $b:22; $c:3;  
  43. MyClass f2  
  44. $a:1; $b:22; $c:3;  
  45. */ 
  46. $b->f3();echo "\n";  
  47. /*  
  48. MySubClass f3  
  49. */ 

重定义父类(同名)属性时,属性的可访问性可以变得更开放,但不能更严格,也就是说,父类中的public属性,不能在子类中修改为private属性。

如果通过子类对象调用父类方法,那么该父类方法在访问属性时,对于重定义了的同名属性,public和protected的属性将访问到子类版本,private属性将访问到父类版本。也可以理解为,public和protected属性可以被重定义(父类的版本被重定义,从而不存在了),而private并未被重定义(父类中的属性仍然存在,通过父类方法进行访问,与子类中是否有同名属性毫不相干)。

  1. class MyClass   
  2. {  
  3.  public $a = 1;  
  4.  protected $b = 2;  
  5.  private $c = 3;  
  6.  public function f1() {  
  7.   echo "\$a:$this->a; \$b:$this->b; \$c:$this->c;\n";  
  8.  }  
  9. }  
  10. class MySubClass extends MyClass   
  11. {  
  12.  public $a = 11;   // 必须为public  
  13.  protected $b = 22; // 必须为protected或public  
  14.  private $c = 33;    
  15.  public function f2() {  
  16.   echo "\$a:$this->a; \$b:$this->b; \$c:$this->c;\n";  
  17.  }  
  18. }  
  19. $b = new MySubClass;  
  20. $b->f1(); // $a:11; $b:22; $c:3;  
  21. $b->f2(); // $a:11; $b:22; $c:33; 

范围解析操作符 ::

又冒号常用于访问类常量、类静态变量,也用于在方法覆盖时调用父类版本。与其搭配的还包括parent、self、static等关键字。

  1. class MyClass   
  2. {  
  3.  const Name0 = "MyClass";  // 类常量  
  4.  public static $id0 = 0;  // 类变量  
  5.  public function put() {  // 将被子类覆盖的方法  
  6.   echo "MyClass put()\n";  
  7.  }  
  8. }  
  9. class MySubClass extends MyClass   
  10. {  
  11.  const Name1 = "MySubClass";  
  12.  public static $id1 = 1;   
  13.  public function put() {  
  14.   parent::put();        // 调用父类版本的对象方法  
  15.   echo parent::Name0 . "\n";  // 父类常量  
  16.   echo parent::$id0 . "\n";   // 父类变量  
  17.   echo self::Name1."\n";    // 子类常量  
  18.   echo self::$id1 . "\n";    // 子类变量  
  19.   echo static::Name1 . "\n";  // 子类常理  
  20.   echo static::$id1 . "\n";   // 子类变量  
  21.  }  
  22. }  
  23. $a = "MyClass";  
  24. $ca = new MyClass;  
  25. $cb = new MySubClass;   
  26. $cb->put();  
  27. echo MyClass::Name0 . "\n";  
  28. echo MyClass::$id0 . "\n";  
  29. echo $a::Name0 . "\n";  
  30. echo $a::$id0 . "\n";  
  31. echo $ca::Name0 . "\n";  
  32. echo $ca::$id0 . "\n"

在子类中访问父类中的成员时,应避免直接使用父类类名,而应使用parent::,以免破坏父类的封装性。

final

声明为final的方法不能被子类覆盖,如果类声明为final,则此类不能被继承。

  1. // 声明为final的类不能被继承  
  2. final class MyClass  
  3. {  
  4.  private $dat;  
  5.  public function __construct($dat) {  
  6.   $this->dat = $dat;  
  7.  }  
  8.  // final方法不能被覆盖,不过此类已经是final类,方法无必要在声明为final了  
  9.  final public function getDat() {  
  10.   return $this->dat;  
  11.  }  
  12. }

Tags: php类 php继承

分享到: