Redian新闻
>
我们如何使用 Next.js 将 React 加载时间缩短 70%

我们如何使用 Next.js 将 React 加载时间缩短 70%

公众号新闻

作者 | Causal
译者 | Sambodhi
策划 | Tina

Causal 是一个多维电子表格,能够处理从基本算术一直到 10 亿次计算的金融模型的一切。Causal 的前端是在 2019 年用 Create React App(CRA)构建的,它为我们提供了很好的服务——它只需要最小的初始设置,并允许快速迭代。随着我们的客户规模和复杂性的增加,性能变得越来越受到关注,我们达到了 CRA 设计支持的极限。最重要的是,CRA 本身并不支持跨多页应用程序的路由分割,所以我们的页面加载时间慢得令人沮丧。为了解决这些问题,我们改用 Next.js,将初始页面加载时间减少了 70%,并将开发者的体验提升到一个新的水平。

什么是 Next.js?

Next.js 是一个框架,带有构建工具和运行时库,用于创建丰富的 React 应用程序。它具有与 CRA 相同的功能,但也包括对 CRA 所缺少的关键功能的内置支持:页面路由、基于页面内容的智能预加载,以及混合静态和服务器端渲染。

从 CRA 迁移到 Next.js

在 2022 年中期,我们认为从 CRA 迁移到 Next.js 的好处是值得投入时间的。我们对于拥有内置的页面路由原语感到特别兴奋,这样我们就不必手动配置路由和 Webpack 构建。

路由

我们之前需要在 CRA 中使用 response-loadable 和 response-router + response-router-dom 来设置我们自己的组件,包括一个大型的 outes.tsx 文件,该文件显式地为应用程序中的每个页面设置了一个路由组件:

import Loadable from "react-loadable";import { Route, Switch } from "react-router-dom";
const EditorLoadable = Loadable({ loader: () => import(/* webpackChunkName: "routes-editor" */ "pages/editor"), loading: ChunkLoading,});
export function RouteSwitch() { return ( <Switch> <Route path={`/model/:id/edit`} render={() => <EditorLoadable />} /> </Switch> );}

与 CRA 相比,Next.js 的优势之一是,Next.js 带有自己的集成链接和路由解决方案,即 next/router。Next.js 只需要在 pages/model/[:id]/edit.tsx 上放置一个带有默认导出的 React 组件的文件,就可以在这个路径上渲染一个页面,其中的 id 属性指示 URL 的 id。

此外,内置的 Next.js Webpack 配置会自动将页面分割成各自的包(bundle)。这意味着访问一个用于本地开发的页面只需要构建该页面所需的包内容。虽然 CRA 支持代码分割,但根据我们的经验,Next.js 配置对于本地重建来说是开箱即用的,速度快得多。

样式

Causal 代码库中的许多旧的 CSS 文件是在团队对 CSS 模块的最佳实践进行标准化之前编写的。其中一些文件使用了“不纯”的 CSS 选择器,这意味着它们可能会影响页面上其他地方的组件所呈现的元素。

例如,我们以前的 Button 组件无意中针对页面上的所有按钮:

// styles/button.scssbutton:disabled {  cursor: not-allowed;}
// components/Button.tsximport "styles/button.scss";
export function Button(props) { return <button {...props} />; }

我们尽可能地将全局 CSS 样式转换为 CSS 模块。这使得组件可以更明确地说明它们采用了哪些样式。

例如,我们在其 CSS 模块中将 Button 组件切换到一个有作用域的类名:

// components/Button/styles.module.scss.button:disabled {  cursor: not-allowed;}// components/Button/index.tsximport styles from "./styles.module.scss";
export function Button({ className, props }) { return <button className={cx(styles.button, className)} {...props} />; }

注意:在最终切换到 Next.js 之前,切换到“纯” CSS 模块也大大改善了 CRA 应用的构建时间。许多 .scss 文件也一直在使用 @USE 和 @EXTEND SCSS 指令来使用其他共享的 .scss 文件来构建样式。这些指令导致共享文件被重新构建为包含它们的每个文件的一部分——导致一些较大文件的每个文件的构建时间都要花费数秒钟!

更多信息请参见 Next.js 关于纯模块的讨论答案。https://github.com/vercel/next.js/discussions/16050#discussioncomment-49022)

