在PHP final class最终类中,组成大于继承浅析代码示例

2023-06-01 00:00:00 示例 大于 浅析

final class最终类,你要么喜欢它们,要么讨厌它们。

最近人们在他们的开源包中更多地使用它们,但这对你意味着什么?你如何在应用代码中处理这个问题?


开发人员多年来一直在讨论组成而不是继承,而最终类是一个完美的用例。

我最近和别人谈的一个例子是'moneyphp/money',它实现了最终类。

让我们深入了解一下。

让我们以'moneyphp/money'为例,看看我们可能如何与之整合。

在最终类之前,我们会扩展Money类并按我的需要使用它。

然而,这已经不可能了,所以我们要找到一种方法来与之合作。

我们将创建一个名为MoneyImplementation的类,它将是我们想要使用的类。

class MoneyImplementation
{
    private Money $money;
 
    public function __construct(
        int|string $amount,
        Currency $currency,
    ) {
        $this->money = new Money(
            amount: $amount,
            currency: $currency,
        );
    }
}

所以在上面的代码中,我们已经用组合法建立了一个类,

这个类将把它的构造代理给money类--把类上的一个属性设置为构造的实例。

接下来的问题是,我们如何在不扩展我们想要调用的API的情况下调用货币类中的方法。

如果不添加这个,我们将不得不添加一个访问器来从我们的类中获取钱的实例,

然后像下面这样使用访问器。

$money = new MoneyImplementation(
    amount: 10_000,
    currency: new Currency(
        code: 'USD',
    ),
);
 
$money->money()->getAmount();


这并不理想,但如果没有任何额外的工作,这也是可以接受的。

但是我们可以用一个小小的PHP魔法使它更进一步。

让我们把这个神奇的方法添加到我们的MoneyImplementation类中。

class MoneyImplementation
{
    private Money $money;
 
    public function __construct(
        int|string $amount,
        Currency $currency,
    ) {
        $this->money = new Money(
            amount: $amount,
            currency: $currency,
        );
    }
 
    public function __call(string $name, array $arguments)
    {
        if (! method_exists($this->money, $name)) {
            throw new RuntimeException(
                message: "Method [$name] does not exist.",
            );
        }
 
        return $this->money->{$name}(...$arguments);
    }
}


我们给我们的类添加了一个方法__call,这样我们就可以直接代理调用我们已经构建好的货币实例。

然而,在这之前,我们可以添加一个安全检查,以确保该方法的存在。

利用这一点,我们现在可以简化API了。

$money = new MoneyImplementation(
    amount: 10_000,
    currency: new Currency(
        code: 'USD',
    ),
);
 
$money->getAmount();


这可能不是说明问题的最好例子,因为我们没有给我们的类添加任何东西。

然而,它说明了我们如何使用组合而不是继承来解决最终包类的问题。

相关文章