当前位置:首页 > PHP教程 > php应用 > 列表

详解PHP中abstract(抽象)、final(最终)和static(静态)原理与用法

发布:smiling 来源: PHP粉丝网  添加日期:2022-07-08 11:04:45 浏览: 评论:0 

本文实例讲述了PHP中abstract(抽象)、final(最终)和static(静态)原理与用法,分享给大家供大家参考,具体如下:

abstract(抽象)

PHP 5 支持抽象类和抽象方法。定义为抽象的类不能被实例化。任何一个类,如果它里面至少有一个方法是被声明为抽象的,那么这个类就必须被声明为抽象的。被定义为抽象的方法只是声明了其调用方式(参数),不能定义其具体的功能实现。

继承一个抽象类的时候,子类必须定义父类中的所有抽象方法;另外,这些方法的访问控制必须和父类中一样(或者更为宽松)。例如某个抽象方法被声明为受保护的,那么子类中实现的方法就应该声明为受保护的或者公有的(严格程度:private>protected>public),而不能定义为私有的。此外方法的调用方式必须匹配,即类型和所需参数数量必须一致。例如,子类定义了一个可选参数,而父类抽象方法的声明里没有,则两者的声明并无冲突。 这也适用于 PHP 5.4 起的构造函数。在 PHP 5.4 之前的构造函数声明可以不一样的。

总结:

抽象类不能被实例化;

类中有任何抽象方法那这个类也必须为抽象的;

抽象类只能申明调用方式和参数,不能定义具体功能实现;

继承抽象类的子类必须实现抽象类的所有抽象方法;

子类中实现的抽象方法的访问控制必须比父类的访问控制更严格;

子类中实现的方法的调用方式及参数数量必须与被实现的方法一致。

例:

  1. <?php 
  2.  
  3. abstract class AbstractClass 
  4.  
  5.  
  6.   // 强制要求子类定义这些方法,不定义功能实现 
  7.  
  8.   abstract protected function getValue(); 
  9.  
  10.   abstract protected function prefixValue($prefix); 
  11.  
  12.  
  13.  
  14.   // 普通方法(非抽象方法),子类可以不重写 
  15.  
  16.   public function printOut() { 
  17.  
  18.     print $this->getValue() . "\n"
  19.  
  20.   } 
  21.  
  22.  
  23.  
  24.  
  25. class ConcreteClass1 extends AbstractClass 
  26.  
  27.  
  28.   protected function getValue() { 
  29.  
  30.     return "ConcreteClass1"
  31.  
  32.   } 
  33.  
  34.  
  35.  
  36.   public function prefixValue($prefix) { 
  37.  
  38.     return "{$prefix}ConcreteClass1"
  39.  
  40.   } 
  41.  
  42.  
  43.  
  44.  
  45. class ConcreteClass2 extends AbstractClass 
  46.  
  47.  
  48.  //访问方式可以更宽松 
  49.  
  50.   public function getValue() { 
  51.  
  52.     return "ConcreteClass2"
  53.  
  54.   } 
  55.  
  56.  
  57.  
  58.   public function prefixValue($prefix) { 
  59.  
  60.     return "{$prefix}ConcreteClass2"
  61.  
  62.   } 
  63.  
  64.  
  65.  
  66.  
  67. $class1 = new ConcreteClass1; 
  68.  
  69. $class1->printOut(); 
  70.  
  71. echo $class1->prefixValue('FOO_') ."\n"
  72.  
  73.  
  74.  
  75. $class2 = new ConcreteClass2; 
  76.  
  77. $class2->printOut(); 
  78.  
  79. echo $class2->prefixValue('FOO_') ."\n"
  80.  
  81. ?> 
  82.  
  1. <?php 
  2.  
  3. abstract class AbstractClass 
  4.  
  5.  
  6.   // 我们的抽象方法仅需要定义需要的参数 
  7.  
  8.   abstract protected function prefixName($name); 
  9.  
  10.  
  11.  
  12.  
  13.  
  14.  
  15. class ConcreteClass extends AbstractClass 
  16.  
  17.  
  18.  
  19.  
  20.   // 我们的子类可以定义父类签名中不存在的 可选参数 
  21.  
  22.   public function prefixName($name$separator = ".") { 
  23.  
  24.     if ($name == "Pacman") { 
  25.  
  26.       $prefix = "Mr"
  27.  
  28.     } elseif ($name == "Pacwoman") { 
  29.  
  30.       $prefix = "Mrs"
  31.  
  32.     } else { 
  33.  
  34.       $prefix = ""
  35.  
  36.     } 
  37.  
  38.     return "{$prefix}{$separator} {$name}"
  39.  
  40.   } 
  41.  
  42.  
  43.  
  44.  
  45. $class = new ConcreteClass; 
  46.  
  47. echo $class->prefixName("Pacman"), "\n"
  48.  
  49. echo $class->prefixName("Pacwoman"), "\n"
  50.  
  51. ?> 

final

如果父类中的方法被声明为 final,则子类无法覆盖该方法。如果一个类被声明为 final,则不能被继承。

这个比较好理解,不做赘述

static

声明类属性或方法为静态,就可以不实例化类而直接访问。静态属性不能通过一个类已实例化的对象来访问(但静态方法可以)。

为了兼容 PHP 4,如果没有指定访问控制,属性和方法默认为公有。

由于静态方法不需要通过对象即可调用,所以伪变量 $this 在静态方法中不可用。

静态属性不可以由对象通过 -> 操作符来访问。

用静态方式调用一个非静态方法会导致一个 E_STRICT 级别的错误。

就像其它所有的 PHP 静态变量一样,静态属性只能被初始化为文字或常量,不能使用表达式。所以可以把静态属性初始化为整数或数组,但不能初始化为另一个变量或函数返回值,也不能指向一个对象。

自 PHP 5.3.0 起,可以用一个变量来动态调用类。但该变量的值不能为关键字 self,parent 或 static。

总结:

静态方法无需实例化,可直接访问;

类实例化的对象无法访问类中的静态属性,但是可以访问静态方法;

伪变量 $this 在静态方法中不可用;

静态属性不可以由对象通过 -> 操作符来访问;

用静态方式调用一个非静态方法会导致一个 E_STRICT 级别的错误;

静态属性只能被初始化为文字或常量,不能使用表达式(函数返回值/宁一个变量/对象);

可以用一个变量来动态调用类。但该变量的值不能为关键字 self,parent 或 static。

  1. <?php 
  2.  
  3. class Foo 
  4.  
  5.  
  6.   public static $my_static = 'foo'
  7.  
  8.  
  9.  
  10.   public function staticValue() { 
  11.  
  12.     return self::$my_static
  13.  
  14.   } 
  15.  
  16.  
  17.  
  18.  
  19. class Bar extends Foo 
  20.  
  21.  
  22.   public function fooStatic() { 
  23.  
  24.     return parent::$my_static
  25.  
  26.   } 
  27.  
  28.  
  29.  
  30.  
  31. print Foo::$my_static . "\n"
  32.  
  33.  
  34.  
  35. $foo = new Foo(); 
  36.  
  37. print $foo->staticValue() . "\n"
  38.  
  39. print $foo->my_static . "\n";   // Undefined "Property" my_static  
  40.  
  41.  
  42.  
  43. print $foo::$my_static . "\n"
  44.  
  45. $classname = 'Foo'
  46.  
  47. print $classname::$my_static . "\n"// As of PHP 5.3.0 
  48.  
  49.  
  50.  
  51. print Bar::$my_static . "\n"
  52.  
  53. $bar = new Bar(); 
  54.  
  55. print $bar->fooStatic() . "\n"
  56.  
  57. ?> 
  58.  
  59.   </programlisting> 
  60.  
  61.  </example> 
  62.  
  63.  
  64.  
  65.  <example> 
  66.  
  67.   <title>静态方法示例</title> 
  68.  
  69.   <programlisting role="php"
  70.  
  71. <![CDATA[ 
  72.  
  73. <?php 
  74.  
  75. class Foo { 
  76.  
  77.   public static function aStaticMethod() { 
  78.  
  79.     // ... 
  80.  
  81.   } 
  82.  
  83.  
  84.  
  85.  
  86. Foo::aStaticMethod(); 
  87.  
  88. $classname = 'Foo'
  89.  
  90. $classname::aStaticMethod(); // 自 PHP 5.3.0 起 
  91.  
  92. ?>

Tags: abstract final static

分享到: