主頁 > 後端開發 > 簡述PHP中trait的使用和同時引入多個trait時同名方法沖突的處理

簡述PHP中trait的使用和同時引入多個trait時同名方法沖突的處理

2023-04-26 07:21:26 後端開發

PHP的類是單一繼承模式,也就是每個類只能繼承一個父類(基類),

但有時需要引入更多通用(共用)的方法,同時這些方法又不適合集成到基類,

那么這時,就需要使用其他方法來引入這些方法,其中trait,就是方法之一,

trait是PHP5.4之后出現的一種代碼復用方法,形式和Class非常相似,同時可以隨意組合任意引入,

trait一般在當前類和其同父類(基類)的其他類都需要使用相同方法時,而其父類(基類)又要盡量避免出現這些方法時使用,

甚至有時可能其他關聯不是特別大的類(分別繼承不同的父類)也可能會使用共同的方法,也可以使用trait的方法,

 

盡量通俗一點的說一下trait:

trait像類,但不是類,不可以直接使用new關鍵字創建物件;簡單理解是用類的形式,封裝一大堆通用(共用)的方法,供其他類參考,

trait和use搭配使用,定義好trait后,“use trait定義的名字;”,就可以直接使用里邊定義的一切了,是不是很簡單?很方便?

現在知道了trait,接下來就通過代碼實體,演示一下trait的具體使用和一些小情況,

 

一、trait的使用

代碼:

// trait
trait traitTest {
    public function test() {
        echo "trait test...\n";
    }
}
// 父類
class ParentClass {
    public function parent() {
        echo "parent...\n";
    }
}
// 子類
class SubClass extends ParentClass {
    use traitTest;
    public function sub() {
        echo "sub...\n";
    }
}

$obj = new SubClass;
$obj->sub();// 呼叫子類方法
$obj->parent();// 呼叫父類的方法
$obj->test();// 呼叫trait里的方法

代碼和結果截圖:

 

上邊的這個例子,子類直接extentds父類,然后又在類內use了trait,這樣當前類(子類)就擁有了這三個的全部方法,

子類的sub方法,父類的parent方法,trait的test方法,在子類內都可以直接呼叫使用,

最基礎的使用就這些,看起來是不是也不算難?甚至感覺挺簡單的?

那么我們進一步思考一下,類的“繼承”難免會出現同名方法,那么這三個里邊如果有同名方法,最侄訓保留哪個?誰的方法會被覆寫呢?

 

二、當父類、子類和trait的方法重名

代碼:

// trait
trait traitTest {
    public function test() {
        echo "trait test...\n";
    }
    public function lookClassName() {
        echo "trait here\n";
        echo __CLASS__ . "\n";
    }
}
// 父類
class ParentClass {
    public function parent() {
        echo "parent...\n";
    }
    public function lookClassName() {
        echo __CLASS__ . "\n";
    }
}
// 子類
class SubClass extends ParentClass {
    use traitTest;

    public function sub() {
        echo "sub...\n";
    }
    public function lookClassName() {
        echo __CLASS__ . "\n";
    }
}

$obj = new SubClass;
$obj->sub();// 呼叫子類方法
$obj->parent();// 呼叫父類的方法
$obj->test();// 呼叫trait里的方法
$obj->lookClassName();// 呼叫同名方法

代碼和結果截圖:

上邊這段例子的結果很明顯的發現,最終當前類(子類)的方法被呼叫了,也就是三個里邊都有同名方法時,當前類的方法優先,

接下來,注釋(洗掉)當前類的lookClassName()方法,

看上邊截圖,很明顯了,當子類(當前類)沒有同名方法,只有父類(基類)和trait中的方法同名時,trait中的方法優先,

結論:當前類(子類)、trait和父類(基類)中有同名方法時“子類高于trait高于父類”,子類的方法會覆寫trait中的方法,而trait的方法會覆寫父類的方法,

 

前邊有提到,trait可以隨意組合,隨意參考,那么是不是可以同時引入多個trait呢?是,在一個類內,可以同時use多個trait,

 

三、類內同時引入多個trait

// trait
trait traitTest {
    public function test() {
        echo "trait test...\n";
    }
    public function lookClassName() {
        echo "trait here\n";
        echo __CLASS__ . "\n";
    }
}

