詳解PHP中abstract(抽象)、final(最終)與static(靜態)原理與用法

coldplay.xixi
發布: 2023-04-09 08:08:01
轉載
3732 人瀏覽過

詳解PHP中abstract(抽象)、final(最終)與static(靜態)原理與用法

本文實例講述了PHP中abstract(抽象)、final(最終)和static(靜態)原理與用法。分享給大家參考,具體如下:

abstract(抽象)

PHP 5 支援抽象類別和抽象方法。定義為抽象的類別不能被實例化。任何一個類,如果它裡面至少有一個方法是被聲明為抽象的,那麼這個類別就必須被宣告為抽象的。被定義為抽象的方法只是宣告了其呼叫方式(參數),不能定義其特定的功能實作

繼承一個抽象類別的時候,子類別必須定義父類別中的所有抽象方法;另外,這些方法的存取控制必須和父類別中一樣(或更為寬鬆)。例如某個抽象方法被宣告為受保護的,那麼子類別中實作的方法就應該宣告為受保護的或公有的(嚴格程度:private>protected>public),而不能定義為私有的。此外方法的呼叫方式必須匹配,即類型和所需參數數量必須一致。例如,子類別定義了一個可選參數,而父類別抽象方法的聲明裡沒有,則兩者的宣告並無衝突。這也適用於 PHP 5.4 起的建構子。在 PHP 5.4 之前的建構子宣告可以不一樣的。

相關學習推薦:PHP程式設計從入門到精通

總結:

  • 抽象類別不能被實例化;

  • 類別中有任何抽象方法那這個類別也必須為抽象的;

  • 抽象類別只能申明呼叫方式和參數,不能定義具體功能實作;

  • 繼承抽象類別的子類別必須實作抽象類別的所有抽象方法;

  • #子類別中實作的抽象方法的存取控制必須比父類別的存取控制更嚴格;

  • 子類別中實作的方法的呼叫方式及參數數量必須與被實現的方法一致。

範例:

<?php
abstract class AbstractClass
{
  // 强制要求子类定义这些方法,不定义功能实现
  abstract protected function getValue();
  abstract protected function prefixValue($prefix);

  // 普通方法(非抽象方法),子类可以不重写
  public function printOut() {
    print $this->getValue() . "\n";
  }
}

class ConcreteClass1 extends AbstractClass
{
  protected function getValue() {
    return "ConcreteClass1";
  }

  public function prefixValue($prefix) {
    return "{$prefix}ConcreteClass1";
  }
}

class ConcreteClass2 extends AbstractClass
{
 //访问方式可以更宽松
  public function getValue() {
    return "ConcreteClass2";
  }

  public function prefixValue($prefix) {
    return "{$prefix}ConcreteClass2";
  }
}

$class1 = new ConcreteClass1;
$class1->printOut();
echo $class1->prefixValue(&#39;FOO_&#39;) ."\n";

$class2 = new ConcreteClass2;
$class2->printOut();
echo $class2->prefixValue(&#39;FOO_&#39;) ."\n";
?>
登入後複製
<?php
abstract class AbstractClass
{
  // 我们的抽象方法仅需要定义需要的参数
  abstract protected function prefixName($name);

}

class ConcreteClass extends AbstractClass
{

  // 我们的子类可以定义父类签名中不存在的 可选参数
  public function prefixName($name, $separator = ".") {
    if ($name == "Pacman") {
      $prefix = "Mr";
    } elseif ($name == "Pacwoman") {
      $prefix = "Mrs";
    } else {
      $prefix = "";
    }
    return "{$prefix}{$separator} {$name}";
  }
}

$class = new ConcreteClass;
echo $class->prefixName("Pacman"), "\n";
echo $class->prefixName("Pacwoman"), "\n";
?>
登入後複製

final

#如果父類別中的方法被宣告為 final,則子類別無法覆寫該方法。如果一個類別被宣告為 final,則不能被繼承。

這個比較好理解,不做贅述

static

聲明類別屬性或方法為靜態,就可以不實例化類別而直接存取。靜態屬性不能透過一個類別已實例化的物件來存取(但靜態方法可以)。

為了相容於 PHP 4,如果沒有指定存取控制,則屬性和方法預設為公有。

由於靜態方法不需要透過物件即可調用,所以偽變數 $this 在靜態方法中不可用。

靜態屬性不可以由物件透過 -> 操作符來存取。

用靜態方式呼叫非靜態方法會導致一個 E_STRICT 層級的錯誤。

就像是它所有的 PHP 靜態變數一樣,靜態屬性只能被初始化為文字或常數,不能使用表達式。所以可以把靜態屬性初始化為整數或數組,但不能初始化為另一個變數或函數回傳值,也不能指向一個物件。

自 PHP 5.3.0 起,可以用一個變數來動態呼叫類別。但該變數的值不能為關鍵字 self,parent 或 static。

總結:

  • 靜態方法無需實例化,可直接存取;

  • 類別實例化的物件無法存取類別中的靜態屬性,但是可以存取靜態方法;

  • #偽變數$this 在靜態方法中不可用;

  • #靜​​態屬性不可以由物件透過-> 運算子來存取;

  • 以靜態方式呼叫非靜態方法會導致一個E_STRICT 層級的錯誤;

  • 靜態屬性只能初始化為文字或常數,不能使用表達式(函數傳回值/寧一個變數/物件);

  • 可以用一個變數來動態呼叫類別。但該變數的值不能為關鍵字 self,parent 或 static。

<?php
class Foo
{
  public static $my_static = &#39;foo&#39;;

  public function staticValue() {
    return self::$my_static;
  }
}

class Bar extends Foo
{
  public function fooStatic() {
    return parent::$my_static;
  }
}

print Foo::$my_static . "\n";

$foo = new Foo();
print $foo->staticValue() . "\n";
print $foo->my_static . "\n";   // Undefined "Property" my_static 

print $foo::$my_static . "\n";
$classname = &#39;Foo&#39;;
print $classname::$my_static . "\n"; // As of PHP 5.3.0

print Bar::$my_static . "\n";
$bar = new Bar();
print $bar->fooStatic() . "\n";
?>
  </programlisting>
 </example>

 <example>
  <title>静态方法示例</title>
  <programlisting role="php">
<![CDATA[
<?php
class Foo {
  public static function aStaticMethod() {
    // ...
  }
}

Foo::aStaticMethod();
$classname = &#39;Foo&#39;;
$classname::aStaticMethod(); // 自 PHP 5.3.0 起
?>
登入後複製

以上是詳解PHP中abstract(抽象)、final(最終)與static(靜態)原理與用法的詳細內容。更多資訊請關注PHP中文網其他相關文章!

相關標籤:
來源:jb51.net
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
作者最新文章
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板
關於我們 免責聲明 Sitemap
PHP中文網:公益線上PHP培訓,幫助PHP學習者快速成長!