Next.js 作为一个强大的 React 框架,为开发者提供了两种路由系统:App Router 和 Pages Router。这两种路由系统各有特色,适用于不同的场景。本文将深入探讨这两种路由系统的区别、优缺点和使用场景,帮助你做出最佳选择。

App Router:新一代的路由革命

App Router 是 Next.js 13 引入的新路由系统,它使用 app 目录来组织路由,带来了许多令人兴奋的新特性。

优点:

  1. React 服务器组件支持:这是一个游戏规则改变者,允许在服务器端渲染复杂组件,大大提升了性能。

  2. 灵活的布局系统:通过嵌套布局,你可以更容易地创建复杂的页面结构。

  3. 内置加载 UI 和错误处理:提供了更好的用户体验,无需额外配置。

  4. 性能优化:得益于服务器组件和其他优化,App Router 通常能提供更好的性能。

  5. 并行路由:允许在同一布局中同时渲染多个页面。

缺点:

  1. 学习曲线较陡:对于习惯了传统 React 开发的人来说,可能需要一些时间来适应。

  2. 第三方库兼容性:一些老旧的库可能不兼容新的服务器组件模式。

  3. 仍在发展中:作为较新的技术,可能会有一些未知的问题或变化。

Pages Router:经典可靠的选择

Pages Router 是 Next.js 的传统路由系统,使用 pages 目录来组织路由。它仍然是许多项目的首选,特别是对于较老的 Next.js 版本。

优点:

  1. 简单易上手:对于初学者来说,学习曲线相对平缓。

  2. 文件系统路由直观:路由结构与文件结构一一对应,易于理解和管理。

  3. 丰富的社区资源:由于使用时间较长,有大量的教程、示例和第三方库支持。

  4. 稳定性高:经过多年的使用和优化,bug 较少,表现稳定。

缺点:

  1. 不支持 React 服务器组件:无法利用这一新特性带来的性能提升。

  2. 布局系统相对简单:实现复杂布局可能需要更多的代码和配置。

  3. 数据获取方法较为固定:主要依赖 getServerSidePropsgetStaticProps,灵活性较低。

实战对比:博客页面实现

让我们通过一个简单的博客页面来对比这两种路由系统的实现方式:

Pages Router 实现

// pages/posts/[id].js
import { useRouter } from 'next/router'; export default function Post({ post }) {
const router = useRouter(); if (router.isFallback) {
return <div>Loading...</div>;
} return (
<div>
<h1>{post.title}</h1>
<p>{post.content}</p>
</div>
);
} export async function getStaticProps({ params }) {
const res = await fetch(`https://api.example.com/posts/${params.id}`);
const post = await res.json();
return { props: { post } };
} export async function getStaticPaths() {
const res = await fetch('https://api.example.com/posts');
const posts = await res.json(); const paths = posts.map(post => ({
params: { id: post.id.toString() },
})); return { paths, fallback: true };
}

在这个例子中,我们使用 getStaticPropsgetStaticPaths 来实现静态生成。这是 Pages Router 的典型用法,适合内容不经常变化的博客文章。

App Router 实现

// app/posts/[id]/page.js
import { notFound } from 'next/navigation'; async function getPost(id) {
const res = await fetch(`https://api.example.com/posts/${id}`);
if (!res.ok) return undefined;
return res.json();
} export default async function Post({ params }) {
const post = await getPost(params.id); if (!post) {
notFound();
} return (
<div>
<h1>{post.title}</h1>
<p>{post.content}</p>
</div>
);
}

App Router 的实现更加简洁。这里我们直接在组件中进行异步数据获取,这得益于 React 服务器组件的支持。同时,我们使用 notFound 函数来处理文章不存在的情况,这是 App Router 提供的内置错误处理机制之一。

如何选择?

选择 App Router 还是 Pages Router,没有绝对的对错。以下是一些建议:

  1. 项目规模和复杂度:对于大型、复杂的项目,App Router 的灵活性和性能优势可能更有吸引力。

  2. 团队熟悉度:如果团队对 Next.js 较为陌生,可能从 Pages Router 开始更容易上手。

  3. 性能需求:如果项目对性能有较高要求,App Router 的服务器组件可能是更好的选择。

  4. 项目时间线:对于需要快速开发的项目,Pages Router 可能更适合,因为学习成本较低。

  5. 未来展望:考虑到 Next.js 的发展方向,长期来看,掌握 App Router 可能更有优势。

个人经验分享

作为一个初使用 Next.js 的开发者,我最初对 App Router 也感到困惑。但是,当我开始处理复杂的布局和需要优化性能的场景时,App Router 的优势就显现出来了。

例如,在一个需要频繁更新的数据密集型应用中,App Router 的服务器组件让我能够在服务器端处理大部分数据逻辑,显著减少了传输到客户端的 JavaScript 数量,提升了应用的整体性能。

然而,对于一些简单的项目或者时间紧迫的情况,我仍然会选择 Pages Router。它简单直接,能让我快速搭建原型并上线。

结论

App Router 和 Pages Router 各有千秋,选择哪一个取决于你的具体需求和场景。

我的建议是:不要害怕尝试新事物。即使你现在使用的是 Pages Router,也值得花些时间了解 App Router。

毕竟,技术在不断进步,保持学习才能不被淘汰。

记住,技术只是工具,真正重要的是解决问题和创造价值。选择最适合你的工具,才能事半功倍。

结语

上次写了一个用于批量清理无用仓库的工具,如果感兴趣那就去看看

介绍文章https://mp.weixin.qq.com/s/t7lgc6b7xJiNhfm5vWo5-A

GitHub 仓库地址https://github.com/yaolifeng0629/del-repos