trait traitTest2 {
    public function test2() {
        echo "trait2 test...\n";
    }
    public function lookClassName() {
        echo "trait2 here\n";
        echo __CLASS__ . "\n";
    }
}

trait traitTest3 {
    public function test3() {
        echo "trait3 test...\n";
    }
    public function lookClassName() {
        echo "trait3 here\n";
        echo __CLASS__ . "\n";
    }
}
// 父類
class ParentClass {
    public function parent() {
        echo "parent...\n";
    }
    public function lookClassName() {
        echo __CLASS__ . "\n";
    }
}
// 子類
class SubClass extends ParentClass {
    use traitTest;
    use traitTest2, traitTest3;
    
    public function sub() {
        echo "sub...\n";
    }
    public function lookClassName() {
        echo __CLASS__ . "\n";
    }
}

$obj = new SubClass;
$obj->sub();// 呼叫子類方法
$obj->parent();// 呼叫父類的方法
$obj->test();// 呼叫trait里的方法
$obj->test2();// 呼叫trait2里的方法
$obj->test3();// 呼叫trait3里的方法
$obj->lookClassName();// 呼叫同名方法

代碼和結果截圖:

當需要同時引入多個trait時,只要use trait1, trait2, trait3,在use后邊跟多個trait名字即可,多個trait之間用逗號分隔,

當然,也可以分開寫,每次use一個trait進來,

此時又有新的問題產生了,如果引入的多個trait都有同名的方法,那么又會是誰優先?誰又被覆寫呢?

 

四、引入多個trait有同名方法

代碼:

// trait
trait traitTest {
    public function test() {
        echo "trait test...\n";
    }
    public function lookClassName() {
        echo "trait here\n";
        echo __CLASS__ . "\n";
    }
}

trait traitTest2 {
    public function test2() {
        echo "trait2 test...\n";
    }
    public function lookClassName() {
        echo "trait2 here\n";
        echo __CLASS__ . "\n";
    }
}

trait traitTest3 {
    public function test3() {
        echo "trait3 test...\n";
    }
    public function lookClassName() {
        echo "trait3 here\n";
        echo __CLASS__ . "\n";
    }
}
// 父類
class ParentClass {
    public function parent() {
        echo "parent...\n";
    }
    public function lookClassName() {
        echo __CLASS__ . "\n";
    }
}
// 子類
class SubClass extends ParentClass {
    use traitTest, traitTest2, traitTest3 {
        traitTest2::lookClassName insteadof traitTest;// traitTest2代替了traitTest
        traitTest3::lookClassName insteadof traitTest2;// traitTest3代替了traitTest2
    }
    
    public function sub() {
        echo "sub...\n";
    }
    // public function lookClassName() {
    //     echo __CLASS__ . "\n";
    // }
}

$obj = new SubClass;
$obj->sub();// 呼叫子類方法
$obj->parent();// 呼叫父類的方法
$obj->test();// 呼叫trait里的方法
$obj->test2();// 呼叫trait2里的方法
$obj->test3();// 呼叫trait3里的方法
$obj->lookClassName();// 呼叫同名方法

代碼和結果截圖:

說明(上邊的原始碼和結果是解沖突之后的): 

當子類沒有(注釋或者洗掉)lookClassName()方法時,呼叫lookClassName方法,則會呼叫trait中的方法,因為三個trait中都有同名方法,此時就會發生致命錯誤(沖突),

報下邊(看截圖)的語法錯誤

此時,就需要解沖突,

解沖突,就需要使用到insteadof關鍵字,含義是“代替”,就是用哪個代替哪個,

    use traitTest, traitTest2, traitTest3 {
        traitTest2::lookClassName insteadof traitTest;// traitTest2代替了traitTest
        traitTest3::lookClassName insteadof traitTest2;// traitTest3代替了traitTest2
    }

解引入多個trait多個重名方法沖突時,需要在引入時使用insteadof關鍵字,逐一說明哪個trait的方法代替了哪個trait的(看上邊引入代碼的注釋),

根據上邊引入的代碼,是traitTest2的lookClassName代替了traitTest的,然后traitTest3的代替了traitTest2的,

因此,最終輸出結果時,呼叫lookClassName(),輸出的就是traitTest3的內容(輸出結果看上邊最近的“代碼和結果截圖”),

