当前位置: 首页 > news >正文

第六部分:第三节 - 路由与请求处理:解析顾客的点单细节

控制器是 NestJS 中负责接收 HTTP 请求的组件,而各种方法装饰器和参数装饰器则帮助我们精确地定义如何响应不同类型的请求,以及如何方便地从请求中提取所需的信息。这就像服务员接收顾客的点单,需要准确记录菜品名称(路由路径)、份数(请求参数)、特殊要求(查询参数)、配料调整(请求体)等各种细节。

控制器前缀 (@Controller('prefix')):

@Controller() 装饰器中指定字符串参数,可以为控制器中的所有路由设置一个公共的前缀。

// src/users/users.controller.ts (修改)
import { Controller, Get } from '@nestjs/common';// 控制器前缀设置为 'api/users'
@Controller('api/users')
export class UsersController {// ... 构造函数和 getHello 等方法 ...@Get() // 实际处理的路径是 /api/usersfindAll(): string { /* ... */ return '所有用户'; }@Get(':id') // 实际处理的路径是 /api/users/:idfindOne(): string { /* ... */ return '单个用户'; }
}

在这里插入图片描述

现在访问用户相关的接口就需要以 /api/users 开头了。这有助于组织 API 端点。

方法装饰器 (@Get, @Post, etc.):

这些装饰器用于标记控制器中的方法,告诉 NestJS 这个方法应该处理哪种 HTTP 方法以及对应的路由路径(相对于控制器前缀)。

// src/products/products.controller.ts (示例)
import { Controller, Get, Post, Put, Delete, Param, Body } from '@nestjs/common';
import { ProductsService } from './products.service';@Controller('api/products')
export class ProductsController {constructor(private readonly productsService: ProductsService) {}@Get() // GET /api/productsfindAll() {return this.productsService.getProducts();}@Get(':id') // GET /api/products/:idfindOne(@Param('id') id: string) { // 使用 @Param('id') 获取路径参数 id// 在实际应用中,可能需要将 id 从 string 转换为 numberreturn `获取产品 ID: ${id}`;}@Post() // POST /api/productscreate(@Body() productData: any) { // 使用 @Body() 获取整个请求体console.log('收到新产品数据:', productData);// 在实际应用中,会调用 Service 添加产品到数据库return { message: '产品创建成功', data: productData };}@Put(':id') // PUT /api/products/:idupdate(@Param('id') id: string, @Body() productData: any) {console.log(`更新产品 ID: ${id}, 数据:`, productData);return { message: `产品 ID ${id} 更新成功`, data: productData };}@Delete(':id') // DELETE /api/products/:idremove(@Param('id') id: string) {console.log(`删除产品 ID: ${id}`);return { message: `产品 ID ${id} 删除成功` };}// @All() 可以匹配所有 HTTP 方法// @All()// handleAll() {//   return '匹配所有方法';// }
}

在这里插入图片描述

参数装饰器 (@Param, @Query, @Body, @Headers, @Req, @Res):

这些装饰器用于标记控制器方法的参数,告诉 NestJS 应该从请求对象中提取哪个部分的数据并作为该参数的值。

  • @Param(): 提取路由参数。可以加上字符串参数指定提取哪个参数 (@Param('id')),或者不加参数提取包含所有路由参数的对象 (@Param()).
  • @Query(): 提取 URL 查询参数。可以加上字符串参数指定提取哪个参数 (@Query('keyword')),或者不加参数提取包含所有查询参数的对象 (@Query()).
  • @Body(): 提取请求体。可以加上字符串参数指定提取请求体中的某个属性 (@Body('name')),或者不加参数提取整个请求体对象 (@Body()). 注意: 使用 @Body() 前,需要确保已经配置了请求体解析中间件(Express 内置的 express.json()express.urlencoded(),NestJS 默认集成了)。
  • @Headers(): 提取请求头。可以加上字符串参数指定提取某个特定的请求头 (@Headers('Content-Type')),或者不加参数提取包含所有请求头的对象 (@Headers()).
  • @Req(): 注入原始的 Express/Fastify 请求对象。
  • @Res(): 注入原始的 Express/Fastify 响应对象。注意: 除非需要直接控制响应流(比如发送文件),否则通常不推荐使用 @Res() 并手动调用 res.send(), res.json() 等。NestJS 会自动处理返回值并发送响应,这样可以更好地利用 NestJS 的拦截器等特性。