如果你觉得这个工具对你有所帮助,请不要忘记给我的 GitHub 仓库点个 Star!你的支持是我前进的动力!

感谢阅读,我们下次再见!

Next.js 中为什么 App Router 可能是未来,但 Pages Router 仍然重要?的更多相关文章

  1. Vue.js中,如何自己维护路由跳转记录?

    在Vue的项目中,如果我们想要做返回.回退操作时,一般会调用router.go(n)这个api,但是实际操作中,使用这个api有风险,就是会让用户跳出当前应用,因为它记录的是浏览器的访问记录,而不是你 ...

  2. 小程序首页onLoad为异步,调用app.js中的全局参数的解决方案。

    一,先说一下遇到的问题: 在首页,为了携带app.js中一些参数去做请求动作,但是由于异步原因,发现请求时候,参数信息还未获取到但请求已经发出去. 若等app.js的全局参数返回来,再携带着它去做请求 ...

  3. Express4.10.2开发框架中默认app.js的代码注释

    //通过require()加载了express.path等模块var express = require('express');var path = require('path');var favic ...

  4. 微信小程序首页index.js获取不到app.js中动态设置的globalData的原因以及解决方法

    前段时间开发了一款微信小程序,运行了也几个月了,在index.js中的onLoad生命周期里获取app.js中onLaunch生命周期中在接口里动态设置的globalData一直没有问题,结果昨天就获 ...

  5. 微信小程序app.js中设置公有变量

    初始化GlobalData 在App.js的最上方可以设置GlobalData的初始值. App({ globalData:{ appid: '1wqas2342dasaqwe232342xxxxxx ...

  6. vue.js移动端app实战3:从一个购物车入门vuex

    什么是vuex? 官方的解释是:Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式.它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化. 简单来说就 ...

  7. 从前端中的IOC理念理解koa中的app.use()

    忙里偷闲,打开平时关注的前端相关的网站,浏览最近最新的前端动态.佼佼者,平凡的我做不到,但还是要争取不做落后者. 前端中的IoC理念,看到这个标题就被吸引了.IoC 理念,不认识呢,点击去一看,果然没 ...

  8. Vue.js中学习使用Vuex详解

    在SPA单页面组件的开发中 Vue的vuex和React的Redux 都统称为同一状态管理,个人的理解是全局状态管理更合适:简单的理解就是你在state中定义了一个数据之后,你可以在所在项目中的任何一 ...

  9. 在egg.js中使用mongodb

    1.egg.js官网只推荐了mysqle,要用mongodb得另找资料.通过查找,大家都在用Mongoose连接,于是乎学习. 网站链接:https://www.npmjs.com/package/e ...

  10. vue cli 中关于vue.config.js中chainWebpack的配置

    Vue CLI  的官方文档上写:调整webpack配置最简单的方式就是在vue.config.js中的configureWebpack选项提供一个对象. Vue CLI 内部的 webpack 配置 ...

随机推荐

  1. python的requirements.txt_维护项目依赖包

    pycharm没有类似maven用于管理依赖包的工具,当一个项目在新的环境运行前,需要将对应依赖的包下载回来,如果一个个下载,会出现缺漏或版本号不对应的情况,这个时候可以用requirements.t ...

  2. VMware 17 Exception 0xc0000094 解决

    VMWare16的虚拟机升级到17时, 可能会出现虚拟机可以正常使用, 但编辑设置就会出现vmui错误的现像. VMware Workstation unrecoverable error: (vmu ...

  3. UART和RS232、RS485的关系是什么?

    串口通讯是电子工程师和嵌入式开发工程师面对的最基本问题,RS232则是其中最简单最常用的通讯方式.但是初学者往往搞不清有关的名词如UART和RS232或RS485之间是什么关系,因为它们经常被放到语句 ...

  4. V4L2视频采集操作流程和接口说明

    背景: V4L2是V4L的升级版本,为linux下视频设备程序提供了一套接口规范.包括一套数据结构和底层V4L2驱动接口. <WAV文件格式分析> 一般操作流程(视频设备): 1.打开设备 ...

  5. sqlmap 环境搭建 sqli-labs平台搭建

    sqlmap 环境搭建: windows 1.先去官网下载:https://sqlmap.org/ 2.在python的Scripts目录下创建一个sqlmap 把官网下载的东西解压到那里 3.添加环 ...

  6. ELK之Elastic-Search 整理(一):基础理论 与 DSL语法 及 Java操作ES

    基础理论和DSL语法 准备工作 什么是ElasticSearch?它和Lucene以及solr的关系是什么? 这些是自己的知识获取能力,自行百度百科 下载ElasticSearch的window版 l ...

  7. P5494 题解

    来一发 \(O(\log n)\) 线性空间的解法. 考虑通过只维护线段树叶子节点的虚树的方法压缩空间,考虑记录下每个节点的编号,然后通过异或完求最低位的 \(1\) 的方式求出 LCA 的深度,然后 ...

  8. Win10 内存内存占用过高的一种解决方案

    前言 最近win10的电脑一开机,什么都没启动,内存占用率高达90%,笔记本虽然是8G的内存,但不应该如此不堪.在网上找到一个十分有效的解决方案. 正文 使用 win10系统自带内存诊断工具 按下 h ...

  9. 痞子衡嵌入式:浅聊恩智浦i.MXRT官方SDK里关于串行Flash相关的驱动与例程资源(上篇)

    大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是恩智浦i.MXRT官方SDK里关于串行Flash相关的驱动与例程资源. 经常有同事以及 i.MXRT 客户咨询痞子衡,咱们恩智浦官方 S ...

  10. Mac mysql5.7.x my.cnf默认配置

    配置如下 [client] port = 3306 default-character-set=utf8 [mysqld] character_set_server=utf8 datadir=/usr ...