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

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

作者头像
ConardLi
发布于 2021-07-16 03:28:07
发布于 2021-07-16 03:28:07
5.7K01
代码可运行
举报
文章被收录于专栏:code秘密花园code秘密花园
运行总次数:1
代码可运行

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

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

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

为啥需要依赖注入?

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

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

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

React 中的依赖注入

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

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

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

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

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

  1. 只使用 jsx 也能实现依赖注入
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
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
代码运行次数:0
运行
AI代码解释
复制
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
代码运行次数:0
运行
AI代码解释
复制
import { injectable } from "inversify";

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

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

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

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
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
代码运行次数:0
运行
AI代码解释
复制
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
代码运行次数:0
运行
AI代码解释
复制
import { Provider } from 'inversify-react';
...

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

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

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
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
代码运行次数:0
运行
AI代码解释
复制
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
代码运行次数:0
运行
AI代码解释
复制
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 删除。

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

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
使用IOC解耦React组件
除了DI外,另一种实现方式是Dependency Lookup(依赖查找),简称DL。
公众号@魔术师卡颂
2021/03/15
9170
JavaScript 中的依赖注入
依赖注入 DI (Dependency Injection) 是编程领域中一个非常常见的设计模式,它指的是将应用程序所需的依赖关系(如服务或其他组件)通过构造函数参数或属性自动注入的过程。这样做的好处是可以减少组件之间的耦合,更容易测试和维护。
ConardLi
2023/01/09
1.9K0
数栈技术分享:聊聊IOC中依赖注入那些事 (Dependency inject)
依赖注入定义为组件之间依赖关系由容器在运行期决定,形象的说即由容器动态的将某个依赖关系注入到组件之中在面向对象编程中,我们经常处理的问题就是解耦,控制反转(IoC)就是常用的面向对象编程的设计原则,其中依赖注入是控制反转最常用的实现。目标解决当前类不负责被依赖类实例的创建和初始化。
袋鼠云数栈
2021/05/19
6770
数栈技术分享:聊聊IOC中依赖注入那些事 (Dependency inject)
数栈技术分享:聊聊IOC中依赖注入那些事 (Dependency inject)
依赖注入定义为组件之间依赖关系由容器在运行期决定,形象的说即由容器动态的将某个依赖关系注入到组件之中在面向对象编程中,我们经常处理的问题就是解耦,控制反转(IoC)就是常用的面向对象编程的设计原则,其中依赖注入是控制反转最常用的实现。目标解决当前类不负责被依赖类实例的创建和初始化。
袋鼠云数栈
2021/12/07
4910
数栈技术分享:聊聊IOC中依赖注入那些事 (Dependency inject)
TypeScript 接口合并, 你不知道的妙用
声明合并(Declaration Merging) 是 Typescript 的一个高级特性,顾名思义,声明合并就是将相同名称的一个或多个声明合并为单个定义。
_sx_
2023/10/20
1.2K0
TypeScript 接口合并, 你不知道的妙用
一统江湖的大前端(10)——inversify.js控制反转
Angular是由Google推出的前端框架,曾经与React和Vue一起被开发者称为“前端三驾马车”,但从随着技术的迭代发展,它在国内前端技术圈中的存在感变得越来越低,通常只有Java技术栈的后端工程师在考虑转型全栈工程师时才会优先考虑使用。Angular没落的原因并不是因为它不够好,反而是因为它过于优秀,还有点高冷,忽略了国内前端开发者的学习意愿和接受能力,就好像一个学霸,明明成绩已经很好了,但还是不断寻求挑战来实现自我突破,尽管他从不吝啬分享自己的所思所想,但他所接触的领域令广大学渣望尘莫及,而学渣们感兴趣的事物在他看来又有些无聊,最终的结果通常都只能是大家各玩各的。
大史不说话
2021/03/02
3.5K0
一统江湖的大前端(10)——inversify.js控制反转
全新 JavaScript 装饰器实战下篇:实现依赖注入
上一篇文章我们介绍了 JavaScript 最新的装饰器提案,以及它和旧版的区别。这篇文章我们将继续深入装饰器,尝试实现一个简易的依赖注入库。
_sx_
2023/10/20
7350
全新 JavaScript 装饰器实战下篇:实现依赖注入
了不起的 IoC 与 DI
本文阿宝哥将从六个方面入手,全方位带你一起探索面向对象编程中 IoC(控制反转)和 DI(依赖注入) 的设计思想。阅读完本文,你将了解以下内容:
阿宝哥
2020/08/19
2.7K0
🧩 Vue 深入组件开发☞#依赖注入#
当我们的组件只需要子父组件之间传递数据的时候我们可以通过 Props 来满足,这个是没有任何问题的。你可以看到下面这个示意图,当我们的组件出现的层级大于 2 以后,也就是我们常说的爷孙组件之间的数据传递,但是在中间的组件也需要提供支持才能满足数据的顺利传输,当中间的组件层级增多就需要维护更多的与其不是特别关注的内容。为了解决这样的应用场景所以提供了依赖注入的模式。
前端小鑫同学
2022/12/26
5440
🧩 Vue 深入组件开发☞#依赖注入#
使用 TypeScript 和依赖注入实现一个聊天机器人[每日前端夜话0x76]
类型和可测试代码是避免错误的两种最有效方法,尤其是代码随会时间而变化。我们可以分别通过利用 TypeScript 和依赖注入(DI)将这两种技术应用于JavaScript开发。
疯狂的技术宅
2019/05/31
11.2K0
如何在Vue3中使用上下文模式,在React中使用依赖注入模式🚀🚀🚀
这两种不同的设计模式,通常用于软件系统中实现组件之间的数据共享和依赖管理。作为耳熟能详的常见功能,这里就不详细展开定义了,我们单纯的从使用角度去解读他们的区别。
萌萌哒草头将军
2023/11/09
4320
如何在Vue3中使用上下文模式,在React中使用依赖注入模式🚀🚀🚀
聊聊 nestjs 中的依赖注入
首先 nestjs 是什么?引用其官网的原话 A progressive Node.js framework for building efficient, reliable and scalable server-side applications.,翻译一下就是:“一个可以用来搭建高效、可靠且可扩展的服务端应用的 node 框架”。目前在 github 上有 42.4k 的 star 数,人气还是很高的。
政采云前端团队
2022/01/18
3.3K0
聊聊 nestjs 中的依赖注入
Redux 入门教程(三):React-Redux 的用法
前两篇教程介绍了 Redux 的基本用法和异步操作,今天是最后一部分,介绍如何在 React 项目中使用 Redux。 为了方便使用,Redux 的作者封装了一个 React 专用的库 React-R
ruanyf
2018/04/12
1.7K0
Redux 入门教程(三):React-Redux 的用法
使用Angular的依赖注入
Service 的表现形式是一个class,可以用来在组件中复用 比如 Http 请求获取数据,日志处理,验证用户输入等都写成Service,供组件使用。
mafeifan
2019/06/15
9990
解读 IoC 框架 InversifyJS
InversityJS 是一个 IoC 框架。IoC(Inversion of Control) 包括依赖注入(Dependency Injection) 和依赖查询(Dependency Lookup)。
牧云云
2018/12/07
1.2K0
AngularDart4.0 指南- 依赖注入 顶
依赖注入是一个重要的应用程序设计模式。 它的用途非常广泛,几乎所有人都称之为DI。
南郭先生
2018/08/14
5.7K0
【Vuejs】1732- 详细聊一聊 Vue3 依赖注入
在 Vue.js 中,依赖注入[1](DI)是一种非常常见的跨组件传递数据的方法,它可以帮助我们更好地管理组件之间的依赖关系。本文将介绍 Vue3 中的依赖注入机制,包括 provide() 和 inject() 函数的使用方法、使用注意以及优缺点和适用场景等方面的内容。
pingan8787
2023/09/01
8120
【Vuejs】1732- 详细聊一聊 Vue3 依赖注入
react-redux入门教程
最近这段时间在重新回顾上个暑假学的内容,很多内容因为用的比较少就给忘掉了,想着就谢谢博客帮助自己复习一下。
henu_Newxc03
2021/12/28
1.2K0
Angular依赖注入详解
依赖注入的基本思想是:将组件所依赖的服务提供者注入进来,而不是在组件内部直接创建。
坑吭吭
2023/10/16
2780
Angular进阶教程2-<依赖注入+RxJS>
组件\color{#0abb3c}{组件}组件不应该直接获取或保存数据,它们应该聚焦于展示数据,而把数据访问和处理的职责委托给某个服务\color{#0abb3c}{服务}服务。那面对组件和服务之间的关系,该如何处理他们之间的依赖关系呢?Angular就引入了依赖注入框架\color{#0abb3c}{依赖注入框架}依赖注入框架去解决这件事情。
玖柒的小窝
2021/12/08
4.2K0
Angular进阶教程2-<依赖注入+RxJS>
相关推荐
使用IOC解耦React组件
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文