如何在Laravel中实现数据校验

laravel中实现数据校验,核心在于使用内置验证器或推荐的表单请求类来保障数据完整性、安全性和业务逻辑正确性。1. 控制器内快速校验适用于简单场景,通过request()->validate()直接校验并自动处理错误重定向;2. validator facade提供更精细控制,适用于非http请求或需手动响应错误的场景;3. form request classes将校验逻辑与控制器分离,提升代码结构和复用性,是复杂场景推荐方式。数据校验能防止安全漏洞、确保数据一致性、提升用户体验并强制执行业务规则。常见规则如required、email、unique等可满足多数需求,自定义规则可通过规则对象、闭包或扩展validator实现。校验失败时,web应用通过session闪存错误并回显旧输入,api则返回422 json错误响应,同时支持自定义错误消息和字段名称提升提示友好度。

如何在Laravel中实现数据校验

在Laravel中实现数据校验,核心在于利用其内置的强大验证器(Validator)或更推荐的表单请求(Form Request)来确保传入应用的数据符合预设的规则,从而保障数据的完整性、安全性和业务逻辑的正确执行。

解决方案

Laravel的数据校验体系非常灵活,提供了多种途径来处理传入的数据。我个人在项目中,会根据校验的复杂度和复用性来选择不同的方式。

1. 控制器内快速校验 (request()->validate())

这是最简单直接的方式,适用于简单的表单或API请求。你可以在控制器方法中直接调用request()助手函数上的validate()方法。

// app/Http/Controllers/PostController.phpnamespace AppHttpControllers;use IlluminateHttpRequest;class PostController extends Controller{    public function store(Request $request)    {        $validatedData = $request->validate([            'title' => 'required|unique:posts|max:255',            'body' => 'required',            'tags' => 'array',            'tags.*' => 'string|max:50', // 校验数组中的每个元素        ], [            'title.required' => '标题是必填的,老铁!',            'title.unique' => '这个标题已经有人用了,换一个吧。',        ]);        // 校验通过,数据会包含在 $validatedData 中        // 你的业务逻辑...        return redirect('/posts')->with('success', '文章发布成功!');    }}

如果校验失败,Laravel会自动将用户重定向回上一个页面,并把错误信息和旧的输入数据闪存到session中,这对于传统的Web应用非常方便。

2. 使用Validator Facade

当你需要更精细地控制校验过程,比如在非HTTP请求场景下校验数据,或者需要手动处理错误响应(例如为API返回JSON错误),Validator Facade就显得很有用了。

// app/Http/Controllers/UserController.phpnamespace AppHttpControllers;use IlluminateHttpRequest;use IlluminateSupportFacadesValidator;class UserController extends Controller{    public function update(Request $request, $id)    {        $validator = Validator::make($request->all(), [            'name' => 'required|string|max:255',            'email' => 'required|email|unique:users,email,' . $id, // 忽略当前用户ID            'password' => 'nullable|string|min:8|confirmed',        ]);        if ($validator->fails()) {            // 对于API请求,通常返回JSON错误            if ($request->expectsJson()) {                return response()->json(['errors' => $validator->errors()], 422);            }            // 对于Web请求,可以手动重定向并带上错误            return redirect()->back()->withErrors($validator)->withInput();        }        // 校验通过,获取干净的数据        $validatedData = $validator->validated();        // 更新用户逻辑...        // User::find($id)->update($validatedData);        return response()->json(['message' => '用户更新成功!']);    }}

这种方式给了你更多的主动权,你可以决定校验失败后如何响应。

3. Form Request Classes (推荐)

对于更复杂、需要复用或逻辑上更清晰的校验场景,我强烈推荐使用表单请求类。它将校验规则和授权逻辑从控制器中分离出来,让控制器保持精简,只专注于业务逻辑。

首先,创建一个表单请求:

php artisan make:request StoreBlogPostRequest

然后,在生成的请求类中定义规则和授权逻辑:

// app/Http/Requests/StoreBlogPostRequest.phpnamespace AppHttpRequests;use IlluminateFoundationHttpFormRequest;class StoreBlogPostRequest extends FormRequest{    /**     * 确定用户是否有权发出此请求。     */    public function authorize(): bool    {        // 比如,只有管理员或文章作者本人才能发布/编辑        // return auth()->user()->isAdmin();        // 或者简单地允许所有登录用户        return true; // 默认情况下,如果所有人都允许,就返回 true    }    /**     * 获取适用于请求的验证规则。     *     * @return array     */    public function rules(): array    {        return [            'title' => 'required|unique:posts|max:255',            'body' => 'required',            'category_id' => 'required|exists:categories,id',            'published_at' => 'nullable|date',        ];    }    /**     * 获取自定义的验证消息。     *     * @return array     */    public function messages(): array    {        return [            'title.required' => '文章标题不能为空哦。',            'title.unique' => '这个标题太热门了,已经被占用了。',            'body.required' => '内容是文章的灵魂,不能没有呀!',        ];    }    /**     * 获取自定义属性名称,用于替换消息中的 :attribute 占位符。     *     * @return array     */    public function attributes(): array    {        return [            'title' => '文章标题',            'body' => '文章内容',            'category_id' => '文章分类',        ];    }}

最后,在控制器方法中类型提示这个请求类:

// app/Http/Controllers/PostController.phpnamespace AppHttpControllers;use AppHttpRequestsStoreBlogPostRequest; // 引入你的表单请求class PostController extends Controller{    public function store(StoreBlogPostRequest $request)    {        // 校验和授权都在 StoreBlogPostRequest 中完成了        // 校验通过后,直接获取校验过的数据        $validated = $request->validated();        // 你的业务逻辑...        // Post::create($validated);        return redirect('/posts')->with('success', '文章发布成功!');    }}

当控制器方法类型提示了StoreBlogPostRequest后,Laravel会自动在方法执行前运行该请求类的authorize()rules()方法。如果授权失败,会返回403响应;如果校验失败,也会自动重定向或返回JSON错误,非常优雅。

为什么我们需要在Laravel中进行数据校验?

数据校验,这玩意儿在任何应用开发里,说它是基石都不过分。在Laravel里,它更是被设计得如此顺手,以至于你几乎没有理由不去用它。那到底为啥非得校验不可呢?

首先,安全是头等大事。想象一下,如果用户能随意提交任何格式的数据,你的数据库可能瞬间就乱套了,SQL注入、XSS攻击,这些安全漏洞简直是敞开大门欢迎。校验就像一道防火墙,把不合规的数据挡在外面,大大降低了潜在的风险。我见过不少新手项目,为了图快直接把用户输入往数据库里塞,结果没多久就出问题了,教训很深刻。

其次,数据完整性和一致性。你希望你的email字段里存的真的是邮箱格式,而不是一串乱码吧?你希望age字段真的是数字,并且在合理的年龄范围内吧?校验就是为了确保进入系统的数据符合你的业务模型和数据类型要求。这不仅仅是为了数据库好看,更是为了后续的业务逻辑能正确运行。如果数据本身就是错的,那后面所有的计算、展示都可能跟着出错,甚至引发连锁反应。

再来,提升用户体验。当用户提交表单,如果哪里填错了,你给他一个友好的提示:“邮箱格式不正确”或者“密码至少需要8位”,而不是一个冰冷的500错误页面,这体验简直是天壤之别。Laravel的校验系统自带错误消息和旧数据闪存,让错误提示变得非常自然和人性化,用户能很快知道问题出在哪儿,并进行修正。

最后,强制业务逻辑。有些时候,校验不仅仅是格式问题,它还承载着业务规则。比如“一个商品库存不能为负数”、“订单金额不能低于某个值”。通过在校验规则中加入minmaxunique甚至自定义规则,你可以直接在数据进入业务层之前就强制执行这些业务约束,避免了在业务逻辑代码中散落大量的if-else判断,让代码更清晰、更易维护。所以,校验这事儿,真不是可有可无的“好习惯”,它根本就是构建健壮应用的核心环节。

Laravel数据校验的常见规则和自定义校验

Laravel的校验规则非常丰富,几乎涵盖了日常开发中所有能想到的场景。但有时候,内置规则也无法满足所有奇葩的业务需求,这时候自定义校验就派上用场了。

常见规则速览:

required: 字段必须存在且不为空。string / integer / numeric / boolean / array: 校验数据类型。min:value / max:value: 长度(字符串、数组)或数值的最小/最大限制。email: 校验是否为有效邮箱格式。unique:table,column,except_id,id_column: 校验字段值在指定表中是否唯一。这在用户注册、更新资料时特别常用,比如更新用户邮箱时要忽略当前用户自己的邮箱。confirmed: 校验字段(通常是密码)与其_confirmation字段是否一致。exists:table,column: 校验字段值在指定表的指定列中是否存在。这对于外键关联非常有用,比如确保提交的category_id确实存在于categories表中。url / active_url: 校验是否为有效URL,后者还会尝试检查DNS记录。date / date_format:format: 校验日期格式。in:foo,bar,...: 校验字段值是否在给定列表中。regex:pattern: 使用正则表达式进行校验。image / mimes:jpeg,png / max:size: 针对文件上传的校验。

这些只是冰山一角,Laravel文档里有完整的列表,值得花时间去翻阅。我通常会把这些常用规则记在脑子里,碰到具体需求时能快速反应过来用哪个。

自定义校验的艺术:

当内置规则不够用时,Laravel提供了几种方式来自定义校验逻辑,其中我最常用也最推荐的是自定义规则对象

1. 自定义规则对象 (推荐)

这是Laravel 5.5+ 引入的特性,它让自定义规则变得非常OOP,可读性高,且易于复用。

php artisan make:rule IsStrongPassword

然后编辑这个规则类:

// app/Rules/IsStrongPassword.phpnamespace AppRules;use Closure;use IlluminateContractsValidationValidationRule;class IsStrongPassword implements ValidationRule{    /**     * 运行验证规则。     */    public function validate(string $attribute, mixed $value, Closure $fail): void    {        // 密码必须包含至少一个大写字母,一个小写字母,一个数字,一个特殊字符,且长度至少8位        if (!preg_match('/^(?=.*[a-z])(?=.*[A-Z])(?=.*d)(?=.*[@$!%*?&])[A-Za-zd@$!%*?&]{8,}$/', $value)) {            $fail("':attribute' 必须包含至少一个大写字母、一个小写字母、一个数字和一个特殊字符,且长度至少为8位。");        }    }}

在你的校验规则中使用它:

use AppRulesIsStrongPassword;// ...public function rules(): array{    return [        'password' => ['required', 'string', new IsStrongPassword()],    ];}

这种方式把复杂的校验逻辑封装起来,让你的rules()方法保持简洁。

2. 使用闭包 (Closure)

对于一次性的、不需复用的简单自定义逻辑,可以直接在规则数组中使用闭包。

use IlluminateValidationRule;// ...public function rules(): array{    return [        'product_code' => [            'required',            'string',            function ($attribute, $value, $fail) {                if (strlen($value)  10) {                    $fail('产品代码长度必须在5到10个字符之间。');                }                // 假设这里有一些复杂的外部API调用来验证产品代码                // if (! $this->checkProductCodeWithExternalApi($value)) {                //     $fail('产品代码无效,请检查。');                // }            },        ],    ];}

闭包非常灵活,可以直接访问到校验的字段名、值以及$fail回调,非常适合即时、上下文相关的校验。

3. 扩展Validator Facade (Validator::extend)

这种方式通常用于创建全局可用的、不依赖于特定上下文的自定义规则。它需要你在AppServiceProviderboot方法中注册。

// app/Providers/AppServiceProvider.phpuse IlluminateSupportFacadesValidator;public function boot(): void{    Validator::extend('foo', function ($attribute, $value, $parameters, $validator) {        return $value == 'foo';    }, 'The :attribute must be "foo".');}

然后在规则中使用'foo'。不过,自从有了自定义规则对象,我个人很少再用Validator::extend了,因为规则对象在组织和管理上更胜一筹。

掌握这些自定义校验的方式,意味着你可以应对任何复杂的业务校验场景,让你的应用既灵活又健壮。

如何优雅地处理Laravel校验失败的错误信息?

处理校验失败的错误信息,这不仅仅是技术问题,更是用户体验的关键一环。一个好的错误提示能引导用户修正错误,而一个糟糕的提示可能让用户直接放弃。Laravel在这方面做得相当不错,提供了非常方便的机制。

