手记

Nest后端开发入门指南

概述

Nest.js 是一个用于构建高效、可扩展的 Node.js 服务器端应用程序的框架,它结合了现代 JavaScript/TypeScript 的功能和最佳实践。本文将详细介绍 Nest.js 的环境搭建、核心概念与架构以及一些基本功能的开发,帮助你掌握 Nest 后端开发。

Nest.js 简介与环境搭建
什么是 Nest.js

Nest.js 是一个用于构建高效、可扩展的 Node.js 服务器端应用程序的框架。它结合了现代 JavaScript/TypeScript 的功能和最佳实践,如依赖注入、面向切面编程、类型检查等。Nest.js 被设计成高度可测试和模块化的,能够轻松地与各种库和工具集成。

Nest.js 的核心目标是为开发者提供一种标准化的方式来创建稳健的 Node.js 应用程序。它提供了大量预定义的装饰器、服务以及工具,简化了开发流程。例如,你可以使用 @Controller 装饰器来定义一个控制器,使用 @Service 装饰器来定义一个服务,并使用 @Module 装饰器来定义一个模块。这使得代码结构清晰,易于维护。

开发环境的搭建

开始使用 Nest.js 之前,你需要搭建好开发环境。这包括安装 Node.js 和 NPM,设置开发工具,以及安装 Nest.js CLI 工具。

安装 Node.js 和 NPM

安装 Node.js 和 NPM(Node.js 包管理器)是运行 Nest.js 应用的前提条件。请访问 Node.js 官方网站 下载并安装最新版本的 Node.js。安装过程中,会自动安装 NPM。

验证安装是否成功,可以通过以下命令检查 Node.js 和 NPM 的版本:

node -v
npm -v

这两个命令分别显示 Node.js 和 NPM 的版本号。

创建第一个 Nest.js 项目

安装好 Node.js 和 NPM 后,接下来需要安装 Nest.js CLI 工具,它可以帮助你快速创建和管理 Nest.js 项目。

  1. 安装 Nest.js CLI 工具

    npm i -g @nestjs/cli
  2. 创建一个新的 Nest.js 项目

    nest new my-nest-app

    这条命令会在当前目录下创建一个名为 my-nest-app 的新目录,并在其中初始化一个 Nest.js 项目。项目目录结构如下:

    my-nest-app/
    ├── node_modules/
    ├── src/
    │   ├── app.controller.ts
    │   ├── app.module.ts
    │   └── app.service.ts
    ├── .gitignore
    ├── nest-cli.json
    ├── package.json
    ├── README.md
    └── tsconfig.json
    • node_modules/:存放项目的依赖库。
    • src/:存放项目的源代码。
    • app.controller.ts:定义了应用的主要控制器。
    • app.module.ts:定义了应用的主要模块。
    • app.service.ts:定义了应用的服务。
    • .gitignore:指定 Git 忽略的文件和目录。
    • nest-cli.json:Nest.js CLI 的配置文件。
    • package.json:项目的 npm 配置文件。
    • README.md:项目的 README 文件。
    • tsconfig.json:TypeScript 编译器设置文件。
核心概念与架构

控制器(Controller)

在 Nest.js 中,控制器用于处理 HTTP 请求。每个控制器通常与一个或多个路由关联,处理客户端传来的请求,并返回响应。控制器通过装饰器 @Controller 来定义。

import { Controller, Get } from '@nestjs/common';

@Controller('users')
export class UsersController {
  @Get()
  findAll(): string {
    return 'This action returns all users';
  }
}

在上面的例子中,UsersController 控制器定义了一个 findAll 方法,它返回一个字符串。@Get() 装饰器表示这个方法处理 HTTP GET 请求。

服务(Service)

服务在 Nest.js 中用于封装业务逻辑。服务可以被多个控制器共享,通常用于执行更复杂的业务逻辑、数据处理或数据库操作。服务通过装饰器 @Injectable 定义。

import { Injectable } from '@nestjs/common';