當然,也可以換個寫法:

    use traitTest, traitTest2, traitTest3 {
        traitTest2::lookClassName insteadof traitTest3;// traitTest2代替了traitTest3
        traitTest3::lookClassName insteadof traitTest2;// traitTest3代替了traitTest2
    }

這個寫法呢,是traitTest2和traitTest3互相代替了,那么此時反而沒有traitTest什么事了,這個時候,再呼叫lookClassName()方法,輸出的就是traitTest的lookClassName()方法的內容,

代碼和結果截圖:

如圖,當traitTest2和traitTest3互相代替后,直接輸出了traitTest的內容,

 

到這基本就該結束了,但,有個特殊情況需要考慮一下,

我們之所以會引入多個trait,說明這幾個trait里都有想使用的方法,那么非常巧合,其中同名方法正好又都想使用,被代替的方法還能使用么?

 

五、當引入多個trait,同名方法解沖突后,同時使用所有沖突的同名方法

解決:我們需要使用到另一個關鍵字“as”,此關鍵字的功能,簡單理解就是給方法取一個別名,

代碼:

// trait
trait traitTest {
    public function test() {
        echo "trait test...\n";
    }
    public function lookClassName() {
        echo "trait here\n";
        echo __CLASS__ . "\n";
    }
}

trait traitTest2 {
    public function test2() {
        echo "trait2 test...\n";
    }
    public function lookClassName() {
        echo "trait2 here\n";
        echo __CLASS__ . "\n";
    }
}

trait traitTest3 {
    public function test3() {
        echo "trait3 test...\n";
    }
    public function lookClassName() {
        echo "trait3 here\n";
        echo __CLASS__ . "\n";
    }
}
// 父類
class ParentClass {
    public function parent() {
        echo "parent...\n";
    }
    public function lookClassName() {
        echo __CLASS__ . "\n";
    }
}
// 子類
class SubClass extends ParentClass {
    use traitTest, traitTest2, traitTest3 {
        traitTest2::lookClassName insteadof traitTest3;// traitTest2代替了traitTest3
        traitTest3::lookClassName insteadof traitTest2;// traitTest3代替了traitTest2
        traitTest2::lookClassName as lookClassName2;// traitTest2的lookClassName改別名lookClassName2
        traitTest3::lookClassName as lookClassName3;// traitTest3的lookClassName改別名lookClassName3
    }
    
    public function sub() {
        echo "sub...\n";
    }
    // public function lookClassName() {
    //     echo __CLASS__ . "\n";
    // }
}

$obj = new SubClass;
$obj->sub();// 呼叫子類方法
$obj->parent();// 呼叫父類的方法
$obj->test();// 呼叫trait里的方法
$obj->test2();// 呼叫trait2里的方法
$obj->test3();// 呼叫trait3里的方法
$obj->lookClassName();// 呼叫同名方法
$obj->lookClassName2();// 呼叫traitTest2更名后的同名方法
$obj->lookClassName3();// 呼叫traitTest3更名后的同名方法

代碼和結果截圖:

根據上圖,就可以看出,當trait2和trait3互相代替,然后同名方法另起別名后,三個trait的同名方法,不再沖突,并且可以分別呼叫各自原本同名的方法,

到此要說的東西基本都說完了,算是對PHP的trait的一個小小的總結,希望可以幫到需要的朋友,

若有不對之處,請賜教,

 

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/551158.html

標籤:PHP

上一篇:這可能是最全面的MySQL面試八股文了

下一篇:返回列表

