PHP中实现代码复用的Trait方法的一些特性
在整理“PHP基础”这篇文档时,看到了trait方法,感觉比较陌生,所以今天上午用两个小时的时间,查阅测试了trait方法的一些特性及用法,整理发布了这篇博客。
Trait 是 PHP5.4 中的新特性,是 PHP 多重继承的一种解决方案。例如,需要同时继承两个 Abstract Class, 这将会是件很麻烦的事情,Trait 就是为了解决这个问题。
trait的简单使用
trait使用前需要先定义,trait的定义方式和类的定义方式差不多
trait first_trait {
function first_method() { /* Code Here */ }
function second_method() { /* Code Here */ }
}
同时,如果要在 Class 中使用该 Trait,那么需要使用 use 关键字
class first_class {
// 注意这行,声明使用 first_trait
use first_trait;
}
$obj = new first_class();// Executing the method from trait
$obj->first_method(); // valid
$obj->second_method(); // valid
我们在类中可以直接声明使用被定义好的trait,之后在类被实例化后,就可以使用调用对象中方法的方式直接调用trait中的方法。无疑这增加了很多扩展性,因为类的继承是单一的,而trait的使用却不是单一的。
一个class中使用多个trait
在同一个 Class 中可以使用多个 Trait,多个 Trait之间通过逗号分隔。
例如:
trait first_trait
{
function first_method() { echo "method1"; }
}
trait second_trait {
function second_method() { echo "method2"; }
}
class first_class {
// now using more than one trait
use first_trait, second_trait;
}
$obj= new first_class();
// Valid
$obj->first_method(); // Print : method1
// Valid
$obj->second_method(); // Print : method2
Trait 冲突
多个 Trait 之间同时使用难免会冲突,例如,如果两个 trait 都插入了一个同名的方法,如果没有明确解决冲突将会产生一个致命错误。
为了解决多个 trait 在同一个类中的命名冲突,可以使用关键字:insteadof 以及 as ;
insteadof 操作符用来明确指定使用哪一个冲突方法;
as 操作符用来给冲突的方法重新命名并引入。
trait first_trait {
function first_function() {
echo "From First Trait";
}
}
trait second_trait {
// 这里的名称和 first_trait 一样,会有冲突
function first_function() {
echo "From Second Trait";
}
}
class first_class {
use first_trait, second_trait {
// 在这里声明使用 first_trait 的 first_function 替换
// second_trait 中声明的
first_trait::first_function insteadof second_trait;
second_trait::first_function as second_trait;
}
}
Trait 的抽象方法
我们可以在Trait 中声明抽象方法,这样,使用它的 Class 必须实现它。
trait first_trait {
function first_method() { echo "method1"; }
abstract public function second_method();// 这里可以加入修饰符,说明调用类必须实现它
}
class first_method {
use first_trait;
function second_method() {
/* Code Here */
}
}
Trait优先级
PHP手册说法:优先顺序是当前类的方法覆盖了 trait 的方法,而 trait 覆盖了被继承的方法。
即优先级:当前类方法 > trait方法 > 被继承的方法。
如果在不同优先级中定义了一个同名的方法,那么优先级高的方法将覆盖优先级低的方法;
两个trait属于同一优先级,若出现同名方法,则会报致命错误,具体参见上面。
class Base {
public function sayHello() {
echo 'Hello ';
}
}
trait SayWorld {
public function sayHello() {
parent::sayHello();
echo 'World!';
}
}
class MyHelloWorld extends Base {
use SayWorld;
}
$o = new MyHelloWorld();
$o->sayHello();
结果会输出:
Hello World!
Trait访问权限控制
与类相似的,我们在定义trait的方法时,也可以指定其访问权限(public、protected、private);
我们还可以使用as来修改方法的访问权限。
trait HelloWorld {
public function sayHello() {
echo 'Hello World!';
}
}
// 修改 sayHello 的访问控制
class MyClass1 {
use HelloWorld { sayHello as protected; }
}
// 给方法一个改变了访问控制的别名
// 原版 sayHello 的访问控制则没有发生变化
class MyClass2 {
use HelloWorld { sayHello as private myPrivateHello; }
}
Trait 之间的嵌套
使用trait时,可以将不同的trait组成新的trait,这样使得代码的复用率变得更高,当然对程序的设计要求也更高。
trait Hello {
public function sayHello() {
echo 'Hello ';
}}
trait World {
public function sayWorld() {
echo 'World!';
}}
trait HelloWorld {
use Hello, World;
}
class MyHelloWorld {
use HelloWorld;
}
$o = new MyHelloWorld();
$o->sayHello();
$o->sayWorld();
//输出 Hello World!
Trait 的静态成员
与类相似,静态成员分静态方法和静态属性,二者都是以static关键字来进行声明的。
trait Counter {
public function inc() {
static $c = 0;
$c = $c + 1;
echo "$c\n";
}
}
trait StaticExample {
public static function doSomething() {
return 'Doing something';
}
}
Trait 属性
在trait中定义了属性后,类的实例可以像访问自身的属性一样访问它。而且可以给它设置访问权限,就差不能实例化了……
需要注意的是,如果 trait 定义了一个属性,那类将不能定义同样名称的属性,否则会产生一个错误。如果该属性在类中的定义与在 trait 中的定义兼容(同样的可见性和初始值)则错误的级别是 E_STRICT,否则是一个致命错误。
trait PropertiesTrait {
public $x = 1;
}
class PropertiesExample {
use PropertiesTrait;
}
$example = new PropertiesExample;
$example->x;
需要特别注意的几点
上面这些就是 Trait 基本的使用了,更详细的可以参k官方手册。
这里总结下注意的几 点:
- Trait 会覆盖调用类继承的父类方法
- Trait 无法如 Class 一样使用 new 实例化
- 单个 Trait 可由多个 Trait 组成
- 在单个 Class 中,可以使用多个 Trait
- Trait 支持修饰词(modifiers),例如 final、static、abstract
- 我们能使用 insteadof 以及 as 操作符解决 Trait 之间的冲突
参K文献
- http://php.net/manual/zh/language.oop5.traits.php
- https://www.doubear.com/article/raityongfaxuexibiji.html
- http://www.kuqin.com/web/20111119/315048.html
- http://www.jb51.net/article/61260.htm
最后
欢迎关注我的个人微信“华伟君”,与我一起学习探讨PHP,通往技术大牛的路上,我们结伴同行!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

