Skip to main content

4月24日内容汇总

· 14 min read
加菲猫

📒 Ubuntu 22.04 LTS 安装

https://phoenixnap.com/kb/ubuntu-22-04-lts

https://releases.ubuntu.com/jammy/

📒 计算机程序的构造和解释 — JavaScript 版

这本由麻省理工学院出版的著作终于有了 JavaScript 语言版本,可以帮助你建立对计算机程序的心智模型。

https://github.com/source-academy/sicp

📒 为什么要使用 Redux Toolkit

Redux 官方发布的这篇博客讲解了 Redux Toolkit 的 Why 和 How,并强烈推荐使用。

一句话总结:Redux Toolkit 是使用 Redux 的最佳实践。

https://redux.js.org/introduction/why-rtk-is-redux-today

⭐️ Node.js 18 新特性解读

📒 那些你应该说再见的 npm 祖传老库

📒 如何实现数组转对象

传入一个 paramKeys 数组,获取 query 参数的值,然后以对象形式返回,使用 reduce 方法:

function getSearchParams(paramKeys: string[]): Record<string, string> {
const searchParams = new URLSearchParams(window.location.search);
return paramKeys.reduce<Record<string, string>>((accu, cur) => {
accu[cur] = searchParams.get(cur) || '';
return accu;
}, {});
}

// 使用
const searchParams = getSearchParams(['name', 'age']);

以上流程还可以封装成自定义 hook:

import * as React from 'react';

function useSearchParams(paramKeys: string[]): Record<string, string> {
const query = window.location.search;
return React.useMemo(() => {
const searchParams = new URLSearchParams(query);
return paramKeys.reduce<Record<string, string>>((accu, cur) => {
accu[cur] = searchParams.get(cur) || '';
return accu;
}, {});
}, [paramKeys, query]);
}

看了 antfu 大佬的代码,还可以使用 Object.fromEntries() 方法:

function useSearchParams(paramKeys: string[]): Record<string, string> {
const searchParams = new URLSearchParams(window.location.search);
return Object.fromEntries(
paramKeys.map((key) => [key, searchParams.get(key)])
);
}

注意:Object.fromEntries() 是 ES2019 中的语法,存在兼容性问题(Chrome >= 73),不过只要正确配置 polyfill 就可以放心使用

📒 使用 defineConfig 约束配置对象

在项目中经常需要用到配置对象,例如 Webpack、rollup 的配置,我们可以使用 TS 来约束配置对象的 API schema,告知用户应该传哪些字段以及对应的类型,这样有两个好处:

  • 对用户更加友好,不需要看文档就能直接上手
  • 在开发阶段就能提前检查出配置项错误,不用到运行阶段再去校验了

一般来说我们需要导出一个接口类型:

export type IConfig = {
name: string;
age: number;
sex?: boolean;
};

用户在使用的时候需要导入类型,然后自己添加注解,这样编写配置对象就能得到类型提示了:

import type { IConfig } from "xxx";

const config: IConfig[] = [
{
name: "dbydm",
age: 23
}
]

但是这样对用户来说还是太麻烦了,我们可以定义一个 defineConfig 函数,这个函数做的事情很简单,就是把接收到的参数原封不动地返回,但在这个过程中,就可以实现参数类型的校验:

type IConfig = {
name: string;
age: number;
sex?: boolean;
};

export function defineConfig(config: IConfig[]) {
return config;
}

用户只需导入 defineConfig 编写配置就可以实现参数类型的校验:

import { defineConfig } from "xxx";

export default defineConfig([
{
name: "dbydm",
age: 23
}
])

📒 Elasticsearch 基础入门详文

📒 如何把前端项目写成一座屎山

📒 浅谈JS内存机制

📒 深入理解 scheduler 原理

📒 前端框架如何实现预渲染

首先预渲染根据渲染时机分为以下两种:

  • 静态站点生成(SSG),构建的时候获取数据进行渲染,数据不一定是最新的
  • 服务端渲染(SSR),用户访问的时候服务端获取数据进行渲染,数据实时获取

