乐闻世界logo
搜索文章和话题

Class Validator

class-validator 是一个用于装饰器和非装饰器环境下的校验库,它允许使用装饰器或手动验证来对类的属性进行校验。这个库是基于 TypeScript 编写的,广泛应用于 Node.js 项目中,尤其是与 TypeScript 和 TypeORM 结合使用时,可以有效地确保数据模型的正确性和一致性。
Class Validator
查看更多相关内容
如何在类验证器NestJS中删除自定义消息中的字段名
在NestJS中,使用类验证器(class-validator)进行数据验证时,默认情况下,错误消息会包含具体的字段名。例如,如果有一个字段名为`username`的验证规则未通过,它可能返回一个错误消息如:“username must be longer than or equal to 10 characters”。 如果希望在自定义的验证消息中去掉字段名,可以通过自定义错误消息并在其中不包含字段名来实现。这可以通过在装饰器中使用字符串模板来完成。例如,考虑以下使用`class-validator`的用户类: ```typescript import { IsLength, IsEmail } from 'class-validator'; export class CreateUserDto { @IsLength(10, 20, { message: '长度必须在10到20字符之间' }) username: string; @IsEmail({}, { message: '提供的值必须是有效的电子邮件地址' }) email: string; } ``` 在上面的例子中,我们自定义了错误消息,并去掉了字段名。这样,当`username`长度不符或`email`格式不正确时,返回的错误消息将仅显示“长度必须在10到20字符之间”和“提供的值必须是有效的电子邮件地址”,而不会显示字段名。 此外,如果需要进一步定制或动态生成错误消息(例如,根据不同的语言环境),可以考虑使用自定义验证装饰器或使用`class-validator`的回调函数功能来生成错误消息。这样可以实现更复杂和动态的验证逻辑。 例如,创建一个自定义验证器来检查字符串是否包含特定的字符,而不在消息中包含字段名: ```typescript import { registerDecorator, ValidationOptions, ValidationArguments } from 'class-validator'; function ContainsLetter(letter: string, validationOptions?: ValidationOptions) { return function (object: any, propertyName: string) { registerDecorator({ name: 'containsLetter', target: object.constructor, propertyName: propertyName, constraints: [letter], options: validationOptions, validator: { validate(value: any, args: ValidationArguments) { const [relatedLetter] = args.constraints; return typeof value === 'string' && value.includes(relatedLetter); }, defaultMessage(args: ValidationArguments) { return `必须包含字母${args.constraints[0]}`; } } }); }; } // 使用 class SomeClass { @ContainsLetter('x', { message: '必须包含字母x' }) myField: string; } ``` 这样,当`myField`不包含字母'x'时,错误消息将仅显示“必须包含字母x”,而不会提到`myField`。这种方法提供了更高的灵活性和控制能力,在实际应用中可以根据需求自由定制。
阅读 15 · 2024年8月24日 17:34
应该如何为nestjs创建响应dto?
在 NestJS 中创建响应 DTO(Data Transfer Object)是一种很好的实践,它有助于定义和管理通过网络发送的数据结构。DTO 不仅可以增强代码的可读性和维护性,还可以提供数据验证功能。以下是创建响应 DTO 的步骤和示例: ### 步骤 1:定义 DTO 结构 首先,你需要确定响应数据的结构。例如,如果你正在构建一个用户 API,返回用户详情时,你可能需要包括用户的 `id`、`name` 和 `email` 字段。 ### 步骤 2:使用类来实现 DTO 在 NestJS 中,通常使用类来实现 DTO,这有助于利用 TypeScript 的类型系统。同时,你可以使用 `class-validator` 和 `class-transformer` 这样的库来进行数据验证和转换。 **示例代码**: ```typescript import { IsNotEmpty, IsEmail, IsUUID } from 'class-validator'; export class UserResponseDto { @IsUUID() id: string; @IsNotEmpty() name: string; @IsEmail() email: string; } ``` ### 步骤 3:在服务或控制器中使用 DTO 定义好 DTO 后,可以在服务或控制器层使用它来确保响应数据的格式和数据的有效性。 **控制器中的使用示例**: ```typescript import { Controller, Get, Param } from '@nestjs/common'; import { UserService } from './user.service'; import { UserResponseDto } from './dto/user-response.dto'; @Controller('users') export class UserController { constructor(private readonly userService: UserService) {} @Get(':id') async getUser(@Param('id') id: string): Promise<UserResponseDto> { const user = await this.userService.findById(id); return new UserResponseDto(user); // 确保服务返回的数据符合 UserResponseDto 的结构 } } ``` ### 步骤 4:全局配置或局部使用管道来自动验证和转换 DTO 在 NestJS 中,你可以配置管道(Pipe)来自动处理数据验证和转换。你可以全局应用这些管道,或者仅在特定的路由上使用它们。 **局部使用管道的示例**: ```typescript import { Controller, Get, Param, UsePipes, ValidationPipe } from '@nestjs/common'; @Controller('users') export class UserController { // ... @Get(':id') @UsePipes(new ValidationPipe({ transform: true })) async getUser(@Param('id') id: string): Promise<UserResponseDto> { // ... } } ``` 通过这种方式,每当有请求调用特定路由时,NestJS 将自动验证查询参数并尝试将其转换为 DTO 类的实例,确保符合你定义的数据结构和验证规则。 ### 总结 使用响应 DTO 不仅有助于保持代码的清晰和组织性,还可以提供自动化的数据验证和转换功能,提高开发效率和应用安全性。
阅读 29 · 2024年8月24日 17:33
如何使用基于环境变量值的类验证器条件验证装饰器(@ValidateIf)
在使用类验证器进行数据验证时,有时我们希望基于环境变量的不同值来决定是否应用某个验证规则。此时,我们可以利用`class-validator`库中的`@ValidateIf`装饰器来实现条件验证。`@ValidateIf`允许我们定义一个函数,该函数返回一个布尔值,来决定是否需要对该字段进行验证。 ### 示例场景 假设我们有一个Node.js应用,其中有一个用户配置的环境变量`NODE_ENV`,它用于标识当前的运行环境(比如`development`, `production`等)。我们需要在生产环境中验证用户的邮箱地址是否真实有效,但在开发环境中可以不进行严格验证,以方便测试。 ### 代码实现 首先,确保已安装`class-validator`和`class-transformer`: ```bash npm install class-validator class-transformer ``` 然后,我们可以创建一个用户类,并使用`@ValidateIf`装饰器来根据环境变量决定是否进行邮箱验证: ```typescript import { validateOrReject, IsEmail, ValidateIf } from 'class-validator'; class User { @ValidateIf(o => process.env.NODE_ENV === 'production') @IsEmail({}, { message: '邮箱格式不正确' }) email?: string; constructor(email?: string) { this.email = email; } } async function validateUser(user: User) { try { await validateOrReject(user); console.log('验证成功!'); } catch (errors) { console.log('验证失败:', errors); } } // 示例测试 // 假设你在开发环境下运行,NODE_ENV=development const userDev = new User('test@'); validateUser(userDev); // 应该验证通过,因为我们不在生产环境 // 假设你在生产环境下运行,NODE_ENV=production const userProd = new User('test@'); validateUser(userProd); // 应该验证失败,因为邮箱格式不正确 ``` ### 注意点 1. **环境变量的管理**:在实际应用中,环境变量通常通过`.env`文件管理,并通过如`dotenv`库加载。 2. **异步验证**:`validateOrReject`函数是异步的,因此需要适当处理异步逻辑。 3. **错误处理**:示例中简单地打印了错误信息,实际应用中可能需要更细致的错误处理策略。 通过上述实现,我们可以根据不同的环境需求灵活地应用验证规则,从而使得应用在开发和生产环境中都能按预期工作。
阅读 16 · 2024年8月24日 17:33
如何允许null,但禁止undefined?
在JavaScript编程中,`null`和`undefined`都可以表示缺乏值,但它们的用途和含义有所不同。`null`通常用于表示程序员已经定义了变量,但它目前没有值。而`undefined`通常表示变量已声明但未初始化。 如果我们想要在代码中允许`null`但禁止`undefined`,我们可以通过几种方法来实现: ### 1. 类型检查 **示例**: ```javascript function processValue(value) { if (typeof value === 'undefined') { throw new Error('undefined is not allowed'); } // 继续处理value,此时value可以是null或其他任何类型 } try { processValue(undefined); } catch (error) { console.error(error.message); // 输出: undefined is not allowed } processValue(null); // 这行代码可以正常执行,因为允许null ``` ### 2. 使用TypeScript 在使用TypeScript时,我们可以设置严格的类型检查来明确区分`null`和`undefined`。 **TypeScript 配置**: 在`tsconfig.json`中启用`strictNullChecks`: ```json { "compilerOptions": { "strict": true, "strictNullChecks": true } } ``` **TypeScript 示例**: ```typescript function processValue(value: number | null) { // 这个函数接受number类型或null,但不接受undefined } processValue(null); // 正常 processValue(123); // 正常 // processValue(undefined); // TypeScript编译错误 ``` ### 3. 默认参数值 在函数参数中使用默认值可以防止`undefined`值,但允许`null`。 **示例**: ```javascript function greet(name = 'Guest') { console.log(`Hello, ${name}!`); } greet(undefined); // 输出: Hello, Guest! greet(null); // 输出: Hello, null! ``` 在上述示例中,当`undefined`作为参数传递时,它会被默认参数值`'Guest'`替代,而`null`则不会被替代。 ### 总结 通过这些方法,我们可以在JavaScript或TypeScript项目中有意识地选择允许`null`但禁止`undefined`的策略,这有助于提高代码的清晰性和健壮性。使用适当的错误处理和类型检查可以确保程序的稳定性,并减少潜在的bug。
阅读 6 · 2024年8月24日 17:30
如何在Nested DTO-NestJS中实现条件验证?
在NestJS中实现Nested DTO的条件验证通常涉及到使用`class-validator`库来进行数据验证。`class-validator`提供了一系列的装饰器,可以帮助我们实现复杂的验证逻辑。对于条件验证,我们可以使用`@ValidateIf()`装饰器来实现特定条件下的数据验证。以下是如何在Nested DTO中使用`@ValidateIf()`来实现条件验证的步骤: ### 步骤 1: 创建Nested DTO 首先,我们需要定义我们的DTO(Data Transfer Object)。假设我们有一个`Order`对象和一个`Product`对象,其中`Order`包含多个`Product`。 ```typescript import { Type } from 'class-transformer'; import { IsInt, ValidateNested, IsOptional, IsBoolean, ValidateIf } from 'class-validator'; class Product { @IsInt() id: number; @IsInt() quantity: number; } class Order { @ValidateNested({ each: true }) @Type(() => Product) products: Product[]; @IsBoolean() isGift: boolean; @IsOptional() @ValidateIf(o => o.isGift === true) @IsInt() giftWrapId: number; } ``` ### 步骤 2: 使用`@ValidateIf()`装饰器 在上面的例子中,`Order`类有一个`isGift`布尔属性和一个`giftWrapId`属性。我们只在订单是礼物(`isGift`为true)的情况下需要验证`giftWrapId`。通过使用`@ValidateIf()`装饰器,我们可以设定条件:仅当`isGift`为true时,才检查`giftWrapId`的值是否是整数。 ### 步骤 3: 在服务中使用DTO 在你的NestJS服务中,你可以使用这些DTO来处理和验证来自客户端的数据: ```typescript import { Body, Controller, Post } from '@nestjs/common'; import { Validate } from 'class-validator'; import { Order } from './dto/order.dto'; @Controller('orders') export class OrdersController { @Post() createOrder(@Body() order: Order) { // 订单逻辑处理 } } ``` 在这个控制器中,任何发送到`POST /orders`的请求体都将自动验证其结构是否符合`Order`类的定义,包括条件验证。 ### 总结 这样,我们就能根据条件验证Nested DTO中的属性了。在实际开发中,这种方法极大地提高了代码的可维护性和数据的一致性。通过`class-validator`的使用,我们可以轻松地实现复杂的验证逻辑,确保我们的应用程序可以正确地处理各种输入情况。
阅读 10 · 2024年8月24日 17:26
如何在typeform和nest.js中通过正则表达式正确设置验证
在使用Typeform和Nest.js开发应用时,使用正则表达式进行数据验证是一种有效的方式,可以确保用户输入的数据符合预期格式。下面我将分别介绍在Typeform和Nest.js中如何设置正则表达式验证。 #### 1. Typeform中设置正则表达式验证 在Typeform中,可以使用正则表达式来增强表单的验证功能。例如,如果您想要验证用户输入的是有效的邮箱地址,可以在相应的文本输入题目中设置正则表达式。 **步骤如下**: - 登录您的Typeform账号并打开您的表单。 - 选择或添加一个 `文本`问题,用以收集邮箱地址。 - 在问题设置中,找到 `Validations`选项并点击。 - 选择 `Add a new rule`,然后在弹出的条件配置中选择 `Text`。 - 在 `matches regex`字段中输入对应的邮箱验证正则表达式,如 `^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$`。 - 完成设置后保存表单。 通过这种方式,当用户输入不符合正则表达式定义的格式时,Typeform会自动提示用户重新输入,保证数据的准确性。 #### 2. Nest.js中设置正则表达式验证 在Nest.js应用中,可以使用类验证器(class-validator)库来实施正则表达式验证。例如,验证用户输入的电话号码是否符合特定格式。 **步骤如下**: - 首先,确保您的项目中已安装 `class-validator`和 `class-transformer`。 ```bash npm install class-validator class-transformer ``` - 定义DTO(Data Transfer Object),并使用 `@IsString`和 `@Matches`装饰器来应用正则表达式验证。 ```typescript import { IsString, Matches } from 'class-validator'; export class CreateUserDto { @IsString() name: string; @IsString() @Matches(/^(\+\d{1,3}[- ]?)?\d{10}$/, { message: '手机号必须是有效的格式', }) phone: string; } ``` 在这个例子中,`@Matches`装饰器用于确保 `phone`字段匹配一定的电话号码格式,如果不符合,则返回自定义的错误消息。 - 在您的Nest.js控制器中,使用这个DTO,并确保全局或局部使用了 `ValidationPipe`。 ```typescript import { Body, Controller, Post, UsePipes, ValidationPipe } from '@nestjs/common'; import { CreateUserDto } from './create-user.dto'; @Controller('users') export class UsersController { @Post() @UsePipes(new ValidationPipe()) createUser(@Body() createUserDto: CreateUserDto) { return 'User created successfully!'; } } ``` 使用 `ValidationPipe`,Nest.js会自动处理输入验证,并在数据不符合要求时抛出异常,从而保护您的应用不接收无效数据。 #### 总结 通过上述的Typeform和Nest.js中的实例,我们可以看到正则表达式是验证用户输入的强大工具。在Typeform中主要通过表单设置实现,在Nest.js中则通过配合类验证器实现数据有效性的校验。根据应用的实际需要选择合适的实现方式,可以显著提升应用的健壮性和用户体验。
阅读 12 · 2024年8月24日 17:20
如何在validationif装饰器nestjs类验证器中使用else条件?
在NestJS的类验证器(class-validator)中,`@ValidateIf()` 装饰器通常用来在某些条件下应用验证规则。如果需要在某条件不满足时应用其他验证规则(即所谓的“else”条件),我们通常需要使用另一个`@ValidateIf()`来指定这个条件的反向逻辑。 以下是一个简单的例子来说明这一点: 假设我们有一个用户注册的功能,其中用户需要提供`email`或`phoneNumber`中的至少一个,我们将使用`@ValidateIf()`来确保如果未提供`email`,则`phoneNumber`必须有效,反之亦然。 ```typescript import { IsEmail, ValidateIf, IsMobilePhone } from 'class-validator'; export class RegisterDto { @ValidateIf(o => !o.phoneNumber) @IsEmail() email?: string; @ValidateIf(o => !o.email) @IsMobilePhone() phoneNumber?: string; } ``` 在这个例子中: - 第一个`@ValidateIf()`装饰器检查`phoneNumber`是否未被提供,如果未提供,那么`email`字段必须符合邮箱格式。 - 第二个`@ValidateIf()`装饰器检查`email`是否未被提供,如果未提供,那么`phoneNumber`字段必须是一个有效的手机号。 这样,我们就间接实现了“如果...则...否则...”的逻辑,确保了用户至少提供了其中一个联系方式,并且所提供的信息是有效的。这种方法在处理复杂的条件验证逻辑时非常有用,能够灵活地对数据进行校验。
阅读 10 · 2024年8月24日 17:02
如何使用类验证器和swagg-netjs显示数组数据的属性
在使用NestJS框架进行应用开发时,经常需要对输入数据进行验证以确保数据的正确性和安全性。使用类验证器(如class-validator)和Swagger(通过@nestjs/swagger模块)可以很方便地实现这一功能,并且可以清晰地展示API文档。下面我将通过一个例子来说明如何在NestJS项目中使用类验证器和Swagger来验证和显示数组数据的属性。 ### Step 1: 设置项目基础 首先,确保你的NestJS项目已经安装了`class-validator`和`@nestjs/swagger`这两个包。如果还没有安装,可以通过以下命令进行安装: ```bash npm install class-validator class-transformer @nestjs/swagger swagger-ui-express ``` ### Step 2: 创建DTO类 在NestJS中,我们通常会使用DTO(Data Transfer Object)类来定义和传输数据结构。在这个例子中,假设我们需要验证一个用户提交的订单信息,订单中包含多个商品项,每个商品项包括商品ID和数量: ```typescript import { Type } from 'class-transformer'; import { IsArray, ValidateNested, IsInt, IsPositive, Min } from 'class-validator'; import { ApiProperty } from '@nestjs/swagger'; class ProductItem { @ApiProperty({ description: '商品ID' }) @IsInt() @IsPositive() productId: number; @ApiProperty({ description: '商品数量', minimum: 1 }) @IsInt() @Min(1) quantity: number; } export class CreateOrderDto { @ApiProperty({ type: [ProductItem], description: '订单中的商品项' }) @IsArray() @ValidateNested({ each: true }) @Type(() => ProductItem) items: ProductItem[]; } ``` 在上述代码中,`ProductItem` 类定义了商品项的数据结构,使用`IsInt`和`IsPositive`确保`productId`是正整数,`IsInt`和`Min(1)`确保`quantity`至少为1。`CreateOrderDto` 类中的`items`属性标记为数组,并且使用`ValidateNested`和`Type`确保数组中的每个元素都符合`ProductItem`的结构。 ### Step 3: 控制器中使用DTO 在相应的控制器中,我们接收和验证客户端提交的数据: ```typescript import { Body, Controller, Post } from '@nestjs/common'; import { ApiCreatedResponse } from '@nestjs/swagger'; import { CreateOrderDto } from './dto/create-order.dto'; @Controller('orders') export class OrdersController { @Post() @ApiCreatedResponse({ description: '创建订单', type: CreateOrderDto }) createOrder(@Body() createOrderDto: CreateOrderDto) { // 在这里处理订单创建逻辑 return 'Order created successfully!'; } } ``` 在`createOrder`方法中,通过`@Body()`装饰器自动将请求体中的数据映射到`CreateOrderDto`实例,并进行验证。 ### Step 4: 配置Swagger 确保在NestJS模块中启用了Swagger,通常在`main.ts`文件中配置: ```typescript import { NestFactory } from '@nestjs/core'; import { AppModule } from './app.module'; import { SwaggerModule, DocumentBuilder } from '@nestjs/swagger'; async function bootstrap() { const app = await NestFactory.create(AppModule); const config = new DocumentBuilder() .setTitle('Example API') .setDescription('The example API description') .setVersion('1.0') .build(); const document = SwaggerModule.createDocument(app, config); SwaggerModule.setup('api', app, document); await app.listen(3000); } bootstrap(); ``` 通过以上步骤,我们不仅实现了对请求数据的有效验证,还通过Swagger自动生成了API文档,使得API的使用和测试变得更加方便。
阅读 15 · 2024年8月24日 17:00
如何在NestJS中使用另一个包中的DTO类进行验证?
在NestJS中,如果想要使用来自另一个包中的DTO类进行验证,可以通过以下步骤实现: ### 步骤 1:安装必要的包 首先,确保你的NestJS项目中安装了`class-validator`和`class-transformer`这两个包。这两个包通常用于DTO验证。 ```bash npm install class-validator class-transformer ``` ### 步骤 2:导入DTO类 确保你有访问权限导入来自外部包的DTO。假设这个外部DTO类名为`ExternalDTO`,位于名为`external-package`的npm包中。 ```typescript import { ExternalDTO } from 'external-package'; ``` ### 步骤 3:在Controller中使用DTO 在你的Controller中,使用装饰器`@Body()`来捕获传入的请求体,并应用`ExternalDTO`类来进行自动验证。 ```typescript import { Controller, Post, Body } from '@nestjs/common'; import { ExternalDTO } from 'external-package'; @Controller('your-controller') export class YourController { @Post() async create(@Body() externalDto: ExternalDTO) { // 这里externalDto已经是一个验证后的对象 // ... return '操作成功!'; } } ``` ### 步骤 4:使用Pipes进行验证 确保在`main.ts`文件或者你的controller局部中设置了`ValidationPipe`,这样NestJS才会自动应用`class-validator`的验证规则。 ```typescript import { NestFactory } from '@nestjs/core'; import { AppModule } from './app.module'; import { ValidationPipe } from '@nestjs/common'; async function bootstrap() { const app = await NestFactory.create(AppModule); app.useGlobalPipes(new ValidationPipe()); await app.listen(3000); } bootstrap(); ``` 或者在特定的controller或route中使用: ```typescript import { Body, Controller, Post, UsePipes, ValidationPipe } from '@nestjs/common'; import { ExternalDTO } from 'external-package'; @Controller('your-controller') export class YourController { @Post() @UsePipes(new ValidationPipe()) async create(@Body() externalDto: ExternalDTO) { // ... } } ``` ### 示例场景 假设你正在开发一个电商平台,需要从一个共享的用户管理服务导入用户的DTO。这个服务提供了一个`UserDTO`类,该类定义了用户名和密码等字段。你可以按照上述步骤导入`UserDTO`并在你的用户注册接口中使用它进行验证。 ### 总结 通过上述步骤,你可以轻松地在NestJS项目中利用来自其他包的DTO进行请求数据的结构和类型验证,确保数据的正确性和安全性。
阅读 7 · 2024年8月24日 17:00
NestJS如何在@Query对象中转换数组
在NestJS中,如果您想在`@Query`对象中处理和转换数组类型的数据,通常有一些方法可以实现这一点。这主要取决于客户端如何发送查询参数以及您想如何在服务器端接收这些参数。下面是一些具体的方法和例子: ### 方法1: 使用逗号分隔的值 客户端可以通过发送逗号分隔的值来发送数组,例如:`?ids=1,2,3`。在服务器端,您可以使用`@Query`装饰器来接收这个字符串并手动将其转换为数组。 ```typescript import { Controller, Get, Query } from '@nestjs/common'; @Controller('items') export class ItemsController { @Get() findAll(@Query('ids') ids: string): string[] { const idsArray = ids.split(',').map(id => parseInt(id, 10)); return idsArray; // [1, 2, 3] } } ``` ### 方法2:直接使用数组格式 客户端可以直接发送数组格式,例如:`?ids[]=1&ids[]=2&ids[]=3`。NestJS可以自动将这些值转换为数组。 ```typescript import { Controller, Get, Query } from '@nestjs/common'; @Controller('items') export class ItemsController { @Get() findAll(@Query('ids') ids: number[]): number[] { return ids; // [1, 2, 3] } } ``` ### 方法3:自定义管道使用 如果您需要进行更复杂的转换或验证,您也可以创建自定义的管道来处理查询参数。 ```typescript import { PipeTransform, Injectable, ArgumentMetadata } from '@nestjs/common'; @Injectable() export class ParseIntArrayPipe implements PipeTransform { transform(value: string, metadata: ArgumentMetadata): number[] { return value.split(',').map(val => parseInt(val, 10)); } } @Controller('items') export class ItemsController { @Get() findAll(@Query('ids', ParseIntArrayPipe) ids: number[]): number[] { return ids; // [1, 2, 3] } } ``` ### 方法4:使用类验证器 如果您需要使用类和验证器进行更严格的数据处理,您可以使用类验证器来定义和验证输入数据。 ```typescript import { Type } from 'class-transformer'; import { IsArray, IsNumber, ValidateNested } from 'class-validator'; class IdsDto { @IsArray() @IsNumber({}, { each: true }) @Type(() => Number) ids: number[]; } @Controller('items') export class ItemsController { @Get() findAll(@Query() query: IdsDto): number[] { return query.ids; // [1, 2, 3] } } ``` 这些方法可以根据您的具体需求选择使用。每种方法都有其优势,例如,方法1和3允许您在没有额外依赖的情况下进行简单的转换和验证,而方法4则提供了更强的类型检查和数据验证。
阅读 10 · 2024年8月24日 16:59