Can I use custom ReflectiveInjector strategies to get component providers?

前端 未结 1 1605
萌比男神i
萌比男神i 2020-12-22 06:55

As you all know we have different strategies for providers: useClass, useExisting, useFactory, useValue. But what if I wo

1条回答
  •  礼貌的吻别
    2020-12-22 07:15

    Under the hood Angular doesn't use ReflectiveInjector to retrieve component providers so even if you manage to extend ReflectiveInjector it will have no effect on the component providers. You can see it here:

     function resolveDep(...) {
           ...
            default:
              const providerDef =
                  (allowPrivateServices ? elDef.element !.allProviders :
                                          elDef.element !.publicProviders) ![tokenKey];
              if (providerDef) {
                const providerData = asProviderData(view, providerDef.index);
                                     ^^^^^^^^^^^^^^^
                if (providerData.instance === NOT_CREATED) {
                  providerData.instance = _createProviderInstance(view, providerDef);
                }
                return providerData.instance;
              }
    

    The method is called when component requests a dependency, for example ViewContainerRef:

    class MyComponent {
       constructor(vc: ViewContainerRef)
    

    And this line:

    const providerData = asProviderData(view, providerDef.index);
    

    shows that dependency is retrieved from the view node, not a reflective injector. So when you do like this:

    constructor(i: Injector) {
       console.log(i instanceOf ReflectiveInjector); // false
    }
    

    you will see that it's not real. It's just a wrapper around resolveDep function which encloses over the view and relevant view node.

    Reflective injector is still used for the host view injectors. This is the injector that you pass when you instantiate a component dynamically:

    componentFactory.create(hostViewInjector)
    

    Here is the relevant code:

    const value = startView.root.injector.get(depDef.token, NOT_FOUND_CHECK_ONLY_ELEMENT_INJECTOR);
    

    And also the module injector is consulted if you the dependency cannot be resolved on the component or the host view injector.
    Here is the relevant code:

    return startView.root.ngModule.injector.get(depDef.token, notFoundValue);
    

    0 讨论(0)
提交回复
热议问题