React开发的设计模式及原则

简介: 设计模式是最常见的,通用问题的可复用解决方案的归纳总结,通常被认为是解决该类问题的最佳实践,使用设计模式能帮助我们写出更容易维护,更健壮的代码。设计模式有很多,通常它们都会遵循一些共同的设计原则,接下来我们一起回顾下React社区里出现过的一些设计模式,以及它们所遵循的设计原则。

介绍

设计模式是对常见的,通用问题的可复用解决方案的归纳总结,通常被认为是解决该类问题的最佳实践,使用设计模式能帮助我们写出更容易维护,更健壮的代码。设计模式有很多,通常它们都会遵循一些共同的设计原则,接下来我们一起回顾下React社区里出现过的一些设计模式,以及它们所遵循的设计原则。

一些设计原则

  1. 单一职责原则(Single-responsibility responsibility) : 每个实体(class, function, module)只应该有一个职责。例如当一个组件接收了太多的props,我们应该考虑组件是不是做了太多的事情,有没有必要进行拆分。
  2. 开闭原则(Open-closed principle):实体(class, function, module) 应该对扩展开放,但是对修改关闭。开闭原则意味着应该存在不直接修改的方式扩展实体的功能。
  3. 依赖反转原则(Dependency inversion principle):依赖于抽象,而不是具体的实现。依赖注入是一种实现依赖反转的方式。
  4. 不要自我重复 (Don't repeat yourself):重复代码会造成代码维护的困难。
  5. Composition over inheritance: 通过组合集成的两个组件是松耦合关系,通过props来约束。但是有继承关系的两个组件是强耦合关系,对父组件的修改可能会导致子组件的未预期的结果。

React设计模式

Container & presentational component

把业务组件划分成container组件和presentational组件。 Presentational component中负责组件的ui渲染,Container component负责数据的获取和事件的响应。

遵循的设计原则:
  1. 单一职责原则: Presentational component负责ui,Container component负责数据和行为。
  2. Don't repeat yourself: Presentational component是纯ui组件,不包含业务逻辑,通常可以被复用。
示例
importReactfrom"react";
// Presentational componentexportdefaultfunctionImageList({ images, onClick }) {
returnimages.map((img, i) =><imgsrc={img} key={i} onClick={onClick} />);
}
// Container componentexportdefaultclassImageListContainerextendsReact.Component {
constructor() {
super();
this.state= {
images: []
    };
  }
componentDidMount() {
fetch("https://images.com")
      .then(res=>res.json())
      .then(({ images }) =>this.setState({ images }));
  }
handleClick() {
// ...  }
render() {
return<ImageListimages={this.state.images} onClick={handleClick} />;
  }
}

HOC

Higher-order component 是一个以组件为参数,返回一个新组件的函数,用于复用组件的逻辑,Redux的 connect 和 Relay的createFragmentContainer都有使用HOC模式。

遵循的设计原则:
  1. Don't repeat yourself:把可复用的逻辑放到HOC中,实现代码复用。
  2. Composition over inheritance: hoc中传入的组件和返回的组件是组合的关系, 也可以把多个HOC进行多次的嵌套组合。
示例
importReactfrom"react";
exportdefaultfunctionwithLoader(Component, url) {
returnclassHOCextendsReact.Component {
constructor(props) {
super(props);
this.state= {
loading: true,
data: {},
      };
    }
componentDidMount() {
fetch(url)
        .then((res) =>res.json())
        .then(({ data }) =>this.setState({ data }))
        .finally(() =>this.setState({loading: false}))
    }
render() {
if (this.state.loading) {
return<div>Loading...</div>;
      }
return<Component {...this.props} data={this.state.data} />;
    }
  };
}

Render prop

Render prop是指组件的使用者通过组件暴露的函数属性来参与定制渲染相关的逻辑。使用Render prop模式的库包括: React Router, Downshift and Formik.

遵循的设计原则:
  1. Don't repeat yourself:把可复用的逻辑放到组件中,实现代码复用。
  2. 依赖反转原则:通过render prop注入渲染相关的实现。
  3. 开闭原则(Open-closed principle): 通过render prop暴露扩展点,而不是直接定制组件。
