0

0

Laravel 文件数组总大小验证:自定义规则实现

碧海醫心

碧海醫心

发布时间:2025-12-02 12:06:26

|

960人浏览过

|

来源于php中文网

原创

Laravel 文件数组总大小验证:自定义规则实现

本教程详细介绍了如何在 laravel 中验证上传文件数组的总大小。由于 laravel 内置验证器主要针对单个文件大小,对于整个文件数组的合计大小验证,需要通过创建自定义验证规则来实现。文章将指导您完成自定义规则的创建、逻辑实现,并将其集成到表单请求的验证规则中,确保所有上传文件的总大小符合预期限制。

在 Laravel 应用中处理文件上传时,我们经常需要对单个文件的大小、类型等进行验证。然而,当用户上传多个文件(作为文件数组)时,有时不仅需要限制每个文件的大小,还需要限制所有文件合计的总大小。Laravel 的内置验证规则,例如 max,通常只作用于单个文件,无法直接验证文件数组的聚合大小。本文将指导您如何通过创建自定义验证规则来解决这一问题。

1. 理解 Laravel 内置文件验证的局限性

考虑以下典型的 Laravel 验证规则,它限制了每个 images 数组中的图片文件类型和最大大小(1000 KB):

public function rules()
{
    return [
        // ... 其他验证规则
        'images.*' => 'mimes:jpg,jpeg,png|max:1000' // 验证每个图片文件不超过 1000 KB
    ];
}

这里的 images.* 规则确保了数组中的每个文件都符合要求。但如果用户上传了 10 张图片,每张 900 KB,那么总大小将达到 9000 KB (9 MB),这可能仍然超出了我们对整个上传批次的总大小限制。为了实现对总大小的验证,我们需要一个自定义的解决方案。

2. 创建自定义验证规则

Laravel 提供了强大的自定义验证规则机制,允许我们定义复杂的验证逻辑。我们将创建一个名为 ArraySize 的自定义规则来验证文件数组的总大小。

首先,使用 Artisan 命令生成一个新的验证规则:

php artisan make:rule ArraySize

这会在 app/Rules 目录下创建一个 ArraySize.php 文件。

3. 实现自定义规则逻辑

打开 app/Rules/ArraySize.php 文件,并根据以下示例代码修改其内容。这个规则的核心逻辑是遍历文件数组,累加每个文件的大小,然后将总大小与我们设定的最大值进行比较。

Quinvio AI
Quinvio AI

AI辅助下快速创建视频,虚拟代言人

下载
maxSize = $maxSize;
    }

    /**
     * 判断验证规则是否通过。
     *
     * @param  string  $attribute 字段名称
     * @param  mixed  $value     待验证的值 (文件数组)
     * @return bool
     */
    public function passes($attribute, $value): bool
    {
        $totalSize = 0;

        // 确保 $value 是一个数组
        if (!is_array($value)) {
            return false;
        }

        foreach ($value as $file) {
            // 确保数组中的每个元素都是 UploadedFile 实例
            if (!$file instanceof UploadedFile) {
                return false;
            }
            $totalSize += $file->getSize(); // getSize() 返回字节数
        }

        // 将总字节数转换为 KB,并判断是否小于允许的最大值
        // 如果总大小小于最大允许值,则验证通过 (返回 true)
        return ($totalSize / 1024) < $this->maxSize;
    }

    /**
     * 获取验证失败时返回的错误消息。
     *
     * @return string
     */
    public function message(): string
    {
        return sprintf('所有图片的总大小必须小于 %d KB。', $this->maxSize);
    }
}

代码解析:

  • __construct(int $maxSize): 构造函数接收一个 $maxSize 参数,用于设定允许的最大总大小,单位为 KB。
  • passes($attribute, $value):
    • 初始化 $totalSize 为 0,用于累加文件大小。
    • 首先检查 $value 是否为数组,如果不是,则验证失败。
    • 遍历 $value 数组,对每个元素:
      • 检查它是否是 Illuminate\Http\UploadedFile 的实例。这确保我们处理的是实际上传的文件对象。
      • 通过 $file->getSize() 获取当前文件的大小(单位为字节),并累加到 $totalSize。
    • 最后,将 $totalSize(字节)转换为 KB ($totalSize / 1024),并与构造函数中传入的 $maxSize 进行比较。如果总大小小于 $maxSize,则 passes 方法返回 true(验证通过),否则返回 false(验证失败)。
  • message(): 定义了当验证失败时返回的错误消息。使用 sprintf 可以动态地将 $maxSize 插入到消息中,提供更具体的信息。

4. 在验证中使用自定义规则

现在,您可以在您的表单请求(Form Request)或控制器中引入并使用这个自定义规则。

首先,在文件顶部引入 ArraySize 规则:

use App\Rules\ArraySize;

然后,在您的 rules() 方法中,将 ArraySize 规则应用于 images 字段。请注意,images 字段代表整个文件数组,而 images.* 字段则代表数组中的每个单独文件。

