前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >如何在 React 组件中优雅的实现依赖注入

如何在 React 组件中优雅的实现依赖注入

作者头像
ConardLi
发布2021-07-16 11:28:07
5.3K0
发布2021-07-16 11:28:07
举报
文章被收录于专栏:code秘密花园code秘密花园

控制反转(Inversion of Control,缩写为IoC),是面向对象编程中的一种设计原则,可以用来减低计算机代码之间的耦合度,其中最常见的方式就是依赖注入(Dependency Injection,简称DI)。

通过控制反转,对象在被创建的时候,由一个调控系统内所有对象的外界实体将其所依赖的对象的引用传递给它。也可以说,依赖被注入到对象中。

一般这个概念在 Java 中提的比较多,但是在前端领域,似乎很少会提到这个概念,其实用好这个思想无论在前后端一样可以帮助我们的组件解耦,本文将介绍一下依赖注入在 React 中的应用。

为啥需要依赖注入?

依赖注入(更广泛地说就是控制反转)主要用来解决下面几个问题:

  • 模块解耦 - 在代码设计中应用,强制保持代码模块分离。
  • 更好的可复用性 - 让模块复用更加容易。
  • 更好的可测试性 - 通过注入模拟依赖可以更方便测试。

其实, React 本身也内置了对依赖注入的支持。

React 中的依赖注入

下面几个常见的代码,其实都应用了依赖注入的思想,我们来看几个例子:

  1. 使用 props 允许依赖注入
代码语言:javascript
复制
function welcome(props) {
  return <h1> Hello, {props.name}</h1>;
}

welcome 组件通过接收 props 然后生成 html,别惊讶,我们最常用的 props 其实就是应用了依赖注入的思想。

  1. 使用 context 是实现依赖注入的另一种方法
代码语言:javascript
复制
function counter() {
  const { message } = useContext(MessageContext);
  return <p>{ message }</p>;
}

由于 context 是沿着组件树向下传递的,我们可以使用组件内部的 hooks 来提取到它。

  1. 只使用 jsx 也能实现依赖注入
代码语言:javascript
复制
const ReviewList = props => ( 
  <List resource="/reviews" perPage={50} {...props}> 
    <Datagrid rowClick="edit"> 
      <DateField source="date" /> 
      <CustomerField source="customer_id " /> 
      <ProductField source="product_id" /> 
      <StatusField source="status" /> 
    </Datagrid> 
  </List> 
);

perPage 参数被传递给 <List>组件,然后组件通过 REST API 获取远程数据。

但是,<List> 组件并不会直接渲染数据,相反,它把渲染数据的重任交给了子组件 <Datagrid><Datagrid> 组件的渲染依赖于 <List><List> 是设置这种依赖关系的调用者。

但是,这些策略可能对小型项目有所帮助。在一些大型项目中往往我们需要更灵活的扩展,除了这些基础的应用之外,我们还需要更好地支持依赖注入。

我们来看几个扩展 React 依赖注入支持的库。

InversifyJS

InversifyJS 是一个强大、轻量的依赖注入库,并且使用非常简单,但是把它和 React 组件结合使用还是有些问题。

因为 InversifyJS 默认使用构造函数注入,但是 React 不允许开发者扩展组件的构造函数。我们通过一个例子来看看如何解决这个问题:

代码语言:javascript
复制
import "reflect-metadata";
import * as React from "react";
import { render } from "react-dom";
import { Hello } from "./Hello";

const App = () => (
  <div>
    <Hello />
  </div>
);

render(<App />, document.getElementById("root"));

通过 InversifyJS 提供的 injectable decorator 可以标记这个 class 是可被注入的。

代码语言:javascript
复制
import { injectable } from "inversify";

export interface IProvider<T> {
  provide(): T;
}

@injectable()
export class NameProvider implements IProvider<string> {
  provide() {
    return "World";
  }
}

在组件中,我们可以直接调用注入的 provide 方法,而组件内部不用关心它的实现。

代码语言:javascript
复制
import * as React from "react";
import { IProvider } from "./providers";

export class Hello extends React.Component {
  private readonly nameProvider: IProvider<string>;

  render() {
    return <h1>Hello {this.nameProvider.provide()}!</h1>;
  }
}

这就是一个最简单的依赖注入,下面我们再来看看几个 InversifyJS 的扩展库。

