__Callstatic

现在很多框架中调用方法都是Foo::bar()这种格式的,但是他们真的是静态方法吗?

这种情况在 larave 中尤其常见,但是开发过程中很明显这些有一部分不是静态的,比如你使用一个模型User,那么你每次实例化出来他都是一个全新的,互不影响,这里就用到了一个魔术方法__callStatic

举个栗子:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
<?php
class Test{
public function __call($name, $arguments)
{
echo 'this is __call'. PHP_EOL;
}
public static function __callStatic($name, $arguments)
{
echo 'this is __callStatic:'. PHP_EOL;
}
}
$test = new Test();
$test->hello();
$test::hi();
//this is __call:hello
//this is __callStatic:hi

当然魔术方法也是很耗性能的一种方式,每次调用的时候后回先扫一遍class没找到方法时才会调用它,而为了代码的整洁和抽象这个方法也能给很大的帮助,在这之间去要有个权衡

下面实现的 log 类,采用的就是这种方法,将方法解耦出来,只要符合规定的接口就能调用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
<?php
class Test{
//获取 logger 的实体
private static $logger;
public static function getLogger(){
return self::$logger?: self::$logger = self::createLogger();
}
private static function createLogger(){
return new Logger();
}
public static function setLogger(LoggerInterface $logger){
self::$logger = $logger;
}
public function __call($name, $arguments)
{
call_user_func_array([self::getLogger(),$name],$arguments);
}
public static function __callStatic($name, $arguments)
{
forward_static_call_array([self::getLogger(),$name],$arguments);
}
}
interface LoggerInterface{
function info($message,array $content = []);
function alert($messge,array $content = []);
}
class Logger implements LoggerInterface {
function info($message, array $content = [])
{
echo 'this is Log method info' . PHP_EOL;
var_dump($content);
}
function alert($messge, array $content = [])
{
echo 'this is Log method alert: '. $messge . PHP_EOL;
}
}
Test::info('喊个口号:',['好好','学习','天天','向上']);
$test = new Test();
$test->alert('hello');

输出:

1
2
3
4
5
6
7
8
9
10
11
12
this is Log method info
array(4) {
[0]=>
string(6) "好好"
[1]=>
string(6) "学习"
[2]=>
string(6) "天天"
[3]=>
string(6) "向上"
}
this is Log method alert: hello

也许有的小伙伴已经看出来了,没错!这段代码就是仿照(抄)的 EasyWechat的日志部分,代码片段

这个博客有个赞赏功能,不试一下吗?