小例子:处理各种请求参数

// src/example/example.controller.ts (假设你创建了一个 example 模块和控制器)
import { Controller, Get, Post, Query, Param, Body, Headers } from '@nestjs/common';@Controller('api/example')
export class ExampleController {@Get('greet') // GET /api/example/greet?name=Alice&greeting=Hellogreet(@Query('name') name?: string,   // 获取查询参数 'name',可选@Query('greeting') greeting = 'Hi' // 获取查询参数 'greeting',提供默认值) {return `${greeting}, ${name || 'Guest'}!`;}@Get('items/:itemId/details') // GET /api/example/items/123/detailsgetItemDetails(@Param('itemId') itemId: string) {// itemId 是从路径中提取的参数return `获取项目 ID: ${itemId} 的详细信息`;}@Post('echo') // POST /api/example/echoecho(@Body() data: any, @Headers('Content-Type') contentType: string) {// data 是请求体,contentType 是请求头 'Content-Type' 的值console.log('请求体:', data);console.log('Content-Type:', contentType);return { receivedData: data, contentType: contentType };}
}

小结: NestJS 的控制器通过方法装饰器 (@Get, @Post 等) 和参数装饰器 (@Param, @Query, @Body 等) 提供了一种声明式的方式来定义路由和提取请求信息。这使得请求处理逻辑清晰、易于阅读和维护,极大地简化了后端开发中处理请求参数的繁琐工作。

练习:

  1. 在你之前的 my-backend 项目中,修改 products.controller.ts 文件,为控制器添加一个前缀,比如 'api/products'
  2. 修改 findOne 方法,使用 @Param('id') 参数装饰器获取产品 ID。假设 ID 是数字类型,尝试为方法参数添加类型注解 id: string
  3. ProductsController 添加一个 POST 方法 create (@Post()),使用 @Body() 参数装饰器获取整个请求体。在方法中打印出接收到的请求体。
  4. ProductsController 添加 PUT (@Put(':id')) 和 DELETE (@Delete(':id')) 方法,分别获取路由参数 ID 和请求体(对于 PUT)。
  5. 运行应用,使用工具(如 Postman)测试你新添加的路由,发送带有参数或请求体的请求,并在服务器控制台查看输出。

相关文章:

  • 【基础知识】SPI协议的种类及异同
  • OpenCV CUDA 模块特征检测与描述------在GPU上执行特征描述符匹配的类cv::cuda::DescriptorMatcher
  • SetThrowSegvLongjmpSEHFilter错误和myFuncInitialize 崩溃
  • 宝塔+fastadmin:给项目添加定时任务
  • 汽车区域电子电气架构(Zonal E/E)的统一
  • CentOS 7上BIND9配置DNS服务器指南
  • SpringSecurity基础入门
  • 使用Mathematica绘制一类矩阵的特征值图像
  • 使用F5-tts复刻音色
  • CVE-2017-4971源码分析与漏洞复现
  • Linux 特权管理与安全——从启用 Root、Sudo 提权到禁用与防护的全景解析
  • idea如何让文件夹分层显示,而不是圆点分割
  • 【牛客-输入输出练习】
  • 安全工具配置
  • STM32外设应用详解——从基础到高级应用的全面指南
  • 基于NLP技术的客户投诉与需求文本分类方法研究
  • 大语言模型的评估指标
  • 差动讯号(2):奇模与偶模
  • RedissonClient主要功能概述
  • 公路水运安全员C证用途及重要性
  • 中沙深化多领域合作,达成60余项共识
  • 印度空军为“阵风”战机换装国产导弹,以增强作战能力推动国防自主
  • 国家发改委:内卷式竞争扭曲市场机制、扰乱公平竞争秩序,必须整治
  • 《歌手2025》能否“爆”下去?
  • 以军称已开始在加沙的新一轮大规模攻势
  • 特朗普称即将与伊朗达成核协议,外交部:中方愿继续发挥建设性作用