分享很好的一篇文章,原文链接:https://github.com/krasimir/react-in-patterns/tree/master/patterns/dependency-injection
React in patterns / Dependency injection
Big part of the modules/components that we write have dependencies. A proper management of these dependencies is critical for the success of the project. There is a technique (some people consider it as a pattern) called dependency injection that helps solving the problem.
In React the need of dependency injector is easily visible. Let’s consider the following application tree:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28// Title.jsx
export default function Title(props) {
return <h1>{ props.title }</h1>;
}
// -----------------------------------
// Header.jsx
import Title from './Title.jsx';
export default function Header() {
return (
<header>
<Title />
</header>
);
}
// -----------------------------------
// App.jsx
import Header from './Header.jsx';
class App extends React.Component {
constructor(props) {
super(props);
this.state = { title: 'React in patterns' };
}
render() {
return <Header />;
}
};
The string “React in patterns” should somehow reach the Title
component. The direct way of doing this is to pass it from App
to Header
and then Header
to pass it to Title
. However, this may work for these three components but what happens if there are multiple properties and deeper nesting. Lots of components will have to mention properties that they are not interested in. It is clear that most React components receive their dependencies via props but the question is how these dependencies reach that point.
We already saw how the higher-order component may be used to inject data. Let’s use the same technique to inject the title
variable:
1 | // inject.jsx |
The title
is hidden in a middle layer (higher-order component) where we pass it as a prop to the original Title
component. That’s all nice but it solves only half of the problem. Now we don’t have to pass the title
down the tree but how this data will reach the enhance.jsx
helper.
Using React’s context
React has the concept of context. The context is something that every component may have access to. It’s something like an event bus but for data. A single model which we can access from everywhere.
1 | // a place where we'll define the context |
Notice that we have to specify the exact signature of the context object. With childContextTypes
and contextTypes
. If those are not specified then the context
object will be empty. That may be a little bit frustrating because we may have lots of stuff to put there. That’s why it is a good practice that our context
is not just a plain object but it has an interface that allows us to store and retrieve data. For example:
1 | // dependencies.js |
Then, if we go back to our example, the very top App
component may look like that:
1 | import dependencies from './dependencies'; |
And our Title
component gets it’s data through the context:
1 | // Title.jsx |
Ideally we don’t want to specify the contextTypes
every time when we need an access to the context. This detail may be wrapped in a higher-order component. And even more, we may write an utility function that is more descriptive and helps us declare the exact wiring. I.e instead of accessing the context directly with this.context.get('title')
we ask the higher-order component to get what we need and to pass it as a prop to our component. For example:
1 | // Title.jsx |
The wire
function accepts first a React component, then an array with all the needed dependencies (which are register
ed already) and then a function which I like to call mapper
. It receives what’s stored in the context as a raw data and returns an object which is the actual React props for our component (Title
). In this example we just pass what we get - a title
string variable. However, in a real app this could be a collection of data stores, configuration or something else. So, it’s nice that we pass exactly what we need and don’t pollute the components with data that they don’t need.
Here is how the wire
function looks like:
1 | export default function wire(Component, dependencies, mapper) { |
Inject
is a higher-order component that gets access to the context and retrieves all the items listed under dependencies
array. The mapper
is a function receiving the context
data and transforms it to props for our component.
At the end of this section we should mention that the usage of context
is not highly recommended by Facebook:
Context is an advanced and experimental feature. The API is likely to change in future releases.
Most applications will never need to use context. Especially if you are just getting started with React, you likely do not want to use context. Using context will make your code harder to understand because it makes the data flow less clear. It is similar to using global variables to pass state through your application.
If you have to use context, use it sparingly.
Regardless of whether you’re building an application or a library, try to isolate your use of context to a small area and avoid using the context API directly when possible so that it’s easier to upgrade when the API changes.
Using the module system
If we don’t want to use the context there are a couple of other ways to achieve the injection. They are not exactly React specific but worth mentioning. One of them is using the module system.
As we know the typical module system in JavaScript has a caching mechanism. It’s nicely noted in the Node’s documentation:
Modules are cached after the first time they are loaded. This means (among other things) that every call to require(‘foo’) will get exactly the same object returned, if it would resolve to the same file.
Multiple calls to require(‘foo’) may not cause the module code to be executed multiple times. This is an important feature. With it, “partially done” objects can be returned, thus allowing transitive dependencies to be loaded even when they would cause cycles.
How’s that helping for our injection? Well, if we export an object we are actually exporting a singleton and every other module that imports the file will get the same object. This allows us to register
our dependencies and later fetch
them in another file.
Let’s create a new file called di.jsx
with the following content:
1 | var dependencies = {}; |
We’ll store the dependencies in dependencies
global variable (it’s global for our module, not at an application level). We then export two functions register
and fetch
that write and read entries. It looks a little bit like implementing setter and getter against a simple JavaScript object. Then we have the wire
function that accepts our React component and returns a higher-order component. In the constructor of that component we are resolving the dependencies and later while rendering the original component we pass them as props. We follow the same pattern where we describe what we need (deps
argument) and extract the needed props with a mapper
function.
Having the di.jsx
helper we are again able to register our dependencies at the entry point of our application (app.jsx
) and inject them wherever (Title.jsx
) we need.
1 | // app.jsx |
If we look at the Title.jsx
file we’ll see that the actual component and the wiring may live in different files. That way the component and the mapper function become easily unit testable.
Injecting with the help of a build process
We are all processing our JavaScript before shipping it to the browser. This biggest benefit of having an intermediate process is the ability to add features which are normally not there. Like for example the support of ES6 destructuring with Babel or static type checking with Flow. There are tools for dependency injection too. InversifyJS is one of them and in the next section we will see how it works with React components.
Dependency injection powered by an IoC container
Not long ago an user in Twitter asked Michel Weststrate(the author of MobX) the following:
How safe is it to use mobx-react
? Or are there any other options for connecting stores to components without passing them explicitly through each component?
The answer was the following:
Dependency injection like InversifyJS also works nicely
InversifyJS is an IoC container. We can use an IoC container to inject a value into React components without passing it explicitly through each component and without using the context.
In this demonstration we are going to use InversifyJS and TypeScript. We are using InversifyJS because it works in both Node.js and web browsers. This is an important feature because some React applications use server-side rendering. We are also using TypeScript because it is the recommended by InversifyJS.
InversifyJS supports two kinds of injections:
- Constructor injection
- Property injection
In order to use “constructor injection” the IoC container needs to be able to create the instances of the classes. In React the components sometimes are just functions (not classes) and we can’t delegate the creation of the instances of the components to the IoC container. This means that constructor injection powered by IoC containers don’t play nicely with React
However, property injection works just fine considering the fact that we want to pass dependencies to components without passing them explicitly through each component.
Let’s take a look to a basic example.
We need to start by configuring the IoC container. In InversifyJs we need to create a dictionary that maps a type identifier with a type. The dictionary entries are known as “type bindings”.
In this case, we are binding the identifier UserStore
to the class UserStore
. This time the identifier is a Class but InversifyJS also allows us to use Symbols
or string literals as identifiers. Symbols or string literals are required when we use interfaces.
1 | import { Kernel, makePropertyInjectDecorator } from "inversify"; |
We also need to generate a decorator using the function makePropertyInjectDecorator
.
The generated pInject
decorator allows us to flag the properties of a class that we want to be injected:
1 | import { pInject } from "./utils/di"; |
Injected properties are lazy evaluated. This means that the value of the userStore
property is only set after we try to access it for the first time.
Based on the React docs we should try to avoid using context:
The main advantage of using an IoC container like InversifyJS is that we are not using the context!
InversifyJS is also great for testing because we can declare a new bindings and inject a mock or stub instead of a real value:
1 | kernel.bind<UserStore>(UserStore).toConstantValue({ pageTitle: "Some text for testing..." }); |
Find some real use cases of InversifyJS with React here and here or learn more about InversifyJS at the official repository here.
Final thoughts
Dependency injection is a tough problem. Especially in JavaScript. It’s not really an issue within React application but appears everywhere. At the time of this writing React offers only the context
as an instrument for resolving dependencies. As we mentioned above this technique should be used sparingly. And of course there are some alternatives. For example using the module system or libraries like InversifyJS.