一看就会的Next.js App Router版 -- Routing(下)(二)

本文涉及的产品
云数据库 MongoDB,独享型 2核8GB
推荐场景:
构建全方位客户视图
简介: 一看就会的Next.js App Router版 -- Routing

Route Handlers

路线处理程序

Route Handlers allow you to create custom request handlers for a given route using the Web Request and Response APIs.

路由处理程序允许您使用 Web 请求为给定路由创建自定义请求处理程序

和响应

蜜蜂。

image.png

Good to know: Route Handlers are only available inside the app directory. They are the equivalent of API Routes inside the pages directory meaning you do not need to use API Routes and Route Handlers together.

提示:路由处理程序仅在 app 目录内可用。它们相当于页面目录中的 API 路由,这意味着您不需要同时使用 API 路由和路由处理程序。

Convention

习俗

Route Handlers are defined in a route.js|ts file inside the app directory:

路由处理程序在 app 目录内的 route.js|ts 文件中定义:

app/api/route.ts

TypeScript

export async function GET(request: Request) {}

Route Handlers can be nested inside the app directory, similar to page.js and layout.js. But there cannot be a route.js file at the same route segment level as page.js.

Route Handlers 可以嵌套在 app 目录中,类似于 page.js 和 layout.js。但是不能有和page.js同级的route.js文件。

Supported HTTP Methods

支持的 HTTP 方法

The following HTTP methods are supported: GET, POST, PUT, PATCH, DELETE, HEAD, and OPTIONS. If an unsupported method is called, Next.js will return a 405 Method Not Allowed response.

以下 HTTP 方法

支持:GET、POST、PUT、PATCH、DELETE、HEAD 和 OPTIONS。如果调用了不受支持的方法,Next.js 将返回 405 Method Not Allowed 响应。

Extended NextRequest and NextResponse APIs

扩展的 NextRequest 和 NextResponse API

In addition to supporting native Request and Response. Next.js extends them with NextRequest and NextResponse to provide convenient helpers for advanced use cases.

除了支持原生的Request

和响应

. Next.js 使用 NextRequest 和 NextResponse 扩展它们,为高级用例提供方便的帮助程序。

Behavior

行为

Static Route Handlers

静态路由处理程序

Route Handlers are statically evaluated by default when using the GET method with the Response object.

将 GET 方法与 Response 对象一起使用时,默认情况下会静态评估路由处理程序。

app/items/route.ts

TypeScript

import { NextResponse } from 'next/server'; export async function GET() {  const res = await fetch('https://data.mongodb-api.com/...', {    headers: {      'Content-Type': 'application/json',      'API-Key': process.env.DATA_API_KEY,    },  });  const data = await res.json();   return NextResponse.json({ data });}

TypeScript Warning: Although Response.json() is valid, native TypeScript types currently shows an error, you can use NextResponse.json() for typed responses instead.

TypeScript 警告:尽管 Response.json() 有效,但原生 TypeScript 类型当前显示错误,您可以使用 NextResponse.json() 代替类型化响应。

Dynamic Route Handlers

动态路由处理程序

Route handlers are evaluated dynamically when:

在以下情况下会动态评估路由处理程序:

  • Using the Request object with the GET method.
  • 使用带有 GET 方法的 Request 对象。
  • Using any of the other HTTP methods.
  • 使用任何其他 HTTP 方法。
  • Using Dynamic Functions like cookies and headers.
  • 使用 cookie 和标头等动态功能。
  • The Segment Config Options manually specifies dynamic mode.
  • Segment Config Options 手动指定动态模式。

For example:

例如:

app/products/api/route.ts

TypeScript

import { NextResponse } from 'next/server'; export async function GET(request: Request) {  const { searchParams } = new URL(request.url);  const id = searchParams.get('id');  const res = await fetch(`https://data.mongodb-api.com/product/${id}`, {    headers: {      'Content-Type': 'application/json',      'API-Key': process.env.DATA_API_KEY,    },  });  const product = await res.json();   return NextResponse.json({ product });}

Similarly, the POST method will cause the Route Handler to be evaluated dynamically.

同样,POST 方法将导致路由处理程序被动态评估。

app/items/route.ts

TypeScript

import { NextResponse } from 'next/server'; export async function GET() {  const res = await fetch('https://data.mongodb-api.com/...', {    headers: {      'Content-Type': 'application/json',      'API-Key': process.env.DATA_API_KEY,    },  });  const data = await res.json();   return NextResponse.json({ data });}

注意:以前,API 路由可以用于处理表单提交等用例。路由处理程序可能不是这些用例的解决方案。准备就绪后,我们将建议为此使用突变。

Route Resolution

路由解析

You can consider a route the lowest level routing primitive.