標籤雲
其他(158071) Python(38099) JavaScript(25391) Java(18001) C(15217) 區塊鏈(8260) C#(7972) AI(7469) 爪哇(7425) MySQL(7144) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5870) 数组(5741) R(5409) Linux(5329) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4561) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2431) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1960) Web開發(1951) HtmlCss(1926) python-3.x(1918) 弹簧靴(1913) C++(1911) xml(1889) PostgreSQL(1874) .NETCore(1855) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • 簡述PHP中trait的使用和同時引入多個trait時同名方法沖突的處理

    PHP的類是單一繼承模式,也就是每個類只能繼承一個父類(基類)。 但有時需要引入更多通用(共用)的方法,同時這些方法又不適合集成到基類。 那么這時,就需要使用其他方法來引入這些方法。其中trait,就是方法之一。 trait是PHP5.4之后出現的一種代碼復用方法,形式和Class非常相似,同時可以 ......

    uj5u.com 2023-04-26 07:21:26 more
  • 這可能是最全面的MySQL面試八股文了

    什么是MySQL MySQL是一個關系型資料庫,它采用表的形式來存盤資料。你可以理解成是Excel表格,既然是表的形式存盤資料,就有表結構(行和列)。行代表每一行資料,列代表該行中的每個值。列上的值是有資料型別的,比如:整數、字串、日期等等。 資料庫的三大范式 第一范式1NF 確保資料庫表欄位的原 ......

    uj5u.com 2023-04-25 09:31:53 more
  • 【Jmeter】按比例分配Api壓測

    先看 【Jmeter】基礎介紹-詳細 【Jmeter】Request1輸出作為Request2輸入-后置處理器 繼續聊提出的第二個問題,即 2.需要按比例分配API請求并發,以模擬真實的API壓力場景 做壓測的時候,一般的需求都是多個API同時壓,不然也看不出真正的tps是多少啊。 比如雖然介面a的 ......

    uj5u.com 2023-04-25 09:30:35 more
  • 這可能是最全面的MySQL面試八股文了

    什么是MySQL MySQL是一個關系型資料庫,它采用表的形式來存盤資料。你可以理解成是Excel表格,既然是表的形式存盤資料,就有表結構(行和列)。行代表每一行資料,列代表該行中的每個值。列上的值是有資料型別的,比如:整數、字串、日期等等。 資料庫的三大范式 第一范式1NF 確保資料庫表欄位的原 ......

    uj5u.com 2023-04-25 09:29:15 more
  • 【Jmeter】按比例分配Api壓測

    先看 【Jmeter】基礎介紹-詳細 【Jmeter】Request1輸出作為Request2輸入-后置處理器 繼續聊提出的第二個問題,即 2.需要按比例分配API請求并發,以模擬真實的API壓力場景 做壓測的時候,一般的需求都是多個API同時壓,不然也看不出真正的tps是多少啊。 比如雖然介面a的 ......

    uj5u.com 2023-04-25 09:22:32 more
  • Springboot啟動原理和自動配置原理

    放本地檔案夾都快吃土了,準備清理檔案夾,關于Springboot的! 啟動原理 @SpringBootApplication public class Start { public static void main(String[] args) { SpringApplication.run(Sta ......

    uj5u.com 2023-04-25 08:05:03 more
  • OpenAI ChatGPT 能取代多少程式員的作業?導致失業嗎?

    閱讀原文:https://bysocket.com/openai-chatgpt-vs-developer/ ChatGPT 能取代多少程式員的作業?導致我們程式員失業嗎?這是一個很好的話題,我這里分享下: 一、ChatGPT 是什么?有什么作用 ChatGPT是一種基于人工智能技術的語言模型,是可 ......

    uj5u.com 2023-04-25 08:04:57 more
  • 自定義Python版本ESL庫訪問FreeSWITCH

    環境:CentOS 7.6_x64Python版本:3.9.12FreeSWITCH版本 :1.10.9 一、背景描述 ESL庫是FreeSWITCH對外提供的介面,使用起來很方便,但該庫是基于C語言實作的,Python使用該庫的話需要使用原始碼進行編譯。如果使用系統自帶的Python版本進行編譯,過 ......

    uj5u.com 2023-04-25 08:04:42 more
  • Rust編程語言入門之高級特性

    高級特性 主要內容 不安全 Rust 高級 Trait 高級 型別 高級函式和閉包 宏 一、不安全 Rust 匹配命名變數 隱藏著第二個語言,它沒有強制記憶體安全保證:Unsafe Rust(不安全的 Rust) 和普通的 Rust 一樣,但提供了額外的“超能力” Unsafe Rust 存在的原因: ......

    uj5u.com 2023-04-25 08:04:32 more
  • Go中的有限狀態機FSM的詳細介紹

    1、FSM簡介 1.1 有限狀態機的定義 有限狀態機(Finite State Machine,FSM)是一種數學模型,用于描述系統在不同狀態下的行為和轉移條件。 狀態機有三個組成部分:狀態(State)、事件(Event)、動作(Action),事件(轉移條件)觸發狀態的轉移和動作的執行。動作的執 ......

    uj5u.com 2023-04-25 08:04:21 more