两种渲染方案都可以实现 首屏性能优化SEO 优化,不同的是 SSR 需要在服务端运行 JS,并且每次用户请求的时候都会进行渲染;SSG 已经将每个页面渲染成静态 html,因此可以将资源托管到 CDN 上

获取数据又可以分为以下几种方式:

  • 本地文件系统读取
  • 调接口获取
  • 查询数据库获取

实际上,React 本身已经提供了服务端渲染和静态生成相关的 API。在前端项目中,我们一般会使用下面的 API 挂载 React 组件:

ReactDOM.render(element, container[, callback])

为了实现 SSR 渲染,我们可以使用下面的 API 将 React 组件直接渲染为 HTML 字符串:

ReactDOMServer.renderToString(element)

使用 renderToString 方法渲染出的 HTML 字符串会带有特定标记,我们可以使用下面的 API 在客户端进行激活,对标记的节点挂载相应的事件监听器:

ReactDOM.hydrate(element, container[, callback])

在 SSG 渲染中,我们不需要在客户端进行激活,因此不用在 HTML 字符串中添加标记,只需渲染出纯的 HTML 字符串:

ReactDOMServer.renderToStaticMarkup(element)

📒 2万字系统总结,带你实现 Linux 命令自由

📒 还在手撸 Nginx 配置?试试这款可视化配置工具吧,真心强大

📒 esno,基于 Esbuild 的神器

📒 「React进阶」换个姿势看 hooks ! 灵感来源组合和HOC 模式下逻辑视图分离新创意

useMemo 类似 Vue 中的计算属性,当依赖项发生变化,会重新计算。但实际上 useMemo 比计算属性更强大,除了缓存值之外,还能缓存组件:

function Index({ value }){
const [number, setNumber] = React.useState(0);
const element = React.useMemo(() => <Test />, [value]);

return (
<div>
{element}
<button onClick={() => setNumber(number + 1)}>点击 {number}</button>
</div>
)
}

有时候在父组件定义的事件处理函数,需要作为 prop 传入子组件。如果父组件重新渲染,会导致函数重新生成,相当于 prop 发生变化,即使子组件内部使用 React.memo() 包裹也会导致重新渲染。常规做法是使用 React.useCallback() 包裹事件处理函数,但实际上用 React.useRef() 包裹也是可以的,都是把事件处理函数缓存到 Fiber 节点上。

function MyApp() {
const onClickRef = React.useRef(() => {
console.log("666");
});

// const onClick = React.useCallback(() => {}, []);

return (
<div>
<h1>Welcome to my app</h1>
<MyButton onClick={onClickRef.current} />
</div>
);
}

「React进阶」换个姿势看 hooks ! 灵感来源组合和HOC 模式下逻辑视图分离新创意

📒 React 18 升级踩坑汇总

1. React.StrictMode 导致所有组件重复挂载两次

使用 CRA 5.0.1 搭建 React 项目,默认的项目模板中,根组件使用了 React.StrictMode 包裹,结果出现了所有组件都重复挂载的情况,导致组件中接口调了两次。看了下文档,确实是 React 18 中引入的 Breaking Change,启用严格模式,会导致所有组件重复挂载两次(即使用了 React.memo 也会重复挂载):

Stricter Strict Mode: In the future, React will provide a feature that lets components preserve state between unmounts. To prepare for it, React 18 introduces a new development-only check to Strict Mode. React will automatically unmount and remount every component, whenever a component mounts for the first time, restoring the previous state on the second mount. If this breaks your app, consider removing Strict Mode until you can fix the components to be resilient to remounting with existing state

tip

使用 CRA 创建的 React 18 项目,建议移除 React.StrictMode

2. React 18 中使用了 antd 的 message 组件控制台打印警告信息

React 18 使用了新的 ReactDOM.createRoot() API 挂载根节点,Concurrent Mode 需要通过此 API 开启,但是 antd 中的 message 等组件内部仍使用 ReactDOM.render() 挂载根节点,此时在控制台会打印警告,注意这并不是报错,仅仅只是 fallback 到 legacy mode 而已。

