几天不写React,已经看不懂语法了

开发 前端
真是几天不写React,语法都看不懂了。本文就来聊聊这几个Use关键词各自的意义。

大家好,我卡颂。

下面这个React组件代码,用到3个use关键词,你理解他们的作用吗?

'use client'

function App() {
  using data = use(ctx);
  
  // ...
}

真是几天不写React,语法都看不懂了。本文就来聊聊这几个use关键词各自的意义。

use client

首先是位于代码顶部的'use client'声明,使用方式类似于严格模式的声明:

'use strict';
// 此处是严格模式下的JavaScript代码

'use client'声明是RSC(React Server Component,服务端组件)协议中的定义。

启用了RSC的React应用,所有组件默认在服务端渲染(可以通过Next v13体验),只有声明'use client'的组件文件,会在前端渲染。

假设我们的React应用组件结构如下,其中红色代表「服务端组件」,蓝色代表「客户端组件」(声明'use client'):

那么当应用打包后,D、E组件会打包成独立文件。在前端,React可以直接渲染A、B、C组件。但是对于D、E,需要以JSONP的形式请求回组件代码再渲染。

完整执行逻辑如下:

using关键字

接下来是data变量前的using关键字:

using data = use(ctx);

using关键字是tc39提案ECMAScript Explicit Resource Management[1]提出的,用于为各种资源(内存、I/O等)提供统一的生命周期管理(何时分配、何时释放等)。

同时,TS v5.2率先引入了这个关键字。所以,接下来的讲解我们以TS中的using关键词为准。

using的作用有点类似useEffect的destroy函数。当我们在useEffect的create函数绑定了事件后,可以在destroy函数解绑:

function App() {
  useEffect(() => {
    console.log('这里是create函数')
    return () => {
      console.log('这里是destroy函数')
    }
  }, [])
}

类似的,当我们通过using关键词声明一个包含[Symbol.dispose]方法的对象后,当离开当前作用域时,声明的[Symbol.dispose]方法会执行:

{
  const getResource = () => {
    return {
      [Symbol.dispose]: () => {
        console.log('离开啦!')
      }
    }
  }
  using resource = getResource();
}
// 代码执行到这里会打印 离开啦!

在[Symbol.dispose]方法内主要执行一些释放资源的操作。

比如,当我们操作数据库时,如果要考虑「操作完断开数据库连接」,可能会写出如下代码:

const db = await connectDB();
try {
  // 执行数据库操作
} finally {
  // 断开数据库连接
  await db.close();
}

如果使用using关键词,代码如下:

const connect = async () => {
  const db = await connectDB();
  return {
    db,
    [Symbol.asyncDispose]: () => db.close()
  };
};

// 使用
{
  using { db } = await connect();
  // 执行数据库操作
} 
// 离开作用域自动断开连接

配合async await使用,可以降低「由于忘记释放资源造成内存泄漏」的可能性。

use方法

最后是React v18.3之后发布的新原生hook —— use:

using data = use(ctx);

这个hook可以接收两种类型数据:

  • React Context

此时use的作用与useContext一样。

  • promise

此时如果这个promise处于pending状态,则最近一个祖先<Suspense/>组件可以渲染fallback。

比如,在如下代码中,如果<Cpn />组件或其子孙组件使用了use,且promise处于pending状态(比如请求后端资源):

function App() {
  return (
    <div>
      <Suspense fallback={<div>loading...</div>}>
        <Cpn />
      </Suspense>
    </div>
  );
}

那么,页面会渲染如下结果:

<div>
  <div>loading...</div>
</div>

当请求成功后,会渲染<Cpn />。

总结

对于开篇提到的代码:

'use client'

function App() {
  using data = use(ctx);
  
  // ...
}

表示:

  • 这是个客户端组件
  • 如果传递给use的变量ctx是React Context,则use的作用等同于useContext。
  • 如果传递给use的变量ctx是promise,则配合最近的<Suspense/>使用。
  • 如果use的返回值包含[Symbol.dispose],则App组件render完成后会执行[Symbol.dispose]方法。

一个文件,三款use相关语法,你是不是已经懵逼了呢?

参考资料

[1]ECMAScript Explicit Resource Management:https://github.com/tc39/proposal-explicit-resource-management。

责任编辑:姜华 来源: 魔术师卡颂
相关推荐

2021-12-09 11:59:49

JavaScript前端提案

2020-03-30 16:45:06

代码看不懂

2019-12-09 08:29:26

Netty架构系统

2020-03-06 11:30:08

JavaGitHub编程

2013-07-08 10:49:03

程序员代码看懂代码

2022-07-26 14:38:08

JavaScriptWeb安全自动化

2022-12-12 07:40:36

服务器项目Serverless

2022-06-16 14:07:26

Java代码代码review

2014-03-12 09:25:33

产品经理Startup

2021-02-23 10:36:09

Linux命令kmdr

2022-02-07 09:05:00

GitHub功能AI

2020-09-21 13:06:58

TikTok网络安全隐私

2017-09-19 15:45:39

2020-11-06 08:36:04

UI设计规范iOS

2019-10-24 08:56:38

语言代码Java

2022-01-05 09:40:03

DIff算法前端

2017-06-16 09:22:22

数据结构算法链表

2023-06-06 07:41:00

Reacthook

2022-01-14 23:44:57

电脑进程设置

2021-10-08 08:58:35

物联网通信发布者
点赞
收藏

51CTO技术栈公众号