部署

一旦我们让 Next.js 在本地工作,下一步就是要改变我们的部署策略。

在这里,CRA 和 Next.js 有根本的区别。CRA 的构建输出只是静态文件,所以提供它相对简单。Next.js 的构建输出确实包括一些静态文件,但它也可能包括运行一个单独服务器的代码。这个服务器负责提供重定向服务,在服务器端渲染动态页面,同时也提供静态页面。

在评估部署我们新的 Next.js 前端的选项时,我们确定了三种可能性:

  1. 不要对 Next.js 使用任何服务器端渲染,使用 next export 构建,并将输出与 CRA 的静态输出完全相同。

  2. 将整个前端托管在 Vercel 上,指向我们的后端(托管在 GCP 中)。

  3. 为 Next.js 服务器编写一个自定义的 Docker 镜像,并将其与我们的后端和其他服务一起托管在 GCP 中。

每种选项都各有利弊:

next export:

优点:要设置的工作量几乎为 0(与 CRA 输出相同)

缺点:不支持服务器端的渲染

托管 Vercel:

优点:只需最少的设置缺点:没有官方对 Yarn2 的支持

缺点:无法轻松连接到数据库以实现更快的服务器端渲染

自定义的 Docker 镜像:

优点:

  • 服务器端渲染的 D 日 ECT DB 连接的最大灵活性是可能的,由于 GCP 的上的托管,后端 API 调用将会非常快

  • 对所需 / 使用的资源进行最细粒度的控制

缺点:所需的最多设置:Vercel 提供了一些示例,但它们并不是开箱即用;Kubernetes 路由 / 网络、扩展等都需要自定义设置

考虑到我们希望获得最大的灵活性,我们选择了选项 3: 编写自定义 Docker 镜像。(不过,我们仍将部署在 Vercel——稍后将详细介绍!)我们在几个页面上进行了少量的服务器端渲染,我们发现到目前为止性能非常好,这在很大程度上归功于与其他服务通信所需的最小网络距离。

预览 App

尽管 Vercel 对于我们的生产部署来说并不可行(如上所述),但它对于其预览 App 仍然相当有用。虽然设置 Vercel 的构建过程需要一些变通方法(因为前面提到的缺乏 Yarn 2 的支持,以及构建我们前端使用的通用包),但好处是巨大的:现在推送到我们 GitHub 仓库的每一个提交都会作为一个预览 App 在 Vercel 上构建和部署,并指向我们的 staging 后端。

这使得前端修改的代码审查体验得到了数量级的改善。审查者只需点击他们正在审查的 PR 中的一个链接,就可以准确地预览该分支在生产中的样子,而不需要在本地拉出分支来进行测试。

虽然这一变化不需要 Next.js,但由于 Vercel 对其自身框架的原生支持,这也是轻而易举的。

结   果

切换到 Next.js 后,终端用户和开发人员的体验都得到了明显的改善。

Causal 模型通常是由几个人创建的,但也有几十个人查看;这些查看者看的是模型仪表板。而在没有对高级服务器端渲染做出任何努力的情况下(使用例如 getServerSideProps),这些仪表盘的加载时间减少了 32%(2.6 秒 → 1.5 秒)!

简单的页面有更显著的速度提升。例如,我们的主页(my.causal.app)的加载速度提高了 71%(1.7 秒 → 0.5 秒),除了从加载状态到载入状态的必要过渡,没有任何布局上的跳跃。

性能上的好处不仅仅是用户体验。Next.js 的开发体验明显比 CRA 快;开发人员从 30%(或更多!)更快的启动时间中受益,快速刷新体验是一种游戏规则的改变者,可以快速迭代小型 UI 调整。到目前为止,最大的改进来自拉取请求预览 App——这是对代码审查体验的重大改进。预览前端代码变化只需要几秒钟而不是几分钟,这使我们能够对较小的拉取请求进行更频繁的审查,也使我们的客户成功团队能够在开发过程的早期提供反馈。

下一步

看到 App 在 Next.js 上运行,我们感到非常兴奋。我们的页面加载速度明显更快,我们的本地构建只需几秒钟而不是几分钟就可以开始,我们需要维护的 Webpack 配置量是几十行而不是几百行。

