traits学习
越来越多的框架和代码开始使用traits方式去组织一些功能,这是非常高效的代码组织结构。
通过trait来减少不必要的类继承关系,让代码更加复用,形成可以拔插的代码集合。
通过逗号分隔,在 use 声明列出多个 trait,可以都插入到一个类中。
单个的例子:
<?php
trait Seller
{
public function sell()
{
print_r("sell all goods");
}
}
class MySeller
{
use Seller;
}
$seller = new MySeller();
$seller->sell();
多个trait的例子:
<?php
trait Money
{
public function show()
{
echo "200$";
}
}
trait Factory
{
public function run()
{
print_r("running all");
}
}
class User
{
use Money, Factory;
public function show()
{
echo "I am an user";
}
}
$user = new User();
$user->show(); // => I am an user
当引入trait的类也有同名方法或者函数,那么当前类的该方法会覆盖trait中的同名方法。
如上面这个例子中User类和Money trait都有show方法,则执行之后调用的是User类的show方法,打印出"I am an user"的内容。
但是如果两个 trait 都插入了一个同名的方法,如果没有明确解决冲突将会产生一个致命错误。我们需要明确指定是否某个trait的方法或者尽量不要让不同trait里面存在同名方法。
1.使用 insteadof 操作符来明确指定使用冲突方法中的哪一个。例如:
<?php
trait Token
{
public function toString($var)
{
return (string)$var . "_token";
}
public function tips()
{
print_r("Token tips!");
}
}
trait Tester
{
public function toString($var)
{
return (string)$var . "_test";
}
public function tips()
{
print_r("Tester tips");
}
}
class Controller
{
use Token, Tester {
Token::toString insteadof Tester;
Tester::tips insteadof Token;
}
}
$con = new Controller();
$con->toString(‘hello‘);
$con->tips(‘my god!‘);
或者使用别名方式去避免同名问题,如:
class Controller
{
user Token, Tester {
Token::toString as toTokenString;
Tester::tips as testerTips;
}
}
以上都是一些理论案例,实际上一些优质和现代的php框架已经在使用trait方式组织整个代码。典型的就是laravel。比如/www/larblog/app/Http/Controllers/Controller.php:
<?php
namespace App\Http\Controllers;
use Illuminate\Foundation\Bus\DispatchesJobs;
use Illuminate\Routing\Controller as BaseController;
use Illuminate\Foundation\Validation\ValidatesRequests;
use Illuminate\Foundation\Auth\Access\AuthorizesRequests;
use Illuminate\Foundation\Auth\Access\AuthorizesResources;
class Controller extends BaseController
{
use AuthorizesRequests, AuthorizesResources, DispatchesJobs, ValidatesRequests;
}
这里就使用了四个trait.