tip

升级前最好仔细看一遍官方的说明,特别是 Breaking Change:

https://github.com/facebook/react/releases/tag/v18.0.0

📒 为什么需要 peerDependencies

例如开发一个 React 组件库的时候,有三个诉求:

  • 该组件库开发的时候需要安装 React;
  • 用户引入该组件库的时候不能重复安装 React;
  • 组件库的 React 版本与目标环境不一致的时候需要被包管理器发现并打印警告;

如果安装到 dependencies 下,显然会导致重复安装;如果安装到 devDependencies 下虽然不会导致重复安装,但包管理器不会检查版本,当版本不一致的时候不会打印警告。所以 peerDependencies 是最优选择。

在老版本 React 项目中引入某些依赖库(例如 antdreact-transition-group),一般不能直接安装最新的版本(大概率会报错),此时应该根据依赖库的 package.json 中指定的 peerDependencies 字段选择合适的依赖库版本

⭐️ 什么是 JavaScript 的函数组合

本篇文章以一个简略的 Markdown 的例子为主线,讲述了什么是函数组合,以及如何使用函数组合的思想编写代码,是一篇非常不错的编程思想类文章。

https://jrsinclair.com/articles/2022/javascript-function-composition-whats-the-big-deal/

📒 一些关于react的keep-alive功能相关知识在这里(下)

📒 一些关于react的keep-alive功能相关知识在这里(上)

📒 理清 HTTP 下的 TCP 流程,让你的 HTTP 水平更上一层

📒 React 18 系列

React 18 全览

React 18 对 Hooks 的影响

React 的心智模型

你不知道的 React v18 的任务调度机制

📒 React 几个小技巧

1. React 内置工具类型

// 使用 React.ComponentType 同时表示类组件和函数组件
type ComponentType<P = {}> = ComponentClass<P> | FunctionComponent<P>;

// 使用 React.Key 来表示列表渲染 key 的类型
type Key = string | number;

参考:

精读《@types react 值得注意的 TS 技巧》

从 @types/react 的类型定义中,我学到了什么

https://github.com/DefinitelyTyped/DefinitelyTyped/blob/master/types/react/v17/index.d.ts

2. 自定义组件如何绑定 className

className 作为 prop 传入,内部使用 classnames 这个库进行拼接:

import cx from "classnames";
import s from "./style.module.less";

type IProps = {
className: string;
}

const App: React.FC<IProps> = ({ className }) => {
return (
<div className={cx(s.wrapper, className)}>
...
</div>
)
}

更进一步,把以上流程封装成高阶组件(HOC):

function withClassName<T>(Component: React.ComponentType<T>): React.FC<T & { className: string }> {
return ({ className, ...restProps }) => {
return (
<div className={className}>
<Component {...restProps} />
</div>
)
}
}

3. Input 如何变为受控组件

Antd 中的 Input 默认是非受控组件,可以绑定 value,然后监听 onChange 修改 value 实现受控(v-model 的原理):

const App: React.FC<{}> = () => {
const [num, setNum] = React.useState(1);

return (
<InputNumber value={num} onChange={setNum} />
)
}

📒 我帮一朋友重构了点代码,他直呼牛批,但基操勿六

📒 React + TypeScript:如何处理常见事件

📒 单例模式 4 种经典实现方法

📒 如何实现 useClickAway

如何监听元素外的点击,类似 Vue 的 ClickOutSide 指令

官方文档:https://ahooks.js.org/hooks/use-click-away

源码:https://github.com/alibaba/hooks/blob/master/packages/hooks/src/useClickAway/index.ts

📒 快速理解 TypeScript 的逆变、协变

📒 都 2022 年了,手动搭建 React 开发环境很难吗

📒 这篇手写 Promise 你一定要康康

📒 超全面的前端新一代构建工具对比: esbuild、Snowpack、Vite、wmr