@Injectable()
export class UserService {
  findAll() {
    return 'This action returns all users';
  }
}

在上面的例子中,UserService 服务定义了一个 findAll 方法,该方法返回一个字符串。@Injectable() 装饰器用于标记该服务可以被依赖注入。

数据库交互(Repository)

Nest.js 支持多种 ORM(对象关系映射)工具,例如 TypeORM、Sequelize 等,它们可以帮助你操作数据库。通常情况下,你使用 @EntityRepository 装饰器来定义一个仓库类。

import { EntityRepository, Repository } from 'typeorm';
import { User } from '../entities/user.entity';

@EntityRepository(User)
export class UserRepository extends Repository<User> {
  async find(): Promise<User[]> {
    return this.find();
  }
}

在上面的例子中,UserRepository 继承了 Repository<User> 类,并定义了一个 find 方法,该方法返回所有用户。

模块(Module)

模块是 Nest.js 应用的组织结构单元。通过模块,你可以定义应用的各个部分,并指定它们所需的配置和依赖关系。模块通过装饰器 @Module 定义。

import { Module } from '@nestjs/common';
import { UsersService } from './users.service';
import { UsersController } from './users.controller';

@Module({
  providers: [UsersService],
  controllers: [UsersController],
})
export class UsersModule {}

在上面的例子中,UsersModule 模块定义了两个属性:providerscontrollersproviders 数组包含服务,controllers 数组包含控制器。

装饰器(Decorator)

装饰器是 Nest.js 的一个关键特性,它们用于修饰类、方法或属性,从而添加额外的功能。例如,@Controller@Get@Post 等都是装饰器。装饰器通常以 @ 开头。

import { Controller, Get } from '@nestjs/common';
import { UserService } from './user.service';

@Controller('users')
export class UsersController {
  constructor(private readonly userService: UserService) {}

  @Get()
  findAll(): string {
    return this.userService.findAll();
  }
}

在上面的例子中,@Controller('users') 装饰器定义了一个控制器,@Get() 装饰器定义了一个 GET 请求处理器。

基本功能开发

RESTful API 设计

RESTful API 是一种 Web 服务设计风格,它基于 HTTP 协议,使用标准的 HTTP 方法如 GET、POST、PUT 和 DELETE 来执行 CRUD(创建、读取、更新、删除)操作。在 Nest.js 中,你可以使用装饰器和控制器来定义 RESTful API。

import { Controller, Get, Post, Body } from '@nestjs/common';
import { CreateUserDto } from './dto/create-user.dto';
import { UserService } from './user.service';

@Controller('users')
export class UsersController {
  constructor(private readonly userService: UserService) {}

  @Get()
  findAll(): string {
    return this.userService.findAll();
  }

  @Post()
  create(@Body() createUserDto: CreateUserDto) {
    return 'This action adds a new user';
  }
}

在上面的例子中,@Get()@Post() 装饰器分别用于定义 GET 和 POST 请求处理器。@Body() 装饰器用于获取请求体中的数据。

数据模型与 ORM

ORM(对象关系映射)是一种将对象模型映射到关系数据库的技术。在 Nest.js 中,常用 ORM 工具有 TypeORM 和 Sequelize。本节以 TypeORM 为例进行说明。

首先,你需要安装 TypeORM:

npm i --save @nestjs/typeorm typeorm

然后,定义一个实体(Entity):

import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm';

@Entity()
export class User {
  @PrimaryGeneratedColumn()
  id: number;

  @Column()
  name: string;
}

定义一个仓库(Repository):

import { EntityRepository, Repository } from 'typeorm';
import { User } from '../entities/user.entity';

@EntityRepository(User)
export class UserRepository extends Repository<User> {
  async find(): Promise<User[]> {
    return this.find();
  }
}

定义一个服务(Service):

import { Injectable } from '@nestjs/common';
import { UserRepository } from '../repositories/user.repository';

@Injectable()
export class UserService {
  constructor(private readonly userRepository: UserRepository) {}

