当前位置: 首页 > 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)测试你新添加的路由,发送带有参数或请求体的请求,并在服务器控制台查看输出。
http://www.xdnf.cn/news/556327.html

相关文章:

  • ⭐️白嫖的阿里云认证⭐️ 第二弹【课时3:大模型辅助内容生产场景】for 「大模型Clouder认证:利用大模型提升内容生产能力」
  • 基于YOLO11深度学习的变压器漏油检测系统【Python源码+Pyqt5界面+数据集+安装使用教程+训练代码】【附下载链接】
  • 通过 API 获取 1688 平台店铺所有商品信息的完整流程
  • Vue+eElement ui el-input输入框 type=number 输入无效。赋值输入框也不显示(问题已解决)
  • FaceFusion 3.2.0 参数配置参考
  • Java实现定时任务的几种常见方式
  • 新闻媒体发稿:社会实践返家乡主题如何选择
  • 《扣子空间:开启AI智能体办公新时代》
  • DAY29 超大力王爱学Python
  • 理解阿里云的MQTT
  • 探秘「4+3原型驱动的交付模式」如何实现软件快速定制
  • MySQL 数据库迁移方法汇总
  • 第五届机器学习与智能系统工程国际学术会议(MLISE 2025)
  • vform自定义表单研究
  • 《算法笔记》11.8小节——动态规划专题->总结 问题 A: 第二题
  • OSA实战笔记一
  • LLM笔记(十一)常见解码/搜索算法
  • canvas浅析(一)
  • Java 09Stream流与File类
  • ragas precision计算的坑
  • 使用frp内网穿透本地的虚拟机
  • 几款常用的虚拟串口模拟器
  • springboot+vue3实现在线购物商城系统
  • MS16-075 漏洞 复现过程
  • Ai学习之openai api
  • 武汉火影数字|数字展厅展馆制作:沉浸式体验,全方位互动
  • Vue 3 深度解析:Composition API、Pinia状态管理与路由守卫实战
  • Rocketmq leader选举机制,通过美国大选解释
  • 第32节:基于ImageNet预训练模型的迁移学习与微调
  • 【MySQL】第六弹——表的CRUD进阶(四)聚合查询(下)