1. Web应用中的错误回显

对于传统的Web应用,当request()->validate()或Form Request校验失败时,Laravel会自动将用户重定向回上一个页面,并将错误信息闪存到Session中。在Blade模板中,你可以非常方便地访问这些错误。

    @csrf    
@error('title')
{{ $message }}
@enderror
@error('body')
{{ $message }}
@enderror
@if ($errors->any())
    @foreach ($errors->all() as $error)
  • {{ $error }}
  • @endforeach
@endif

old('field_name'):这个助手函数能帮你把上次提交的旧数据重新填充到表单字段中,用户就不用重新输入一遍了,非常贴心。@error('field_name') ... @enderror:这是Blade的指令,它只会显示指定字段的第一个错误信息。如果你想展示所有错误,可以用$errors->any()$errors->all()来遍历。

我个人觉得这种方式在开发初期非常高效,不用写额外的错误处理逻辑。

2. API中的JSON错误响应

对于API接口,我们不能简单地重定向。当校验失败时,通常需要返回一个标准的JSON格式错误响应,并附带HTTP状态码422(Unprocessable Entity)。

当你使用Validator::make()手动校验时:

use IlluminateSupportFacadesValidator;public function storeApi(Request $request){    $validator = Validator::make($request->all(), [        'name' => 'required|string|max:255',        'email' => 'required|email|unique:users',    ]);    if ($validator->fails()) {        return response()->json(['errors' => $validator->errors()], 422);    }    // 校验通过,业务逻辑...    // User::create($request->all());    return response()->json(['message' => '用户创建成功!'], 201);}

$validator->errors()会返回一个MessageBag实例,它在转换为JSON时会变成一个对象,键是字段名,值是该字段的错误消息数组。

{    "errors": {        "name": [            "名称是必填的。"        ],        "email": [            "邮箱格式不正确。",            "该邮箱已被注册。"        ]    }}

如果你使用的是Form Request,Laravel会自动帮你处理这个,当请求头中包含Accept: application/json时,它会自动返回上述格式的JSON响应。这省去了很多重复的代码。

3. 自定义错误消息

默认的错误消息有时可能过于通用,不够友好。你可以为每个规则和字段自定义错误消息。

validate()Validator::make()中:在规则数组后面传递一个消息数组。键的格式是field.rule

$request->validate([    'title' => 'required|unique:posts',], [    'title.required' => '文章标题不能为空!',    'title.unique' => '这个标题已经被人抢先了,换个新的吧。',]);

在Form Request类中:在Form Request类中定义messages()方法。

// app/Http/Requests/StoreBlogPostRequest.phppublic function messages(): array{    return [        'title.required' => '请务必填写文章标题。',        'title.unique' => '此标题已被占用,请尝试其他。',        'body.required' => '文章内容是空的,这可不行。',    ];}

自定义属性名称 (attributes()方法):有时候错误消息中的:attribute占位符会显示原始的字段名(比如user_name),这可能不够友好。你可以在Form Request中定义attributes()方法来为字段提供更友好的显示名称。

// app/Http/Requests/StoreBlogPostRequest.phppublic function attributes(): array{    return [        'title' => '文章标题',        'body' => '文章内容',        'category_id' => '所属分类',    ];}

这样,默认的错误消息(如The :attribute field is required.)就会显示成“文章标题字段是必填的。”,用户体验会

以上就是如何在Laravel中实现数据校验的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/160046.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月5日 14:10:35
下一篇 2025年12月5日 14:24:50

相关推荐

  • CSS mask属性无法获取图片:为什么我的图片不见了?

    CSS mask属性无法获取图片 在使用CSS mask属性时,可能会遇到无法获取指定照片的情况。这个问题通常表现为: 网络面板中没有请求图片:尽管CSS代码中指定了图片地址,但网络面板中却找不到图片的请求记录。 问题原因: 此问题的可能原因是浏览器的兼容性问题。某些较旧版本的浏览器可能不支持CSS…

    2025年12月24日
    900
  • Uniapp 中如何不拉伸不裁剪地展示图片?

    灵活展示图片:如何不拉伸不裁剪 在界面设计中,常常需要以原尺寸展示用户上传的图片。本文将介绍一种在 uniapp 框架中实现该功能的简单方法。 对于不同尺寸的图片,可以采用以下处理方式: 极端宽高比:撑满屏幕宽度或高度,再等比缩放居中。非极端宽高比:居中显示,若能撑满则撑满。 然而,如果需要不拉伸不…

    2025年12月24日
    400
  • 如何让小说网站控制台显示乱码,同时网页内容正常显示?

    如何在不影响用户界面的情况下实现控制台乱码? 当在小说网站上下载小说时,大家可能会遇到一个问题:网站上的文本在网页内正常显示,但是在控制台中却是乱码。如何实现此类操作,从而在不影响用户界面(UI)的情况下保持控制台乱码呢? 答案在于使用自定义字体。网站可以通过在服务器端配置自定义字体,并通过在客户端…

    2025年12月24日
    800
  • 如何在地图上轻松创建气泡信息框?

    地图上气泡信息框的巧妙生成 地图上气泡信息框是一种常用的交互功能,它简便易用,能够为用户提供额外信息。本文将探讨如何借助地图库的功能轻松创建这一功能。 利用地图库的原生功能 大多数地图库,如高德地图,都提供了现成的信息窗体和右键菜单功能。这些功能可以通过以下途径实现: 高德地图 JS API 参考文…

    2025年12月24日
    400
  • 如何使用 scroll-behavior 属性实现元素scrollLeft变化时的平滑动画?

    如何实现元素scrollleft变化时的平滑动画效果? 在许多网页应用中,滚动容器的水平滚动条(scrollleft)需要频繁使用。为了让滚动动作更加自然,你希望给scrollleft的变化添加动画效果。 解决方案:scroll-behavior 属性 要实现scrollleft变化时的平滑动画效果…

    2025年12月24日
    000
  • 如何为滚动元素添加平滑过渡,使滚动条滑动时更自然流畅?

    给滚动元素平滑过渡 如何在滚动条属性(scrollleft)发生改变时为元素添加平滑的过渡效果? 解决方案:scroll-behavior 属性 为滚动容器设置 scroll-behavior 属性可以实现平滑滚动。 html 代码: click the button to slide right!…

    2025年12月24日
    500
  • 为什么设置 `overflow: hidden` 会导致 `inline-block` 元素错位?

    overflow 导致 inline-block 元素错位解析 当多个 inline-block 元素并列排列时,可能会出现错位显示的问题。这通常是由于其中一个元素设置了 overflow 属性引起的。 问题现象 在不设置 overflow 属性时,元素按预期显示在同一水平线上: 不设置 overf…

    2025年12月24日 好文分享
    400
  • 网页使用本地字体:为什么 CSS 代码中明明指定了“荆南麦圆体”,页面却仍然显示“微软雅黑”?

    网页中使用本地字体 本文将解答如何将本地安装字体应用到网页中,避免使用 src 属性直接引入字体文件。 问题: 想要在网页上使用已安装的“荆南麦圆体”字体,但 css 代码中将其置于第一位的“font-family”属性,页面仍显示“微软雅黑”字体。 立即学习“前端免费学习笔记(深入)”; 答案: …

    2025年12月24日
    000
  • 如何选择元素个数不固定的指定类名子元素?

    灵活选择元素个数不固定的指定类名子元素 在网页布局中,有时需要选择特定类名的子元素,但这些元素的数量并不固定。例如,下面这段 html 代码中,activebar 和 item 元素的数量均不固定: *n *n 如果需要选择第一个 item元素,可以使用 css 选择器 :nth-child()。该…

    2025年12月24日
    200
  • 使用 SVG 如何实现自定义宽度、间距和半径的虚线边框?

    使用 svg 实现自定义虚线边框 如何实现一个具有自定义宽度、间距和半径的虚线边框是一个常见的前端开发问题。传统的解决方案通常涉及使用 border-image 引入切片图片,但是这种方法存在引入外部资源、性能低下的缺点。 为了避免上述问题,可以使用 svg(可缩放矢量图形)来创建纯代码实现。一种方…

    2025年12月24日
    100
  • 如何让“元素跟随文本高度,而不是撑高父容器?

    如何让 元素跟随文本高度,而不是撑高父容器 在页面布局中,经常遇到父容器高度被子元素撑开的问题。在图例所示的案例中,父容器被较高的图片撑开,而文本的高度没有被考虑。本问答将提供纯css解决方案,让图片跟随文本高度,确保父容器的高度不会被图片影响。 解决方法 为了解决这个问题,需要将图片从文档流中脱离…

    2025年12月24日
    000
  • 为什么我的特定 DIV 在 Edge 浏览器中无法显示?

    特定 DIV 无法显示:用户代理样式表的困扰 当你在 Edge 浏览器中打开项目中的某个 div 时,却发现它无法正常显示,仔细检查样式后,发现是由用户代理样式表中的 display none 引起的。但你疑问的是,为什么会出现这样的样式表,而且只针对特定的 div? 背后的原因 用户代理样式表是由…

    2025年12月24日
    200
  • inline-block元素错位了,是为什么?

    inline-block元素错位背后的原因 inline-block元素是一种特殊类型的块级元素,它可以与其他元素行内排列。但是,在某些情况下,inline-block元素可能会出现错位显示的问题。 错位的原因 当inline-block元素设置了overflow:hidden属性时,它会影响元素的…

    2025年12月24日
    000
  • 为什么 CSS mask 属性未请求指定图片?

    解决 css mask 属性未请求图片的问题 在使用 css mask 属性时,指定了图片地址,但网络面板显示未请求获取该图片,这可能是由于浏览器兼容性问题造成的。 问题 如下代码所示: 立即学习“前端免费学习笔记(深入)”; icon [data-icon=”cloud”] { –icon-cl…

    2025年12月24日
    200
  • 为什么使用 inline-block 元素时会错位?

    inline-block 元素错位成因剖析 在使用 inline-block 元素时,可能会遇到它们错位显示的问题。如代码 demo 所示,当设置了 overflow 属性时,a 标签就会错位下沉,而未设置时却不会。 问题根源: overflow:hidden 属性影响了 inline-block …

    2025年12月24日
    000
  • 如何利用 CSS 选中激活标签并影响相邻元素的样式?

    如何利用 css 选中激活标签并影响相邻元素? 为了实现激活标签影响相邻元素的样式需求,可以通过 :has 选择器来实现。以下是如何具体操作: 对于激活标签相邻后的元素,可以在 css 中使用以下代码进行设置: li:has(+li.active) { border-radius: 0 0 10px…

    2025年12月24日
    100
  • 为什么我的 CSS 元素放大效果无法正常生效?

    css 设置元素放大效果的疑问解答 原提问者在尝试给元素添加 10em 字体大小和过渡效果后,未能在进入页面时看到放大效果。探究发现,原提问者将 CSS 代码直接写在页面中,导致放大效果无法触发。 解决办法如下: 将 CSS 样式写在一个单独的文件中,并使用 标签引入该样式文件。这个操作与原提问者观…

    2025年12月24日
    000
  • 如何模拟Windows 10 设置界面中的鼠标悬浮放大效果?

    win10设置界面的鼠标移动显示周边的样式(探照灯效果)的实现方式 在windows设置界面的鼠标悬浮效果中,光标周围会显示一个放大区域。在前端开发中,可以通过多种方式实现类似的效果。 使用css 使用css的transform和box-shadow属性。通过将transform: scale(1.…

    2025年12月24日
    200
  • 为什么我的 em 和 transition 设置后元素没有放大?

    元素设置 em 和 transition 后不放大 一个 youtube 视频中展示了设置 em 和 transition 的元素在页面加载后会放大,但同样的代码在提问者电脑上没有达到预期效果。 可能原因: 问题在于 css 代码的位置。在视频中,css 被放置在单独的文件中并通过 link 标签引…

    2025年12月24日
    100
  • 为什么我的 Safari 自定义样式表在百度页面上失效了?

    为什么在 Safari 中自定义样式表未能正常工作? 在 Safari 的偏好设置中设置自定义样式表后,您对其进行测试却发现效果不同。在您自己的网页中,样式有效,而在百度页面中却失效。 造成这种情况的原因是,第一个访问的项目使用了文件协议,可以访问本地目录中的图片文件。而第二个访问的百度使用了 ht…

    2025年12月24日
    000

发表回复

登录后才能评论
关注微信