php中的implements 使用詳解


php類中接口的應用關鍵字是interface、implements了,接口是一種成員屬性全部為抽象或常量的特殊抽象類,implements主要是對類名,類所擁有的方法,以及所傳參數起約束和規范做用,有點像 abstract 抽象類。

類中接口的應用

1.關鍵字:interface

2.關鍵字:implements

1.接口的介紹與創建

接口:一種成員屬性全部為抽象或常量的特殊抽象類。

規則:

1.類中全部為抽象方法。

2.抽象方法前不用加abstract。

3.接口抽象方法屬性為public。

4.成員屬性必須為常量。

格式代碼如下:

interface demo { 
const NAME = "常量對象屬性"; 
function myfun1(); //抽象方法
function myfun2(); //抽象方法,不用具體寫入邏輯 
} 

一,接口的定義和調用

<?php
interface Cinema
{
    const film = '加勒比海盜';
    public function show();
}

class Order implements Cinema
{
    public function show()
    {
        echo "影院 接口開放了<br>";
    }
}

$face = new Order();
echo $face->show();       
echo Cinema::film;   

說明:上面的例子要注意一點,接口的方法名是show,繼承接口的類中必須有show這個方法,不然就會報錯。也就是說接口的方法是假的,真正起作用的是在繼承的類中的方法,這里接口和php的抽象類是不是有點像了?
二,對參數約束比較嚴

<?php

interface Cinema
{
    public function show(Order $show,$num);
}

// 顯示正常
class Order implements Cinema
{
    public $number='0011排';
    public function show(Order $show,$num)
    {
        echo $show->number.$num;
    }
}

$face= new Order();

$face->show(new Order,$num='3人');//輸出 0011排3人

說明:上面的這個例子繼承接口類中,調用接口的方法時,所傳參數要和接口中的參數名要一至。不然就會報錯。

三,接口間的繼承和調用接口傳遞參數

<?php
interface Cinema
{
    public function show();
}

interface Cinema1 extends Cinema
{
    public function show1(Order1 $object,$num);
}

class Order implements Cinema1
{
    public function show()
    {
        echo "准備就緒<br>";
    }

    public function show1(Order1 $object,$num) //注意:show1(Order1 $object,$num)中的Order1必須根繼承類的名子要一樣class Order1。如果不一樣,會報fatal錯誤
    {
        //var_dump($object);
        echo $object->number."$num<br>";
    }
}

class Order1
{
    public $number="0012排";
    function fun(){
        echo ' =================';
    }
}

$show = new Order1;
$show->fun();          
$test = new Order();
$test->show();           
$test->show1($show,$num='6人');
// 輸出 ===============准備就緒0012排6人

說明:上面的例子可以看到,接口Cinemal1繼承了接口Cinemal,類Order繼承了接口Cinemal1。不知道你發現沒有,class類Order當中包括有二個方法,一個是show,一個show1,並且一個也不能少,如果少一個,報fatal錯誤。show1(Order1 $object,$num)中的Order1必須根繼承類的名子要一樣class Order1。如果不一樣,也會報fatal錯誤。那如果一個接口被多個類繼承,並且類名又不一樣,怎么辦呢?那就要用self了,下面會提到:(這是官網手冊的一個示例,鑒於類名的Sting與關鍵詞沖突,我這里改成了Strings,大家可自行更改,不沖突即可):

<?php

interface Comparable
{
    function compare(self $compare);
}

class Strings implements Comparable
{
    private $string;

    function __construct($string)
    {
        $this->string = $string;
    }

    function compare(self $compare)
    {
        return $this->string == $compare->string;
    }
}

class Integer implements Comparable
{
    private $integer;

    function __construct($int)
    {
        $this->integer = $int;
    }

    function compare(self $compare)
    {
        return $this->integer == $compare->integer;
    }
}

$first_int = new Integer(999);
$second_int = new Integer(999);
$first_string = new Strings("慕容博");
$second_string = new Strings("慕容復");

var_dump($first_int->compare($second_int)); // bool(true)
var_dump($first_string->compare($second_string)); // bool(false)

按照官網手冊的意思:可以使用“self”強制對象暗示實現類的方法,但是,我們發現無論是PHP5+那個版本,都報了fatal error:

大致意思是繼承類和接口方法變量兼容問題。

正確做法:去掉 self 即可或者把self 替換成 接口的Comparable名。如:function compare(Comparable $compare)

 四,一個接口多個繼承

 
         
<?php
interface demo {
const NAME = "電影名稱";
function fun1();
function fun2();
}
interface demo2 {
function fun3();
function fun4();
}
interface demo3 {
const TEST = "這里是測試Test";
function fun5();
}
class MyDemo implements demo, demo2 {
function fun1() {
echo "你好";
}
function fun2() {
echo "----------";
}
function fun3() {
echo "我也好<br />";
}
function fun4() {
echo "大家都好<br />";
}
}
class YourDemo extends MyDemo implements demo3 {
function fun5() {
echo "繼承類后引用接口";
}
}
$p = new YourDemo;
$p->fun1();
$p->fun2();
$p->fun3();
$p->fun4();
$p->fun5();

以上輸出

你好----------我也好
大家都好
繼承類后引用接口

上面的例子中我們可以看到接口都使用關鍵字 interface 來定義,並使用關鍵字 implements 來實現接口中的方法,再舉個例子:

<?php
//定義接口
interface User{
    function getDiscount();
    function getUserType();
}

class VipUser implements User{ //VIP用戶 接口實現
    private $discount = 0.8;    // VIP 用戶折扣系數
    function getDiscount() {
        return $this->discount;
    }
    function getUserType() {
        return "VIP用戶";
    }
}
class Goods{
    var $price = 88;
    var $vc;
    function run(User $vc){     //定義 User 接口類型參數,這時並不知道是什么用戶
        $this->vc = $vc;
        $discount = $this->vc->getDiscount();
        $usertype = $this->vc->getUserType();
        echo $usertype."商品價格:".$this->price*$discount;
    }
}
$display = new Goods();
$display ->run(new VipUser); //VIP用戶商品價格:70.4

這個例子演示了一個 PHP 接口的簡單應用。該例子中,User 接口實現用戶的折扣,而在 VipUser 類里面實現了具體的折扣系數。最后商品類 Goods 根據 User 接口來實現不同的用戶報價.

最后總結下:

抽象類和接口的區別

接口是特殊的抽象類,也可以看做是一個模型的規范。接口與抽象類大致區別如下:

1.一個子類如果 implements 一個接口,就必須實現接口中的所有方法(不管是否需要);如果是繼承一個抽象類,只需要實現需要的方法即可。

2.如果一個接口中定義的方法名改變了,那么所有實現此接口的子類需要同步更新方法名;而抽象類中如果方法名改變了,其子類對應的方法名將不受影響,只是變成了一個新的方法而已(相對老的方法實現)。

3.抽象類只能單繼承,當一個子類需要實現的功能需要繼承自多個父類時,就必須使用接口。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM