Lazy Services

See also

こちらもご覧ください

Another way to inject services lazily is via a service subscriber.

サービスを遅延して注入するもう 1 つの方法は、サービス サブスクライバーを使用することです。

Why Lazy Services?

In some cases, you may want to inject a service that is a bit heavy to instantiate, but is not always used inside your object. For example, imagine you have a NewsletterManager and you inject a mailer service into it. Only a few methods on your NewsletterManager actually use the mailer, but even when you don't need it, a mailer service is always instantiated in order to construct your NewsletterManager.

場合によっては、インスタンス化するのが少し重いサービスを注入したいかもしれませんが、オブジェクト内で常に使用されるとは限りません。たとえば、NewsletterManager があり、それにメーラー サービスを挿入するとします。実際にメーラーを使用する NewsletterManager のメソッドはごくわずかですが、必要がない場合でも、NewsletterManager を構築するためにメーラー サービスが常にインスタンス化されます。

Configuring lazy services is one answer to this. With a lazy service, a "proxy" of the mailer service is actually injected. It looks and acts like the mailer, except that the mailer isn't actually instantiated until you interact with the proxy in some way.

遅延サービスを構成することは、これに対する 1 つの答えです。レイジー サービスでは、メーラー サービスの「プロキシ」が実際に注入されます。何らかの方法でプロキシと対話するまで、メーラーは実際にはインスタンス化されないことを除いて、メーラーのように見え、機能します。

Caution

注意

Lazy services do not support final classes, but you can use Interface Proxifying to work around this limitation.

Lazy サービスは final クラスをサポートしていませんが、Interface Proxifying を使用してこの制限を回避できます。

In PHP versions prior to 8.0 lazy services do not support parameters with default values for built-in PHP classes (e.g. PDO).

8.0 より前の PHP バージョンでは、遅延サービスは組み込み PHP クラス (PDO など) のデフォルト値を持つパラメータをサポートしていません。

6.2

6.2

Starting from Symfony 6.2, you don't have to install any package (e.g. symfony/proxy-manager-bridge) in order to use the lazy service instantiation.

Symfony 6.2 から、遅延サービスのインスタンス化を使用するためにパッケージ (symfony/proxy-manager-bridge など) をインストールする必要がなくなりました。

Configuration

You can mark the service as lazy by manipulating its definition:

サービスの定義を操作することで、サービスを遅延としてマークできます。
  • YAML
    YAML
  • XML
    XML
  • PHP
    PHP
1
2
3
4
# config/services.yaml
services:
    App\Twig\AppExtension:
        lazy: true

Once you inject the service into another service, a lazy ghost object with the same signature of the class representing the service should be injected. A lazy ghost object is an object that is created empty and that is able to initialize itself when being accessed for the first time). The same happens when calling Container::get() directly.

サービスを別のサービスに注入したら、サービスを表すクラスと同じシグネチャを持つレイジー ゴースト オブジェクトを注入する必要があります。 lazyghost オブジェクトは、空で作成され、最初のアクセス時に自身を初期化できるオブジェクトです)。 Container::get() を直接呼び出す場合も同様です。

To check if your lazy service works you can check the interface of the received object:

遅延サービスが機能するかどうかを確認するには、受信したオブジェクトのインターフェイスを確認します。
1
2
dump(class_implements($service));
// the output should include "Symfony\Component\VarExporter\LazyGhostObjectInterface"

Interface Proxifying

Under the hood, proxies generated to lazily load services inherit from the class used by the service. However, sometimes this is not possible at all (e.g. because the class is final and can not be extended) or not convenient.

内部的には、サービスを遅延ロードするために生成されたプロキシは、サービスによって使用されるクラスから継承されます。ただし、これがまったく不可能な場合 (たとえば、クラスが final であり、拡張できないため)、または便利でない場合があります。

To workaround this limitation, you can configure a proxy to only implement specific interfaces.

この制限を回避するには、特定のインターフェイスのみを実装するようにプロキシを構成できます。
  • YAML
    YAML
  • XML
    XML
  • PHP
    PHP
1
2
3
4
5
6
7
8
# config/services.yaml
services:
    App\Twig\AppExtension:
        lazy: 'Twig\Extension\ExtensionInterface'
        # or a complete definition:
        lazy: true
        tags:
            - { name: 'proxy', interface: 'Twig\Extension\ExtensionInterface' }

The virtual proxy injected into other services will only implement the specified interfaces and will not extend the original service class, allowing to lazy load services using final classes. You can configure the proxy to implement multiple interfaces by adding new "proxy" tags.

他のサービスに挿入された仮想プロキシは、指定されたインターフェイスのみを実装し、元のサービス クラスを拡張しないため、最終クラスを使用してサービスを遅延ロードできます。新しい「proxy」タグを追加することで、複数のインターフェースを実装するようにプロキシを構成できます。

Tip

ヒント

This feature can also act as a safe guard: given that the proxy does not extend the original class, only the methods defined by the interface can be called, preventing to call implementation specific methods. It also prevents injecting the dependency at all if you type-hinted a concrete implementation instead of the interface.

この機能は安全ガードとしても機能します。プロキシが元のクラスを拡張しない場合、インターフェイスによって定義されたメソッドのみを呼び出すことができ、実装固有のメソッドを呼び出すことができなくなります。また、インターフェイスの代わりに具体的な実装をタイプヒントした場合、依存関係の注入をまったく防ぎます。