PHP通过使用两个魔术方法来实现方法重载:__call和__callStatic。这两个方法在PHP文档中描述如下:
1 public mixed __call ( string $name , array $arguments )
2 public static mixed __callStatic ( string $name , array $arguments )
3
4 __call() is triggered when invoking inaccessible methods in an object context.
5
6 __callStatic() is triggered when invoking inaccessible methods in a static context.
__call(),当代码调用对象的某个不存在的方法时该魔术方法会被自动调用。__callStatic()方法则只用于静态类方法。当尝试调用类中不存在的静态方法时,__callStatic()魔术方法将被自动调用。对这个翻译简单的做下评论:放屁!评论完了。
看看官网的中文翻译:当调用一个不可访问方法(如未定义,或者不可见)时,__call() 会被调用。当在静态方法中调用一个不可访问方法(如未定义,或者不可见)时,__callStatic() 会被调用。
这个翻译还算准确,不过有点小漏洞:全局范围内用className::方式调用的时候,也会触发__callStatic。比较好的翻译:在对象上下文中调用不可访问方法,__call会触发。在静态上下文中调用不可访问方法,__callStatic会被触发。
实际上决定调用哪个魔术方法取决于上下文环境(context)而不是要调用什么。实际上一个方法不可见,就无法判断是否静态,我哪知道要调用什么?所以我对第一个翻译的评论是比较合理的。对于某人说继承类中::语义不明,实际上有继承关系类中::属于类的内部访问,这里::既不是静态也不表示非静态,而仅仅起到一个范围解析的作用(相关内容可阅读PHP文档)。具体要触发哪一个魔术方法,根据上下文环境来确定。
了解到两个魔术方法之后,你是否感觉这跟重载一点边都不沾?不要这么想,这只是和类C语言中的重载长得不像而已,本质是一样的。当触发某个魔术方法的时候,PHP会自动传递 $name , $arguments两个参数,第一个是函数名称,第二个是参数列表。如果两个函数名称相同而参数列表不同,可以将不同的参数类型解析出来进行不同的处理,完美的实现了类C语言重载的功能。具体怎么解析string和array我就不讲了。甚至还可以拿参数列表为基准,对不同的函数名称进行处理。或者进行参数与函数名的任意组合。可见PHP的重载功能完全的包含了类C语言的重载,并且实现更为强大而灵活的功能。