JWT是一種基於JSON的開放標準,用於在各方之間安全地傳輸信息,主要用於身份驗證和信息交換。 1.JWT由Header、Payload和Signature三部分組成。 2.JWT的工作原理包括生成JWT、驗證JWT和解析Payload三個步驟。 3.在PHP中使用JWT進行身份驗證時,可以生成和驗證JWT,並在高級用法中包含用戶角色和權限信息。 4.常見錯誤包括簽名驗證失敗、令牌過期和Payload過大,調試技巧包括使用調試工具和日誌記錄。 5.性能優化和最佳實踐包括使用合適的簽名算法、合理設置有效期、

會話劫持可以通過以下步驟實現:1.獲取會話ID,2.使用會話ID,3.保持會話活躍。在PHP中防範會話劫持的方法包括:1.使用session_regenerate_id()函數重新生成會話ID,2.通過數據庫存儲會話數據,3.確保所有會話數據通過HTTPS傳輸。

SOLID原則在PHP開發中的應用包括:1.單一職責原則(SRP):每個類只負責一個功能。 2.開閉原則(OCP):通過擴展而非修改實現變化。 3.里氏替換原則(LSP):子類可替換基類而不影響程序正確性。 4.接口隔離原則(ISP):使用細粒度接口避免依賴不使用的方法。 5.依賴倒置原則(DIP):高低層次模塊都依賴於抽象,通過依賴注入實現。

在PHPStorm中如何進行CLI模式的調試?在使用PHPStorm進行開發時,有時我們需要在命令行界面(CLI)模式下調試PHP�...

PHP8.1中的枚舉功能通過定義命名常量增強了代碼的清晰度和類型安全性。 1)枚舉可以是整數、字符串或對象,提高了代碼可讀性和類型安全性。 2)枚舉基於類,支持面向對象特性,如遍歷和反射。 3)枚舉可用於比較和賦值,確保類型安全。 4)枚舉支持添加方法,實現複雜邏輯。 5)嚴格類型檢查和錯誤處理可避免常見錯誤。 6)枚舉減少魔法值,提升可維護性,但需注意性能優化。

如何在系統重啟後自動設置unixsocket的權限每次系統重啟後,我們都需要執行以下命令來修改unixsocket的權限:sudo...

靜態綁定(static::)在PHP中實現晚期靜態綁定(LSB),允許在靜態上下文中引用調用類而非定義類。 1)解析過程在運行時進行,2)在繼承關係中向上查找調用類,3)可能帶來性能開銷。