我们计划很快实现更多的服务器端渲染,首先是嵌入式图表和表格,这些通常是由匿名访客查看的。我们希望看到这些用户的体验能因更快地加载时间而得到明显改善。

当然,现代 Web 应用的性能远不止首次加载时间那么简单。更重要的是用户互动的性能,这在 Causal 中特别难以优化,因为我们是一个渲染复杂网格、图表和表格的重数据应用。在未来的博客文章中,我们将分享更多关于如何解决这些性能问题的内容。

原文链接:

https://www.causal.app/blog/next-js

声明:本文为InfoQ翻译,未经许可禁止转载。

今日好文推荐

反转!马斯克正在求被裁工程师复职,尤其是Android和iOS开发

苹果暂停除研发外岗位招聘,市值一夜蒸发7160亿元;腾讯和联通合资公司因为云计算;国美停发工资,要求员工签理解承诺书|Q资讯

阿里巴巴开源下一代云原生网关Higress:基于Envoy,支持Nginx Ingress零成本快速迁移

马斯克整顿西方职场,Twitter员工突然进入“黑色星期五”:办公室关闭,裁员名单确认前不必上班

微信扫码关注该文公众号作者

戳这里提交新闻线索和高质量文章给我们。
相关阅读
当年俄罗斯人为什么会拥抱普京(ZT)心衰患者如何使用SGLT2抑制剂?最新专家共识划重点[评测]ASUS ROG Crosshair X670E Extreme 评测More Chinese Minors Are Online But Fewer Addicted, Report Says由《隐入尘烟》说到电影《武训传》的随想篇首个在ImageNet上精度超过80%的二值神经网络BNext问世!Fedora 38有望将关机时间缩短87%新加坡最新地铁线路即将开通,通勤时间缩短40%,受惠数百万人!​Retell Lecture / Summarize Spoken Text 究竟如何练习Why Wind and Solar Are Ready to Power China’s Next 40 Years取消国际航班熔断机制!入境隔离时间缩短为5+3!留学生寒假回国有望!马斯克入主推特如虎添翼,他的超级权力将被如何使用?【资讯】入境航班熔断取消,入境人员集中隔离时间缩短了两天!如何使用乳清蛋白粉The Next Era of ReformWebpack落幕?其作者在Next.js 13中带来快700倍的替代品信息超载时代下如何保持有效阅读如何使用常见词提升口语水平?英国女王辞世,突然想起了慈禧太后Why Chinese Cinema Is Still Waiting on the Next Generation攻略 | 如何使用康文署设施?Next.js 13提供新的实验性特性,实现App“动态无限制”首个在ImageNet上精度超过80%的二值神经网络BNext问世,-1与+1的五年辛路历程股价跌超20%,陈列平公司NextCure停止主导项目研发Htmx意外走红,我们从React“退回去”后:代码行数减少 67%,JS 依赖项从 255 下降到 9沉浸式文旅赶追影视级质感,元宇宙时代虚实共生重塑|SFC EVENTS投研社 x NeXT SCENE超越ConvNeXt!Conv2Former:用于视觉识别的Transformer风格的ConvNet芝加哥艺术学院(三)塞尚(上)我如何使用现场 USB 设备恢复我的 Linux 系统 | Linux 中国【Call-for-Participation】LEAP 11.0- SoCal NextGen Career Forum再次试探重要支撑线,NextEra Energy将如何选择?触摸美国 54 收获季节【重磅好消息】回国更容易了!隔离时间缩短为5+3,取消熔断,登机前检测简化!NeurIPS 2022 | 重振PointNet++雄风!PointNeXt:改进模型训练和缩放策略审视PointNet++精度超越ConvNeXt的新CNN!HorNet:通过递归门控卷积实现高效高阶的空间信息交互
logo
联系我们隐私协议©2024 redian.news
Redian新闻
Redian.news刊载任何文章,不代表同意其说法或描述,仅为提供更多信息,也不构成任何建议。文章信息的合法性及真实性由其作者负责,与Redian.news及其运营公司无关。欢迎投稿,如发现稿件侵权,或作者不愿在本网发表文章,请版权拥有者通知本网处理。