示例
importReactfrom"react";
classLoaderextendsReact.Component {
constructor(props) {
super(props);
this.state= {
loading: true,
data: {},
    };
  }
componentDidMount() {
fetch(url)
      .then((res) =>res.json())
      .then(({ data }) =>this.setState({ data }))
      .finally(() =>this.setState({ loading: false }));
  }
render() {
if (this.state.loading) {
return<div>Loading...</div>;
    }
returnthis.props.renderData(this.state.data);
  }
}

Compound components

Compound components是指通过多个组件的组合来完成特定任务,这些组件通过共享的状态、逻辑进行关联。典型的例子是Select和Select.Option组件。使用Compound components模式的库包括:semantic ui;

遵循的设计原则:
  1. 单一职责原则(Single-responsibility responsibility): 拆分成多个组件,每个组件承担自己的职责。
  2. 开闭原则(Open-closed principle): 需要迭代增强功能时,可以通过创建新的子组件的方式进行扩展。
示例
importReactfrom"react";
constSelectContext=React.createContext({});
exportfunctionSelect({ value, onChange, children }) {
const [open, setOpen] =React.useState(false);
const [val, setValue] =React.useState(value);
return (
<divclassName={`select`}><divclassName="select-value"onClick={() => {
setOpen(true);
        }}
>        {val}
</div><SelectContext.Providervalue={{
value: val,
setOpen,
setValue: (newValue) => {
setValue(newValue);
if (value!==newValue) {
onChange(newValue);
            }
          },
        }}
>        {open&&children}
</SelectContext.Provider></div>  );
}
functionOption({ children, value }) {
const {
setOpen,
setValue,
value: selectedValue,
  } =React.useContext(SelectContext);
return (
<divclassName={`select-option ${value===selectedValue?"selected" : ""}`}
onClick={() => {
setValue(value);
setOpen(false);
      }}
>      {children}
</div>  );
}
functionOptionGroup({ children, label }) {
return (
<divclassName="select-option-group"><divclassName="select-option-group-label">{label}</div>      {children}
</div>  );
}
Select.Option=Option;
Select.OptionGroup=OptionGroup;
functionDemo() {
const [city, setCity] =React.useState("北京市");
return (
<Selectvalue={city} onChange={setCity}><Select.Optionvalue="北京市">北京市</Select.Option><Select.OptionGrouplabel="河北省"><Select.Optionvalue="石家庄市">石家庄市</Select.Option><Select.Optionvalue="保定市">保定市</Select.Option></Select.OptionGroup></Select>  );
}

Custom hooks

自定义hooks可以做到把与state和生命周期关联的可复用逻辑封装到独立的函数中, 上面的提及的一些模式都是基于组件的方案,自定义hooks是更细粒度的解决方案。

遵循的设计原则:
  1. Don't repeat yourself:把可复用的逻辑放到自定义hooks中,实现代码复用。
  2. 单一职责原则:每个自定义hooks是都是一个独立的逻辑单元。
示例:
import { useState, useEffect } from"react";
functionuseLoader(url) {
const [data, setData] =useState({});
const [loading, setLoading] =useState(false);
useEffect(() => {
setLoading(true);
fetch(url)
      .then((res) =>res.json())
      .then(({ data }) => {
setData({ data });
      })
      .finally(() =>setLoading(false));
  }, [url]);
return { data, loading };
}

结尾