inversify-inject-decorators

该工具库主要提供了 lazyInject 之类的方法,它可以给出了一个惰性的注入,意思是在对象初始化时不需要提供依赖,当我们没办法改构造函数时,这个库就派上用场啦。

另外,除了字面上所说的惰性,另外一个非常重要的功能就是允许你将 inversifyJs 集成到任何自己控制类实例创建的库或者框架,比如 React

下面是一个 @lazyInject 的使用示例:

代码语言:javascript
复制
import getDecorators from "inversify-inject-decorators";
import { Container, injectable, tagged, named } from "inversify";
 
let container = new Container();
let { lazyInject } = getDecorators(container);
let TYPES = { Weapon: "Weapon" };
 
interface Weapon {
    name: string;
    durability: number;
    use(): void;
}
 
@injectable()
class Sword implements Weapon {
    public name: string;
    public durability: number;
    public constructor() {
        this.durability = 100;
        this.name = "Sword";
    }
    public use() {
        this.durability = this.durability - 10;
    }
}
 
class Warrior {
    @lazyInject(TYPES.Weapon)
    public weapon: Weapon;
}
 
container.bind<Weapon>(TYPES.Weapon).to(Sword);
 
let warrior = new Warrior();
console.log(warrior.weapon instanceof Sword); // true

inversify-react

inversify-react 是一个唯一执行依赖注入的库。就像使用 React Context.Provider一样,我们从这个库也能拿到一个 Provider

代码语言:javascript
复制
import { Provider } from 'inversify-react';
...

<Provider container={myContainer}>
    ...
</Provider>

然后我们就能在子组件中使用依赖了:

代码语言:javascript
复制
import { resolve, useInjection } from 'inversify-react';
...

// In functional component – via hooks
const ChildComponent: React.FC = () => {
    const foo = useInjection(Foo);
    ...
};

// or in class component – via decorated fields
class ChildComponent extends React.Component {
    @resolve
    private readonly foo: Foo;
    ...
}

react-inversify

虽然和上一个库名字很像,但是两个库的做法是不一样的,这种方法更接近于 React 的思想,因为对象是作为属性传递的,而不是在组件内部实例化。

代码语言:javascript
复制
import * as React from 'react';
import * as inversify from 'inversify';
import { Todos } from "./model";
import { connect } from 'react-inversify';
 
class Dependencies {
    constructor(todos) {
        this.todos = todos;
    }
}
 
inversify.decorate(inversify.injectable(), Dependencies);
inversify.decorate(inversify.inject(Todos.TypeTag), Dependencies, 0);
 
class TodoItemView extends React.Component {
    // ... use this.props.checked,  this.props.text, etc. All these calculated by code below.
}
 ssed as React properties.
// Mapping function returns final TodoItemView's properties.
export default connect(Dependencies, (deps, ownProps) => ({
    checked: ownProps.item.isChecked(),
    text: ownProps.item.getText(),
    todos: deps.todos,
    item: ownProps.item
}))(TodoItemView);
代码语言:javascript
复制
import * as React from 'react';
import * as ReactDOM from 'react-dom';
import * as inversify from 'inversify';
import { Provider, ChangeNotification } from 'react-inversify';
 
var container = new inversify.Container(); // your DI container
var changeNotification = new ChangeNotification(); // handles changes in model objects

ReactDOM.render(
    <Provider container={container} changeNotification={changeNotification}>
        <TodoListView />
    </Provider>,
    document.getElementById('app')
);

参考:

  • https://www.npmjs.com/package/inversify-inject-decorators
  • https://www.npmjs.com/package/inversify-react
  • https://www.npmjs.com/package/react-inversify
  • https://blog.bitsrc.io/advanced-dependency-injection-in-react-af962bb94d35

最后

React 生态系统中的许多流行库都在使用依赖注入,例如 React RouterRedux。此外,React 还直接支持依赖注入。

但是,对于一些高级的用法,我们需要类似 InversifyJS 之类的库,选择一个适合你的库吧!希望本文能帮到你。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2021-07-12,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 code秘密花园 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 为啥需要依赖注入?
  • React 中的依赖注入
  • InversifyJS
  • inversify-inject-decorators
  • inversify-react
  • react-inversify
  • 参考:
  • 最后
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档