0

0

PHP 文件上传大小验证的正确实现方法

花韻仙語

花韻仙語

发布时间:2026-03-07 11:55:09

|

635人浏览过

|

来源于php中文网

原创

PHP 文件上传大小验证的正确实现方法

本文详解如何在 php 中为多文件上传添加可靠的最大尺寸校验,修复常见逻辑错误(如错误的数组索引、忽略上传错误码、误判空文件等),确保在调用 phpmailer 发送邮件前彻底拦截超限附件。

本文详解如何在 php 中为多文件上传添加可靠的最大尺寸校验,修复常见逻辑错误(如错误的数组索引、忽略上传错误码、误判空文件等),确保在调用 phpmailer 发送邮件前彻底拦截超限附件。

在使用 PHP 处理多文件上传并集成 PHPMailer 发送带附件邮件时,文件大小验证失效是一个高频问题。典型表现是:代码看似包含 if ($_FILES['uploaded-file']['size'][$i] > $max_size) 判断,但超限文件仍被上传并附加到邮件中——根本原因往往不是逻辑缺失,而是对 $_FILES 超全局数组结构的误读与校验时机的错位。

✅ 正确理解 $_FILES 数组结构

PHP 的文件上传数据以二维数组形式组织在 $_FILES['field_name'] 下,每个子键(如 'name', 'size', 'tmp_name', 'error', 'type')本身已是独立的一维数组,对应每个上传文件。错误写法 $_FILES['uploaded-file']['name']['size'][$i] 试图在字符串(文件名)上取 'size' 键,必然失败且静默忽略(PHP 发出 Notice 但不中断执行)。

✅ 正确访问方式:

光子AI
光子AI

AI电商服饰商拍平台

下载
$fileSize = $_FILES['uploaded-file']['size'][$i];        // ✔️ 正确:size 是顶层键
$fileName = $_FILES['uploaded-file']['name'][$i];         // ✔️ 正确
$tmpPath  = $_FILES['uploaded-file']['tmp_name'][$i];     // ✔️ 正确
$errorCode = $_FILES['uploaded-file']['error'][$i];       // ✔️ 必须检查!

✅ 完整、健壮的校验流程(含示例代码)

以下是一个生产就绪的校验片段,已整合进 PHPMailer 流程,并修复原文所有关键缺陷:

立即学习PHP免费学习笔记(深入)”;

<?php
use PHPMailer\PHPMailer\PHPMailer;
use PHPMailer\PHPMailer\Exception;
require 'vendor/autoload.php';

// 配置参数
$maxFileSize = 2 * 1024 * 1024; // 2MB(注意:原文 1204 是笔误,应为 1024)
$uploadDir = 'uploads/';

// 创建邮件实例
$mail = new PHPMailer(true);
$mail->isSMTP();
$mail->Host = 'smtp.example.com';
$mail->SMTPAuth = true;
$mail->Username = 'your@email.com';
$mail->Password = 'your-app-password';
$mail->setFrom($_POST['email'] ?? '', $_POST['name'] ?? '');
$mail->addAddress('recipient@example.com');

// 构建邮件内容(略)

// ✅ 关键:严格校验文件上传状态与大小
$attachmentsValid = true;
$uploadedFiles = [];

if (isset($_FILES['uploaded-file']) && is_array($_FILES['uploaded-file']['name'])) {
    $fileCount = count($_FILES['uploaded-file']['name']);

    for ($i = 0; $i < $fileCount; $i++) {
        $name = $_FILES['uploaded-file']['name'][$i];
        $tmpName = $_FILES['uploaded-file']['tmp_name'][$i];
        $size = $_FILES['uploaded-file']['size'][$i];
        $error = $_FILES['uploaded-file']['error'][$i];

        // ❌ 跳过无效上传(空文件、客户端取消、超限等)
        if ($error !== UPLOAD_ERR_OK) {
            $attachmentsValid = false;
            error_log("File upload error #{$error} for '{$name}'");
            continue;
        }

        // ❌ 跳过空文件(size === 0)
        if ($size === 0) {
            $attachmentsValid = false;
            error_log("Empty file detected: '{$name}'");
            continue;
        }

        // ✅ 核心:校验文件大小
        if ($size > $maxFileSize) {
            $attachmentsValid = false;
            echo "<p style='color:red;'>Błąd: Plik '{$name}' przekracza dozwolony rozmiar (max. 2 MB).</p>";
            continue;
        }

        // ✅ 安全生成唯一文件名,防止路径遍历 & 冲突
        $safeName = basename($name);
        $uniqueName = uniqid() . '_' . $safeName;
        $targetPath = $uploadDir . $uniqueName;

        // ✅ 移动临时文件(必须在 size 校验后!)
        if (!move_uploaded_file($tmpName, $targetPath)) {
            $attachmentsValid = false;
            error_log("Failed to move uploaded file: {$name}");
            continue;
        }

        $uploadedFiles[] = $targetPath;
    }
}

// ❌ 任一文件校验失败,立即终止发送
if (!$attachmentsValid) {
    echo "<p style='color:red;'>Wysyłka została anulowana z powodu błędów załączników.</p>";
    exit;
}

// ✅ 所有附件校验通过,添加到邮件
foreach ($uploadedFiles as $filePath) {
    if (file_exists($filePath)) {
        $mail->addAttachment($filePath);
    }
}

// 发送邮件
try {
    $mail->send();
    header('Location: sent.html');
    exit;
} catch (Exception $e) {
    error_log("Mailer error: " . $mail->ErrorInfo);
    echo "Wystąpił błąd podczas wysyłki: " . $mail->ErrorInfo;
}

⚠️ 关键注意事项与最佳实践

  • 永远先检查 $_FILES['field']['error'][$i]:UPLOAD_ERR_OK(值为 0)是唯一安全继续的信号。其他错误(如 UPLOAD_ERR_INI_SIZE, UPLOAD_ERR_FORM_SIZE)表明文件已被 PHP 层拒绝,此时 size 可能为 0 或不可信。
  • isset($_FILES['field']) ≠ 有有效文件:该检查仅确认表单提交了该字段,但可能为空数组或含错误项。必须结合 is_array() 和循环内 error 码判断。
  • 避免硬编码路径与不安全文件名:使用 basename() 过滤原始文件名,结合 uniqid() 生成唯一目标名,杜绝路径遍历(../../etc/passwd)和并发覆盖风险。
  • 清理临时文件:若校验失败,已 move_uploaded_file 的文件需手动 unlink();本例中建议在 try/catch 外统一清理,或改用 tempnam() + 显式删除。
  • 前端校验仅为辅助:HTML 的 accept 和 JS 的 file.size 检查可提升用户体验,但绝不可替代服务端验证——恶意用户可轻易绕过。

通过以上结构化校验,你将获得一个真正可靠的文件尺寸防线:超限文件在进入邮件发送流程前即被精准识别、友好提示并彻底阻断,同时兼顾安全性、可维护性与错误可观测性。

相关文章

PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

845

2023.08.22

scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

451

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

373

2023.10.25

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

718

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

219

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1561

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

649

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

1168

2024.03.22

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

23

2026.03.06

热门下载

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

精品课程

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

共137课时 | 13.1万人学习

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

共6课时 | 11.3万人学习

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

共13课时 | 1.0万人学习

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

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