上面提及的曾经在社区流行的设计模式,往往遵守了一些设计原则,从而能帮助开发者写出健壮,易维护的代码。但是我们需要能根据实际的场景做出判断,是否需要引入这些模式,毕竟还有一个设计原则是YAGNI (You aren't gonna need it)。

目录
相关文章
|
21天前
|
前端开发 JavaScript API
React开发需要了解的10个库
本文首发于微信公众号“前端徐徐”,介绍了React及其常用库。React是由Meta开发的JavaScript库,用于构建动态用户界面,广泛应用于Facebook、Instagram等知名网站。文章详细讲解了Axios、Formik、React Helmet、React-Redux、React Router DOM、Dotenv、ESLint、Storybook、Framer Motion和React Bootstrap等库的使用方法和应用场景,帮助开发者提升开发效率和代码质量。
87 4
React开发需要了解的10个库
|
1月前
|
设计模式 算法 搜索推荐
后端开发中的设计模式应用与实践
在软件开发的广袤天地中,后端技术如同构筑高楼大厦的钢筋水泥,支撑起整个应用程序的骨架。本文旨在通过深入浅出的方式,探讨后端开发领域内不可或缺的设计模式,这些模式犹如精雕细琢的工具箱,能够助力开发者打造出既健壮又灵活的系统架构。从单例模式到工厂模式,从观察者模式到策略模式,每一种设计模式都蕴含着深刻的哲理与实践价值,它们不仅仅是代码的组织方式,更是解决复杂问题的智慧结晶。
|
2月前
|
设计模式 算法 搜索推荐
后端开发中的设计模式应用
在软件开发的浩瀚海洋中,设计模式犹如一座座灯塔,为后端开发者指引方向。本文将深入探讨后端开发中常见的设计模式,并通过实例展示如何在实际项目中巧妙应用这些模式,以提升代码的可维护性、扩展性和复用性。通过阅读本文,您将能够更加自信地应对复杂后端系统的设计与实现挑战。
64 3
|
2月前
|
设计模式 安全 数据库连接
后端开发中的设计模式应用
在软件开发的浩瀚海洋中,设计模式如同灯塔,为后端开发者指引方向。它们不仅仅是代码的模板,更是解决复杂问题的智慧结晶。本文将深入探讨几种常见的设计模式,包括单例模式、工厂模式和观察者模式,并揭示它们在实际应用中如何提升代码的可维护性、扩展性和重用性。通过实例分析,我们将一窥这些模式如何在后端开发中大放异彩,助力构建高效、灵活的软件系统。
|
25天前
|
前端开发 JavaScript 开发者
React 组件化开发最佳实践
【10月更文挑战第4天】React 组件化开发最佳实践
47 4
|
2月前
|
设计模式 Java 关系型数据库
设计模式——设计模式简介和七大原则
设计模式的目的和核心原则、单一职责原则、接口隔离原则、依赖倒转原则、里氏替换原则、开闭原则、迪米特法则、合成复用原则
设计模式——设计模式简介和七大原则
|
2月前
|
XML 移动开发 前端开发
使用duxapp开发 React Native App 事半功倍
对于Taro的壳子,或者原生React Native,都会存在 `android` `ios`这两个文件夹,而在duxapp中,这些文件夹的内容是自动生成的,那么对于需要在这些文件夹中修改的配置内容,例如包名、版本号、新架构开关等,都通过配置文件的方式配置了,而不需要需修改具体的文件
|
2月前
|
设计模式 算法 搜索推荐
后端开发中的设计模式应用
在软件开发的浩瀚海洋中,设计模式犹如灯塔一般指引着方向。它们不是一成不变的规则,而是前人智慧的结晶。本文将深入探讨几种在后端开发中常用的设计模式,如单例、工厂、观察者和策略模式,并阐述如何在实际项目中灵活运用这些模式来提升代码质量、可维护性和扩展性。通过对比传统开发方式与应用设计模式后的差异,我们将揭示设计模式在解决复杂问题和优化系统架构中的独特价值。
|
2月前
|
资源调度 JavaScript 前端开发
使用vite+react+ts+Ant Design开发后台管理项目(二)
使用vite+react+ts+Ant Design开发后台管理项目(二)
|
2月前
|
设计模式 JavaScript Java
后端开发中的设计模式应用
本文将深入探讨后端开发中常见的设计模式,包括单例模式、工厂模式和观察者模式。每种模式不仅会介绍其定义和结构,还会结合实际案例展示如何在后端开发中应用这些模式来优化代码的可维护性与扩展性。通过对比传统方法与设计模式的应用,读者可以更清晰地理解设计模式的优势,并学会在项目中灵活运用这些模式解决实际问题。