public function rules()
{
    return [
        'title'               => 'required|max:125',
        'quantity'            => 'required|numeric|min:0',
        'retail_price'        => 'required|numeric|min:0',
        'diamond_shape_id'    => 'required',
        'diamond_cut_id'      => 'required',
        'diamond_color_id'    => 'required',
        'diamond_clarity_id'  => 'required',
        'carat_weight'        => 'required',
        'diamond_polish_id'   => 'required',
        'diamond_symmetry_id' => 'required',
        'video_url'           => ['url', new EmbeddableUrl],
        'images.*'            => 'mimes:jpg,jpeg,png|max:1000', // 验证单个文件
        'images'              => ['required', new ArraySize(30000)], // 验证文件数组总大小,30000 KB = 30 MB
    ];
}

在上述示例中,new ArraySize(30000) 表示我们希望所有上传图片的合计大小不得超过 30000 KB (即 30 MB)。同时,images.* 规则仍然会验证每个单独的文件不超过 1000 KB。

5. 注意事项与最佳实践

  • 单位一致性:在 ArraySize 规则中,$maxSize 参数的单位是 KB。确保在实例化规则时传入的值是正确的 KB 单位。
  • 错误消息本地化:如果您的应用支持多语言,您可以通过在 resources/lang/{locale}/validation.php 文件中添加自定义消息,或者在规则的 message() 方法中返回 __('validation.custom.images.array_size') 来实现错误消息的本地化。
  • 与其他规则结合:将 images.* (针对单个文件) 和 images (针对文件数组) 的验证规则结合使用,可以提供更全面的验证。
  • 性能考量:对于非常大的文件数组或文件,计算总大小可能会消耗一些时间。在极端情况下,可以考虑在文件上传前进行前端预校验,以减少不必要的服务器请求。
  • 空数组处理:在上述 rules() 中,我们为 images 字段添加了 required 规则。这意味着如果 images 数组为空,它将首先被 required 规则捕获。如果您允许空数组,但仍然想验证非空数组的总大小,可以移除 required 规则。

总结

通过创建自定义验证规则,我们成功解决了 Laravel 中对上传文件数组总大小进行验证的需求。这种方法不仅功能强大,而且高度可定制,能够适应各种复杂的业务逻辑。掌握自定义验证规则的技巧,将大大提升您在 Laravel 中处理复杂数据验证的能力。

相关专题

更多
php文件怎么打开
php文件怎么打开

打开php文件步骤:1、选择文本编辑器;2、在选择的文本编辑器中,创建一个新的文件,并将其保存为.php文件;3、在创建的PHP文件中,编写PHP代码;4、要在本地计算机上运行PHP文件,需要设置一个服务器环境;5、安装服务器环境后,需要将PHP文件放入服务器目录中;6、一旦将PHP文件放入服务器目录中,就可以通过浏览器来运行它。

2707

2023.09.01

php怎么取出数组的前几个元素
php怎么取出数组的前几个元素

取出php数组的前几个元素的方法有使用array_slice()函数、使用array_splice()函数、使用循环遍历、使用array_slice()函数和array_values()函数等。本专题为大家提供php数组相关的文章、下载、课程内容,供大家免费下载体验。

1668

2023.10.11

php反序列化失败怎么办
php反序列化失败怎么办

php反序列化失败的解决办法检查序列化数据。检查类定义、检查错误日志、更新PHP版本和应用安全措施等。本专题为大家提供php反序列化相关的文章、下载、课程内容,供大家免费下载体验。

1527

2023.10.11

php怎么连接mssql数据库
php怎么连接mssql数据库

连接方法:1、通过mssql_系列函数;2、通过sqlsrv_系列函数;3、通过odbc方式连接;4、通过PDO方式;5、通过COM方式连接。想了解php怎么连接mssql数据库的详细内容,可以访问下面的文章。

974

2023.10.23

php连接mssql数据库的方法
php连接mssql数据库的方法

php连接mssql数据库的方法有使用PHP的MSSQL扩展、使用PDO等。想了解更多php连接mssql数据库相关内容,可以阅读本专题下面的文章。

1444

2023.10.23

html怎么上传
html怎么上传

html通过使用HTML表单、JavaScript和PHP上传。更多关于html的问题详细请看本专题下面的文章。php中文网欢迎大家前来学习。

1235

2023.11.03

PHP出现乱码怎么解决
PHP出现乱码怎么解决

PHP出现乱码可以通过修改PHP文件头部的字符编码设置、检查PHP文件的编码格式、检查数据库连接设置和检查HTML页面的字符编码设置来解决。更多关于php乱码的问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

1529

2023.11.09

php文件怎么在手机上打开
php文件怎么在手机上打开

php文件在手机上打开需要在手机上搭建一个能够运行php的服务器环境,并将php文件上传到服务器上。再在手机上的浏览器中输入服务器的IP地址或域名,加上php文件的路径,即可打开php文件并查看其内容。更多关于php相关问题,详情请看本专题下面的文章。php中文网欢迎大家前来学习。

1307

2023.11.13

云朵浏览器入口合集
云朵浏览器入口合集

本专题整合了云朵浏览器入口合集,阅读专题下面的文章了解更多详细地址。

20

2026.01.20

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 9万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 9万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号