您可以将路由视为最低级别的路由原语。

  • They do not participate in layouts or client-side navigations like page.
  • 他们不参与页面布局或客户端导航。
  • There cannot be a route.js file at the same route as page.js.
  • route.js 文件不能与 page.js 位于同一路径。
Page Route Result
app/page.js app/route.js Conflict
app/page.js app/api/route.js Valid
app/[user]/page.js app/api/route.js Valid

Each route.js or page.js file takes over all HTTP verbs for that route.

每个 route.js 或 page.js 文件接管该路由的所有 HTTP 动词。

app/page.js

javascript

复制代码

exportdefaultfunctionPage() {  return<h1>Hello, Next.js!</h1>;} // ❌ Conflict// `app/route.js`export async function POST(request) {}

Examples

例子

The following examples show how to combine Route Handlers with other Next.js APIs and features.

以下示例展示了如何将路由处理程序与其他 Next.js API 和功能相结合。

Revalidating Static Data

重新验证静态数据

You can revalidate static data fetches using the next.revalidate option:

您可以重新验证静态数据

使用 next.revalidate 选项获取:

app/items/route.ts

TypeScript

javascript

复制代码

import { NextResponse } from'next/server'; exportasyncfunctionGET() {  const res = awaitfetch('https://data.mongodb-api.com/...', {    next: { revalidate: 60 }, // Revalidate every 60 seconds  });  const data = await res.json();   return NextResponse.json(data);}

Alternatively, you can use the revalidate segment config option:

或者,您可以使用 revalidate segment 配置选项:

ini

export const revalidate = 60;

Dynamic Functions

动态函数

Route Handlers can be used with dynamic functions from Next.js, like cookies and headers.

路由处理程序可以与来自 Next.js 的动态函数一起使用,例如 cookie 和标头。

Cookies

饼干

You can read cookies with cookies from next/headers. This server function can be called directly in a Route Handler, or nested inside of another function.

您可以从 next/headers 读取带有 cookie 的 cookie。这个服务器函数可以直接在路由处理程序中调用,也可以嵌套在另一个函数中。

This cookies instance is read-only. To set cookies, you need to return a new Response using the Set-Cookie header.

此 cookie 实例是只读的。要设置 cookie,您需要使用 Set-Cookie 返回一个新的 Response

标头。

app/api/route.ts

TypeScript

import { cookies } from 'next/headers'; export async function GET(request: Request) {  const cookieStore = cookies();  const token = cookieStore.get('token');   return new Response('Hello, Next.js!', {    status: 200,    headers: { 'Set-Cookie': `token=${token}` },  });}

Alternatively, you can use abstractions on top of the underlying Web APIs to read cookies (NextRequest):

或者,您可以在底层 Web API 之上使用抽象来读取 cookie (NextRequest):

app/api/route.ts

TypeScript

import { type NextRequest } from 'next/server'; export async function GET(request: NextRequest) {  const token = request.cookies.get('token');}

Headers

标头

You can read headers with headers from next/headers. This server function can be called directly in a Route Handler, or nested inside of another function.

您可以从 next/headers 中读取带有标题的标题。这个服务器函数可以直接在路由处理程序中调用,也可以嵌套在另一个函数中。

This headers instance is read-only. To set headers, you need to return a new Response with new headers.

此标头实例是只读的。要设置标头,您需要返回带有新标头的新响应。

app/api/route.ts

TypeScript

import { headers } from 'next/headers'; export async function GET(request: Request) {  const headersList = headers();  const referer = headersList.get('referer');   return new Response('Hello, Next.js!', {    status: 200,    headers: { referer: referer },  });}

Alternatively, you can use abstractions on top of the underlying Web APIs to read headers (NextRequest):

或者,您可以在底层 Web API 之上使用抽象来读取标头 (NextRequest):

app/api/route.ts

TypeScript

import { type NextRequest } from 'next/server'; export async function GET(request: NextRequest) {  const requestHeaders = new Headers(request.headers);}

Redirects

重定向

app/api/route.ts

TypeScript

import { redirect } from 'next/navigation'; export async function GET(request: Request) {  redirect('https://nextjs.org/');}
Dynamic Route Segments

动态路线段

We recommend reading the Defining Routes page before continuing.

我们建议在继续之前阅读定义路由页面。

Route Handlers can use Dynamic Segments to create request handlers from dynamic data.

Route Handlers 可以使用 Dynamic Segments 从动态数据创建请求处理程序。

app/items/[slug]/route.js

export async function GET(  request: Request,  {    params,  }: {    params: { slug: string };  },) {  const slug = params.slug; // 'a', 'b', or 'c'}
Route Example URL params
app/items/[slug]/route.js /items/a { slug: 'a' }
app/items/[slug]/route.js /items/b { slug: 'b' }
app/items/[slug]/route.js /items/c { slug: 'c' }

Streaming

串流

app/api/route.ts

TypeScript

lua

// https://developer.mozilla.org/en-US/docs/Web/API/ReadableStream#convert_async_iterator_to_streamfunction iteratorToStream(iterator: any) {  return new ReadableStream({    async pull(controller) {      const { value, done } = await iterator.next();       if (done) {        controller.close();      } else {        controller.enqueue(value);      }    },  });} function sleep(time: number) {  return new Promise((resolve) => {    setTimeout(resolve, time);  });} const encoder = new TextEncoder(); async function* makeIterator() {  yield encoder.encode('<p>One</p>');  await sleep(200);  yield encoder.encode('<p>Two</p>');  await sleep(200);  yield encoder.encode('<p>Three</p>');} export async function GET() {  const iterator = makeIterator();  const stream = iteratorToStream(iterator);   return new Response(stream);}

Request Body

请求正文

You can read the Request body using the standard Web API methods:

您可以使用标准 Web API 方法读取请求正文:

app/items/route.ts

TypeScript

import { NextResponse } from 'next/server'; export async function POST(request: Request) {  const res = await request.json();  return NextResponse.json({ res });}

CORS

You can set CORS headers on a Response using the standard Web API methods:

您可以使用标准 Web API 方法在响应上设置 CORS 标头:

app/api/route.ts

TypeScript

export async function GET(request: Request) {  return new Response('Hello, Next.js!', {    status: 200,    headers: {      'Access-Control-Allow-Origin': '*',      'Access-Control-Allow-Methods': 'GET, POST, PUT, DELETE, OPTIONS',      'Access-Control-Allow-Headers': 'Content-Type, Authorization',    },  });}
Edge and Node.js Runtimes

Edge 和 Node.js 运行时

Route Handlers have an isomorphic Web API to support both Edge and Node.js runtimes seamlessly, including support for streaming. Since Route Handlers use the same route segment configuration as pages and layouts, they support long-awaited features like general-purpose statically regenerated Route Handlers.

Route Handlers 有一个同构的 Web API 来无缝支持 Edge 和 Node.js 运行时,包括对流的支持。由于路由处理程序使用与页面和布局相同的路由段配置,因此它们支持期待已久的功能,如通用静态重新生成的路由处理程序。

You can use the runtime segment config option to specify the runtime:

您可以使用运行时段配置选项来指定运行时:

arduino

export const runtime = 'edge'; // 'nodejs' is the default

Non-UI Responses

非 UI 响应

You can use Route Handlers to return non-UI content. Note that sitemap.xml, robots.txt, favicon.ico, and open graph images all have built-in SEO support.

您可以使用路由处理程序返回非 UI 内容。请注意,sitemap.xml、robots.txt、favicon.ico 和开放图形图像都具有内置的 SEO 支持。

app/rss.xml/route.ts

TypeScript

xml

export async function GET() {  return new Response(`<?xml version="1.0" encoding="UTF-8" ?><rss version="2.0"> <channel>  <title>Next.js Documentation</title>  <link>https://nextjs.org/docs</link>  <description>The React Framework for the Web</description></channel> </rss>`);}
Segment Config Options

段配置选项

Route Handlers use the same route segment configuration as pages and layouts.

路由处理程序使用与页面和布局相同的路由段配置。

app/items/route.ts

TypeScript

dart

export const dynamic = 'auto';export const dynamicParams = true;export const revalidate = false;export const fetchCache = 'auto';export const runtime = 'nodejs';export const preferredRegion = 'auto';
See the API reference for more details.

有关详细信息,请参阅 API 参考。

Middleware

中间件

Middleware allows you to run code before a request is completed. Then, based on the incoming request, you can modify the response by rewriting, redirecting, modifying the request or response headers, or responding directly.

中间件允许您在请求完成之前运行代码。然后,根据传入的请求,您可以通过重写、重定向、修改请求或响应标头或直接响应来修改响应。

Middleware runs before cached content and routes are matched. See Matching Paths for more details.

中间件在缓存内容和路由匹配之前运行。有关详细信息,请参阅匹配路径。

Convention

习俗

Use the file middleware.ts (or .js) in the root of your project to define Middleware. For example, at the same level as pages or app, or inside src if applicable.

使用项目根目录中的文件 middleware.ts(或 .js)来定义中间件。例如,与页面或应用处于同一级别,或者在 src 内部(如果适用)。

相关实践学习
MongoDB数据库入门
MongoDB数据库入门实验。
快速掌握 MongoDB 数据库
本课程主要讲解MongoDB数据库的基本知识,包括MongoDB数据库的安装、配置、服务的启动、数据的CRUD操作函数使用、MongoDB索引的使用(唯一索引、地理索引、过期索引、全文索引等)、MapReduce操作实现、用户管理、Java对MongoDB的操作支持(基于2.x驱动与3.x驱动的完全讲解)。 通过学习此课程,读者将具备MongoDB数据库的开发能力,并且能够使用MongoDB进行项目开发。 &nbsp; 相关的阿里云产品:云数据库 MongoDB版 云数据库MongoDB版支持ReplicaSet和Sharding两种部署架构,具备安全审计,时间点备份等多项企业能力。在互联网、物联网、游戏、金融等领域被广泛采用。 云数据库MongoDB版(ApsaraDB for MongoDB)完全兼容MongoDB协议,基于飞天分布式系统和高可靠存储引擎,提供多节点高可用架构、弹性扩容、容灾、备份回滚、性能优化等解决方案。 产品详情: https://www.aliyun.com/product/mongodb
目录
相关文章
|
3月前
|
小程序 JavaScript 前端开发
uni-app开发微信小程序:四大解决方案,轻松应对主包与vendor.js过大打包难题
uni-app开发微信小程序:四大解决方案,轻松应对主包与vendor.js过大打包难题
794 1
|
4月前
|
监控 JavaScript 前端开发
深入理解 Nuxt.js 中的 app:error 钩子
【9月更文挑战第25天】在 Nuxt.js 中,`app:error` 钩子是一个强大的工具,用于处理应用程序中的各种错误。它可以在服务器端渲染或客户端运行时触发,帮助提升应用稳定性和用户体验。通过在 `nuxt.config.js` 中定义该钩子,开发者可以实现错误页面显示、错误日志记录及错误恢复等功能,确保应用在遇到问题时能妥善处理并恢复正常运行。
60 10
|
5月前
|
资源调度 JavaScript Linux
【Azure 应用服务】本地Node.js部署上云(Azure App Service for Linux)遇到的三个问题解决之道
【Azure 应用服务】本地Node.js部署上云(Azure App Service for Linux)遇到的三个问题解决之道
|
4月前
|
开发者 UED
深入理解 Nuxt.js 中的 app:error 钩子
【9月更文挑战第26天】在 Nuxt.js 中,钩子函数是在特定生命周期阶段执行代码的机制,`app:error` 钩子用于处理应用中的错误,包括服务器端和客户端渲染时出现的问题。它提供了一个集中处理错误的机制,提升了用户体验。当组件渲染过程中出现错误时,`app:error` 钩子会被触发,可以在 `nuxt.config.js` 文件中定义该钩子。通过分析错误对象 `err` 和上下文信息 `context`,开发者可以更好地处理各种错误情况。相比组件内的 `try/catch` 或浏览器原生错误处理,`app:error` 提供了更全局和有针对性的错误处理方式。
|
5月前
|
JavaScript 前端开发
【Azure Developer】在App Service上放置一个JS页面并引用msal.min.js成功获取AAD用户名示例
【Azure Developer】在App Service上放置一个JS页面并引用msal.min.js成功获取AAD用户名示例
|
5月前
|
JavaScript 前端开发 UED
揭秘Vue.js高效开发:Vue Router如何让单页面应用路由管理变得如此简单?
【8月更文挑战第30天】随着Web应用复杂性的增加,单页面应用(SPA)因出色的用户体验和高效的页面加载性能而备受青睐。Vue.js凭借简洁的语法和灵活的组件系统成为构建SPA的热门选择,其官方路由管理器Vue Router则简化了路由管理。本文通过实战示例介绍如何利用Vue Router实现高效的SPA路由管理,包括命名路由、动态路由及其核心优势。
58 0
|
5月前
|
JavaScript Windows
【Azure 应用服务】用App Service部署运行 Vue.js 编写的项目,应该怎么部署运行呢?
【Azure 应用服务】用App Service部署运行 Vue.js 编写的项目,应该怎么部署运行呢?
|
5月前
|
开发框架 JavaScript 前端开发
【Azure Developer】App Service + PubSub +JS 实现多人版黑客帝国文字流效果图
【Azure Developer】App Service + PubSub +JS 实现多人版黑客帝国文字流效果图
|
5月前
|
前端开发 JavaScript Linux
【Azure 应用服务】在Azure App Service for Linux环境中,部署的Django应用,出现加载css、js等静态资源文件失败
【Azure 应用服务】在Azure App Service for Linux环境中,部署的Django应用,出现加载css、js等静态资源文件失败
|
6月前
|
JavaScript Java 测试技术
基于springboot+vue.js+uniapp的成人教育APP附带文章源码部署视频讲解等
基于springboot+vue.js+uniapp的成人教育APP附带文章源码部署视频讲解等
62 2