亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > PHP > 正文

PHP的學習 Traits新特性

2020-03-22 19:34:26
字體:
來源:轉載
供稿:網友
  • 在閱讀yii2源碼的時候接觸到了trait,就學習了一下,寫下博客記錄一下。

    自 PHP 5.4.0 起,PHP 實現了代碼復用的一個方法,稱為 traits。

    Traits 是一種為類似 PHP 的單繼承語言而準備的代碼復用機制。Trait 為了減少單繼承語言的限制,使開發人員能夠自由地在不同層次結構內獨立的類中復用方法集。Traits 和類組合的語義是定義了一種方式來減少復雜性,避免傳統多繼承和混入類(Mixin)相關的典型問題。

    Trait 和一個類相似,但僅僅旨在用細粒度和一致的方式來組合功能。Trait 不能通過它自身來實例化。它為傳統繼承增加了水平特性的組合;也就是說,應用類的成員不需要繼承。

    Trait 示例

    <?phptrait ezcReflectionReturnInfo {    function getReturnType() { /*1*/ }    function getReturnDescription() { /*2*/ }}html' target='_blank'>class ezcReflectionMethod extends ReflectionMethod {    use ezcReflectionReturnInfo;    /* ... */}class ezcReflectionFunction extends ReflectionFunction {    use ezcReflectionReturnInfo;    /* ... */}?>

    優先級

    從基類繼承的成員被 trait 插入的成員所覆蓋。優先順序是來自當前類的成員覆蓋了 trait 的方法,而 trait 則覆蓋了被繼承的方法。

    優先順序示例

    <?phpclass 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!

    從基類繼承的成員被插入的 SayWorld Trait 中的sayHello方法所覆蓋。其行為 MyHelloWorld 類中定義的方法一致。優先順序是當前類中的方法會覆蓋 trait 方法,而 trait 方法又覆蓋了基類中的方法。

    另一個優先級順序的例子

    <?phptrait HelloWorld {    public function sayHello() {        echo 'Hello World!';    }}class TheWorldIsNotEnough {    use HelloWorld;    public function sayHello() {        echo 'Hello Universe!';    }}$o = new TheWorldIsNotEnough();$o->sayHello();?>

    以上例程會輸出:Hello Universe!

    多個 trait

    通過逗號分隔,在 use 聲明列出多個 trait,可以都插入到一個類中。

    多個 trait 的用法的例子

    <?phptrait Hello {    public function sayHello() {        echo 'Hello ';    }}trait World {    public function sayWorld() {        echo 'World';    }}class MyHelloWorld {    use Hello, World;    public function sayExclamationMark() {        echo '!';    }}$o = new MyHelloWorld();$o->sayHello();$o->sayWorld();$o->sayExclamationMark();?>

    以上例程會輸出:Hello World!

    沖突的解決

    如果兩個 trait 都插入了一個同名的方法,如果沒有明確解決沖突將會產生一個致命錯誤。

    為了解決多個 trait 在同一個類中的命名沖突,需要使用insteadof操作符來明確指定使用沖突方法中的哪一個。

    以上方式僅允許排除掉其它方法,as操作符可以將其中一個沖突的方法以另一個名稱來引入。

    沖突解決的例子

    <?phptrait A {    public function smallTalk() {        echo 'a';    }    public function bigTalk() {        echo 'A';    }}trait B {    public function smallTalk() {        echo 'b';    }    public function bigTalk() {        echo 'B';    }}class Talker {    use A, B {        B::smallTalk insteadof A;        A::bigTalk insteadof B;    }}class Aliased_Talker {    use A, B {        B::smallTalk insteadof A;        A::bigTalk insteadof B;        B::bigTalk as talk;    }}?>

    在本例中 Talker 使用了 trait A 和 B。由于 A 和 B 有沖突的方法,其定義了使用 trait B 中的 smallTalk 以及 trait A 中的 bigTalk。

    Aliased_Talker 使用了as操作符來定義了talk來作為 B 的 bigTalk 的別名。

    修改方法的訪問控制

    使用as語法還可以用來調整方法的訪問控制。

    修改方法的訪問控制的例子

    <?phptrait 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。在 trait 定義時通過使用一個或多個 trait,它能夠組合其它 trait 中的部分或全部成員。

    從 trait 來組成 trait的例子

    <?phptrait 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 的抽象成員

    為了對使用的類施加強制要求,trait 支持抽象方法的使用。

    表示通過抽象方法來進行強制要求的例子

    <?phptrait Hello {    public function sayHelloWorld() {        echo 'Hello'.$this->getWorld();    }    abstract public function getWorld();}class MyHelloWorld {    private $world;    use Hello;    public function getWorld() {        return $this->world;    }    public function setWorld($val) {        $this->world = $val;    }}?>

    Trait 的靜態成員

    Traits 可以被靜態成員靜態方法定義。

    靜態變量的例子

    <?phptrait Counter {    public function inc() {        static $c = 0;        $c = $c + 1;        echo "$c/n";    }}class C1 {    use Counter;}class C2 {    use Counter;}$o = new C1(); $o->inc(); // echo 1$p = new C2(); $p->inc(); // echo 1?>

    靜態方法的例子

    <?phptrait StaticExample {    public static function doSomething() {        return 'Doing something';    }}class Example {    use StaticExample;}Example::doSomething();?>

    靜態變量和靜態方法的例子

    <?phptrait Counter {    public static $c = 0;    public static function inc() {        self::$c = self::$c + 1;        echo self::$c . "/n";    }}class C1 {    use Counter;}class C2 {    use Counter;}C1::inc(); // echo 1C2::inc(); // echo 1?>

    屬性

    Trait 同樣可以定義屬性。

    定義屬性的例子

    <?phptrait PropertiesTrait {    public $x = 1;}class PropertiesExample {    use PropertiesTrait;}$example = new PropertiesExample;$example->x;?>

    如果 trait 定義了一個屬性,那類將不能定義同樣名稱的屬性,否則會產生一個錯誤。如果該屬性在類中的定義與在 trait 中的定義兼容(同樣的可見性和初始值)則錯誤的級別是E_STRICT,否則是一個致命錯誤。

    沖突的例子

    <?phptrait PropertiesTrait {    public $same = true;    public $different = false;}class PropertiesExample {    use PropertiesTrait;    public $same = true; // Strict Standards    public $different = true; // 致命錯誤}?>
    Use的不同

    不同use的例子

    <?phpnamespace Foo/Bar;use Foo/Test;  // means /Foo/Test - the initial / is optional?><?phpnamespace Foo/Bar;class SomeClass {    use Foo/Test;   // means /Foo/Bar/Foo/Test}?>

    第一個use是用于 namespace 的 use Foo/Test,找到的是/Foo/Test,第二個 use 是使用一個trait,找到的是/Foo/Bar/Foo/Test。

    __CLASS__和__TRAIT__

    __CLASS__ 返回 use trait 的 class name,__TRAIT__返回 trait name

    示例如下

    <?phptrait TestTrait {    public function testMethod() {        echo "Class: " . __CLASS__ . PHP_EOL;        echo "Trait: " . __TRAIT__ . PHP_EOL;    }}class BaseClass {    use TestTrait;}class TestClass extends BaseClass {}$t = new TestClass();$t->testMethod();//Class: BaseClass//Trait: TestTrait
    Trait單例

    實例如下

    <?phptrait singleton {        /**     * private construct, generally defined by using class     */    //private function __construct() {}        public static function getInstance() {        static $_instance = NULL;        $class = __CLASS__;        return $_instance ?: $_instance = new $class;    }        public function __clone() {        trigger_error('Cloning '.__CLASS__.' is not allowed.',E_USER_ERROR);    }        public function __wakeup() {        trigger_error('Unserializing '.__CLASS__.' is not allowed.',E_USER_ERROR);    }}/*** Example Usage*/class foo {    use singleton;        private function __construct() {        $this->name = 'foo';    }}class bar {    use singleton;        private function __construct() {        $this->name = 'bar';    }}$foo = foo::getInstance();echo $foo->name;$bar = bar::getInstance();echo $bar->name;
    調用trait方法

    雖然不很明顯,但是如果Trait的方法可以被定義為在普通類的靜態方法,就可以被調用

    實例如下

    <?php trait Foo {     function bar() {         return 'baz';     } } echo Foo::bar(),"http://n"; ?>
    PHP編程

    鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。

  • 發表評論 共有條評論
    用戶名: 密碼:
    驗證碼: 匿名發表
    亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
    精品久久久国产| 2019中文在线观看| 成人免费xxxxx在线观看| 精品国产成人在线| 欧美日韩不卡合集视频| 午夜精品在线视频| 国产日韩综合一区二区性色av| 欧美高清视频一区二区| 国产成人jvid在线播放| 88国产精品欧美一区二区三区| 亚洲a一级视频| www高清在线视频日韩欧美| 伊人久久久久久久久久久| 欧美一区二三区| 欧美日韩在线视频观看| 在线观看国产精品91| 日韩中文字幕在线免费观看| 国产精品你懂得| 欧美男插女视频| 一区三区二区视频| 国产精品一区久久久| 欧美性猛交xxxx黑人猛交| 亚洲精品久久7777777| 中文字幕在线看视频国产欧美| 久久99精品久久久久久琪琪| 国产精品偷伦免费视频观看的| 日韩亚洲国产中文字幕| 久久久天堂国产精品女人| 久久综合久久八八| 精品偷拍各种wc美女嘘嘘| 欧美精品激情在线观看| 91精品国产综合久久香蕉| 在线亚洲男人天堂| 国产日韩精品在线观看| 欧美乱妇高清无乱码| 在线电影欧美日韩一区二区私密| 亚洲第一中文字幕在线观看| 欧美久久精品一级黑人c片| 狠狠色狠狠色综合日日小说| 91久久久久久国产精品| 亚洲黄色在线看| 亚洲精品狠狠操| 精品久久久免费| 国产成人一区二区三区小说| 国产精品久久久久久久久免费看| 国产日韩精品一区二区| 亚洲天堂一区二区三区| 午夜伦理精品一区| 欧美诱惑福利视频| 欧洲美女7788成人免费视频| 欧美一级视频在线观看| 久久亚洲精品小早川怜子66| 91精品久久久久久久久| 久久久久久久久久久久久久久久久久av| 日本国产高清不卡| 中文一区二区视频| 国产精品99久久久久久人| 国产精品2018| 欧美高清不卡在线| 国产精品入口福利| 日本一区二三区好的精华液| 91免费综合在线| 国产美女精品视频| 久久精品色欧美aⅴ一区二区| 国产精品777| 精品久久久国产精品999| 午夜精品www| 日韩精品视频在线免费观看| 国产亚洲精品美女久久久久| 中文字幕亚洲专区| 国产精品免费一区| 久久好看免费视频| 国产精品久久久久久搜索| 日韩欧美综合在线视频| 91夜夜未满十八勿入爽爽影院| 4438全国亚洲精品在线观看视频| 欧美二区乱c黑人| 国产精品成人av在线| 欧美大胆a视频| 日本一区二三区好的精华液| 在线观看亚洲区| 日韩免费精品视频| 日韩高清电影好看的电视剧电影| 欧美激情精品久久久| 亚洲黄一区二区| 国产精品av免费在线观看| 亚洲欧美综合另类中字| 97在线精品国自产拍中文| 国产亚洲精品美女久久久| 亚洲成人在线视频播放| 欧美美女15p| 国产精品久久久久久久7电影| 国产精品一区专区欧美日韩| 久久天天躁狠狠躁老女人| 久久天天躁狠狠躁老女人| 国产精品久久久久国产a级| 精品国产乱码久久久久久婷婷| 欧美专区在线播放| 亚洲在线观看视频网站| 精品视频在线播放| 亚洲a一级视频| 不卡在线观看电视剧完整版| 欧美日韩国内自拍| 亚洲国产精品久久久久秋霞不卡| 欧美成人h版在线观看| 国产激情999| 91精品国产免费久久久久久| 51久久精品夜色国产麻豆| 最近的2019中文字幕免费一页| 日韩欧美黄色动漫| 日本久久久久久| 亚洲人成自拍网站| 亚洲aa中文字幕| 国产精品一区av| 欧美大尺度激情区在线播放| 色综合伊人色综合网| 成人午夜黄色影院| 九九综合九九综合| 色综合伊人色综合网站| 91老司机精品视频| 一区二区三区视频免费在线观看| 九九热精品视频在线播放| 亚洲女人被黑人巨大进入al| 久久电影一区二区| 欧美日韩成人在线观看| 日韩av一区二区在线观看| 538国产精品一区二区在线| 国产69精品久久久久9| 国产亚洲精品美女| 精品中文字幕在线| 欧美电影在线观看网站| 日韩在线观看免费av| 亚洲a级在线观看| 久久黄色av网站| 97久久精品人人澡人人爽缅北| 亚洲美女在线视频| 久久久久国产精品免费网站| 欧美性猛交xxxx久久久| 操91在线视频| 国产精品手机播放| 国产丝袜一区二区三区免费视频| 亚洲最大的网站| 91免费观看网站| 国产精品久久久久久婷婷天堂| 精品国产一区二区在线| 97在线观看视频国产| 日本欧美中文字幕| 日韩电影免费在线观看| 欧美中文字幕第一页| 国产91久久婷婷一区二区| 国产成人精品在线视频| 亚洲精品成人免费| 日韩av三级在线观看| 国产成人短视频| 欧美影院成年免费版| 91av在线免费观看视频| 欧美猛交免费看| 亚洲成人激情在线观看| 亚洲欧美日韩国产中文专区| 亚洲国产精品久久久久秋霞蜜臀| 欧美另类精品xxxx孕妇| 亚洲人成在线一二| 日韩专区中文字幕| 2023亚洲男人天堂|