检查实例的类是否实现了接口?

Wil*_*lco 142 php oop interface

给定一个类实例,是否可以确定它是否实现了特定的接口?据我所知,没有内置函数可以直接执行此操作.我有什么选择(如果有的话)?

小智 245

interface IInterface
{
}

class TheClass implements IInterface
{
}

$cls = new TheClass();
if ($cls instanceof IInterface) {
    echo "yes";
}
Run Code Online (Sandbox Code Playgroud)

您可以使用"instanceof"运算符.要使用它,左操作数是一个类实例,右操作数是一个接口.如果对象实现特定接口,则返回true.


Jes*_*ord 96

正如wherehere指出的那样,你可以使用class_implements().与Reflection一样,这允许您将类名指定为字符串,并且不需要该类的实例:

interface IInterface
{
}

class TheClass implements IInterface
{
}

$interfaces = class_implements('TheClass');

if (isset($interfaces['IInterface'])) {
    echo "Yes!";
}
Run Code Online (Sandbox Code Playgroud)

class_implements() 是SPL扩展的一部分.

请参阅:http://php.net/manual/en/function.class-implements.php

性能测试

一些简单的性能测试显示了每种方法的成本:

给定一个对象的实例

Object construction outside the loop (100,000 iterations)
 ____________________________________________
| class_implements | Reflection | instanceOf |
|------------------|------------|------------|
| 140 ms           | 290 ms     | 35 ms      |
'--------------------------------------------'

Object construction inside the loop (100,000 iterations)
 ____________________________________________
| class_implements | Reflection | instanceOf |
|------------------|------------|------------|
| 182 ms           | 340 ms     | 83 ms      | Cheap Constructor
| 431 ms           | 607 ms     | 338 ms     | Expensive Constructor
'--------------------------------------------'

只给出一个类名

100,000 iterations
 ____________________________________________
| class_implements | Reflection | instanceOf |
|------------------|------------|------------|
| 149 ms           | 295 ms     | N/A        |
'--------------------------------------------'

昂贵的__construct()在哪里:

public function __construct() {
    $tmp = array(
        'foo' => 'bar',
        'this' => 'that'
    );  

    $in = in_array('those', $tmp);
}
Run Code Online (Sandbox Code Playgroud)

这些测试基于这个简单的代码.

  • 感谢您的性能测试.很有帮助. (12认同)

Bil*_*win 55

nlaq指出,instanceof可以用来测试对象是否是实现接口的类的实例.

instanceof不区分类类型和接口.你不知道,如果对象是一个恰好是叫IInterface.

您还可以使用PHP中的反射API来更具体地测试它:

$class = new ReflectionClass('TheClass');
if ($class->implementsInterface('IInterface'))
{
  print "Yep!\n";
}
Run Code Online (Sandbox Code Playgroud)

http://php.net/manual/en/book.reflection.php

  • 另见[`class_implements()`](http://php.net/manual/en/function.class-implements.php) (6认同)
  • 这可以用于"静态"类 (2认同)
  • 如果使用名称空间,那么接口和具有相同名称的类之间不会存在歧义,您可以再次安全地使用`instanceof`. (2认同)

d.r*_*aev 18

只是为了帮助将来搜索is_subclass_of也是一个很好的变种(对于PHP 5.3.7+):

if (is_subclass_of($my_class_instance, 'ISomeInterfaceName')){
    echo 'I can do it!';
}
Run Code Online (Sandbox Code Playgroud)


Pil*_*lan 8

更新

此处缺少该is_a 功能作为替代。

我做了一些性能测试来检查哪种方式是性能最好的。

超过 10 万次迭代的结果

      instanceof [object] took   7.67 ms | +  0% | ..........
            is_a [object] took  12.30 ms | + 60% | ................
             is_a [class] took  17.43 ms | +127% | ......................
class_implements [object] took  28.37 ms | +270% | ....................................
       reflection [class] took  34.17 ms | +346% | ............................................
Run Code Online (Sandbox Code Playgroud)

添加了一些点以实际“感觉”看到差异。

由此生成:https : //3v4l.org/8Cog7

结论

如果您有要检查的对象,请使用instance of已接受答案中提到的方法。

如果您有要检查的课程,请使用is_a.

奖金

考虑到您想要基于您需要的接口实例化一个类的情况,使用is_a. 只有一种例外——构造函数为空时。

例子: is_a(<className>, <interfaceName>, true);

它将返回bool。第三个参数“ allow_string ”允许它在实例化类的情况下检查类名。


Sta*_*arx 5

您还可以执行以下操作

public function yourMethod(YourInterface $objectSupposedToBeImplementing) {
   //.....
}
Run Code Online (Sandbox Code Playgroud)

如果$objectSupposedToBeImplementing没有实现YourInterface接口,它将抛出可恢复的错误.