  async findAll(): Promise<User[]> {
    return this.userRepository.find();
  }
}

请求验证与错误处理

请求验证通常用于确保客户端发送的数据符合预期格式。Nest.js 通过第三方库 class-validatorclass-transformer 实现请求验证。

首先,安装相关依赖:

npm i --save class-validator class-transformer

定义一个 DTO(数据传输对象):

import { IsString, IsNotEmpty } from 'class-validator';
import { Transform } from 'class-transformer';

export class CreateUserDto {
  @IsString()
  @IsNotEmpty()
  @Transform(({ value }) => value.trim())
  name: string;
}

在控制器中使用 DTO 进行验证:

import { Controller, Post, Body, ValidationPipe } from '@nestjs/common';
import { UserService } from './user.service';
import { CreateUserDto } from './dto/create-user.dto';

@Controller('users')
export class UsersController {
  constructor(private readonly userService: UserService) {}

  @Post()
  create(@Body(ValidationPipe) createUserDto: CreateUserDto) {
    return 'This action adds a new user';
  }
}

验证管道 ValidationPipe 会自动验证请求体中的数据是否符合 CreateUserDto 定义的规则。

日志记录与调试

日志记录对于调试和监控应用运行状态非常重要。Nest.js 集成了 winston 库进行日志记录。

首先,安装 winston

npm i --save winston

然后,在模块中定义日志服务:

import { Module } from '@nestjs/common';
import { LoggerService } from './logger.service';

@Module({
  providers: [LoggerService],
  exports: [LoggerService],
})
export class LoggerModule {}

定义 LoggerService 服务:

import { Injectable, Scope } from '@nestjs/common';
import { Logger, transports } from 'winston';

@Injectable({ scope: Scope.DEFAULT })
export class LoggerService {
  private logger: Logger;

  constructor() {
    this.logger = new Logger({
      level: 'info',
      transports: [
        new transports.Console({
          format: info => `${info.timestamp} ${info.level} ${info.message}`,
        }),
      ],
    });
  }

  debug(message: any) {
    this.logger.debug(message);
  }

  info(message: any) {
    this.logger.info(message);
  }

  warn(message: any) {
    this.logger.warn(message);
  }

  error(message: any) {
    this.logger.error(message);
  }
}

使用服务记录日志:

import { Injectable } from '@nestjs/common';
import { LoggerService } from '../logger/logger.service';

@Injectable()
export class UserService {
  constructor(private readonly loggerService: LoggerService) {}

  async findAll() {
    this.loggerService.info('Finding all users');
    // 业务逻辑...
  }
}
高级特性介绍

中间件的使用

中间件在框架中起到拦截请求和响应的作用,可以用于身份验证、日志记录、错误处理等。Nest.js 提供了多种中间件类型,包括全局中间件、管道中间件等。

全局中间件

全局中间件可以处理所有传入的请求。例如:

import { NestFactory } from '@nestjs/core';
import { AppModule } from './app.module';
import { LoggerMiddleware } from './logger/logger.middleware';

async function bootstrap() {
  const app = await NestFactory.create(AppModule);
  app.use((req, res, next) => {
    console.log('Request:', req.method, req.url);
    next();
  });
  await app.listen(3000);
}
bootstrap();

管道中间件

管道中间件用于特定控制器或方法。例如:

import { Controller, Get, UseInterceptors } from '@nestjs/common';
import { LoggingInterceptor } from './logger/logger.interceptor';

@Controller('users')
export class UsersController {
  @Get()
  @UseInterceptors(LoggingInterceptor)
  findAll() {
    return 'This action returns all users';
  }
}

模块化开发

模块化开发是 Nest.js 的核心特性之一,它鼓励你将应用划分为多个独立的模块,每个模块负责不同的功能。模块可以导入其他模块,并使用 @Module 装饰器定义。

import { Module } from '@nestjs/common';
import { UsersService } from './users.service';
import { UsersController } from './users.controller';

@Module({
  providers: [UsersService],
  controllers: [UsersController],
})
export class UsersModule {}

访问令牌与认证

认证是保护应用安全的重要手段。Nest.js 支持多种认证机制,如 JWT(JSON Web Token)。

首先,安装 jsonwebtoken

npm i --save jsonwebtoken

定义 JWT 服务:

import { Injectable } from '@nestjs/common';
import * as jwt from 'jsonwebtoken';

@Injectable()
export class AuthService {
  private readonly secret = 'mySecretKey';

  sign(userId: string): string {
    return jwt.sign({ userId }, this.secret);
  }

  verify(token: string): string {
    return jwt.verify(token, this.secret);
  }
}

异步处理与 RxJS

Nest.js 支持异步操作和响应式编程。通过 RxJS,你可以处理复杂的异步流,例如从数据库获取数据、处理网络请求等。

首先,安装 rxjs

npm i --save rxjs

定义一个异步服务:

import { Injectable } from '@nestjs/common';
import { Observable } from 'rxjs';

@Injectable()
export class UserService {
  async findAll(): Promise<string[]> {
    return ['Alice', 'Bob', 'Charlie'];
  }

  findAllAsync(): Observable<string[]> {
    return new Observable(observer => {
      setTimeout(() => {
        observer.next(['Alice', 'Bob', 'Charlie']);
        observer.complete();
      }, 1000);
    });
  }
}

在控制器中使用异步服务:

import { Controller, Get } from '@nestjs/common';
import { UserService } from './user.service';

@Controller('users')
export class UsersController {
  constructor(private readonly userService: UserService) {}

  @Get()
  async findAll(): Promise<string[]> {
    return this.userService.findAll();
  }

  @Get('/async')
  findAllAsync() {
    return this.userService.findAllAsync();
  }
}
项目部署与维护

构建与发布

构建 Nest.js 项目通常使用 tsc(TypeScript 编译器)进行编译。Nest.js CLI 提供了 build 命令来执行构建操作。

nest build

构建完成后,可以在 dist/ 目录下找到编译后的 JavaScript 文件。你可以将这些文件部署到生产环境。

服务器部署

部署 Nest.js 应用可以使用多种方案,包括 Node.js 服务器、Docker 容器、Kubernetes 等。

使用 Node.js 服务器

你可以使用 pm2forever 等工具来运行 Node.js 应用。例如:

npm install pm2 -g
pm2 start dist/main.js --name my-nest-app

使用 Docker 容器

使用 Docker 部署可以简化部署流程。首先,创建一个 Dockerfile

FROM node:14

WORKDIR /app

COPY package*.json ./

RUN npm ci

COPY . .

RUN npm run build

CMD ["node", "dist/main.js"]

然后,使用 Docker 构建和运行:

docker build -t my-nest-app .
docker run -p 3000:3000 my-nest-app

资源优化

资源优化包括代码优化、资源管理和性能调优。例如,可以通过调整缓存策略、使用 CDN、优化数据库查询等方式提升应用性能。

CI/CD 集成

CI/CD(持续集成和持续部署)是现代软件开发的最佳实践。Nest.js 应用可以通过多种 CI/CD 工具进行自动化构建和部署,例如 Jenkins、GitLab CI、GitHub Actions 等。

使用 GitHub Actions

创建一个 .github/workflows/ci.yml 文件:

name: CI

on:
  push:
    branches: [master]

jobs:
  build:
    runs-on: ubuntu-latest

    steps:
    - uses: actions/checkout@v2
    - name: Use Node.js
      uses: actions/setup-node@v2
      with:
        node-version: '14'
    - name: Install dependencies
      run: npm ci
    - name: Build
      run: npm run build
    - name: Test
      run: npm run test

该配置文件会监听 master 分支上的推送事件,并执行构建和测试操作。

通过以上步骤,你可以充分利用 Nest.js 的强大功能开发高性能的 Node.js 应用,并通过合理的设计和部署,确保应用的稳健性和可维护性。

0人推荐
随时随地看视频
慕课网APP