AI编程助手
AI免费问答

PHP 中无文件操作实现附件发送与数据存储

霞舞   2025-08-23 14:56   285浏览 原创

php 中无文件操作实现附件发送与数据存储

本教程详细阐述如何在 PHP 中避免创建临时文件,通过内存操作直接处理 Base64 编码的 PDF 数据。我们将探讨如何从 XML 中提取 Base64 字符串,将其解码并作为附件通过 SendGrid 发送邮件,同时将原始 Base64 数据存储到数据库,从而提升安全性、性能并简化代码。

引言:告别临时文件

在传统的 PHP 应用开发中,当需要处理文件内容(如从字符串生成 PDF 并作为邮件附件发送,或存储到数据库)时,开发者常倾向于先将数据写入临时文件,处理后再删除。这种方法虽然直观,但存在诸多弊端:

  1. 安全隐患敏感数据在磁盘上短暂存在,可能被未授权访问或在系统崩溃时遗留。
  2. 性能开销:频繁的磁盘 I/O 操作(写入和读取)会显著降低应用程序的响应速度,尤其在高并发场景下。
  3. 资源管理:需要额外的逻辑来确保临时文件被正确删除,否则可能导致磁盘空间浪费或文件句柄泄露。
  4. 代码复杂性:引入文件路径管理、文件权限等问题,增加代码的维护成本。

为了解决这些问题,现代 PHP 开发推荐采用内存处理方式,直接在变量中操作数据,避免不必要的磁盘交互。

核心策略:内存中的数据流处理

本教程将以一个具体场景为例:从一个包含 Base64 编码 PDF 的 XML 字符串中提取数据,将其作为邮件附件发送,并同时将 Base64 字符串存储到数据库。整个过程将完全在内存中完成,无需创建任何临时文件。

1. 数据提取:从 XML 中获取 Base64 字符串

首先,我们需要一个函数来从给定的 XML 结构中提取出 <GraphicImage> 标签内的 Base64 编码字符串。

<?php

/**
 * 从包含 <GraphicImage> 标签的字符串中提取内容。
 * 假设 <GraphicImage> 标签内的内容是 Base64 编码的字符串。
 *
 * @param string $string 原始字符串,例如 XML 片段。
 * @return array 提取出的内容数组。
 */
function multiSplit($string) {
    $output = [];
    // 按 <GraphicImage> 分割字符串
    $cols = explode("<GraphicImage>", $string);

    foreach ($cols as $col) {
        // 确保只处理包含 </GraphicImage> 的部分
        if (strpos($col, "</GraphicImage>") !== false) {
            // 按 </GraphicImage> 分割,取第一个元素即为标签内的内容
            $dashcols = explode("</GraphicImage>", $col);
            $output[] = $dashcols[0];
        }
    }
    return $output;
}

// 假设 $output 变量包含类似 '<root><GraphicImage>JVBERi0xLjQKJdPr6eEKMSAwIG9iag...</GraphicImage></root>' 的 XML 字符串
// 实际应用中,$output 可能来自文件读取、API 响应等
// $output_xml_string = '<root><GraphicImage>JVBERi0xLjQKJdPr6eEKMSAwIG9iag0KPDw...实际Base64编码PDF内容...</GraphicImage><OtherTag>...</OtherTag></root>';
// $DHL_extracted = multiSplit($output_xml_string);
// $base64_pdf_string = $DHL_extracted[1]; // 假设 Base64 字符串在第二个匹配项中

注意事项

  • multiSplit 函数返回的数组中,$DHL[1](或根据实际 XML 结构调整索引)应包含纯净的 Base64 编码 PDF 字符串。
  • 切勿在此步骤使用 print_r($DHL[1], true)。print_r 是用于调试的,它会将变量信息格式化为字符串,导致 Base64 编码失效。我们需要的只是原始的 Base64 字符串。

2. 内存中的 Base64 解码与编码

一旦获取了纯净的 Base64 编码 PDF 字符串,我们就可以在内存中对其进行解码,得到原始的二进制 PDF 内容。

// 假设 $base64_pdf_string 已经包含了从 XML 中提取出的纯净 Base64 编码 PDF 字符串
// 例如:$base64_pdf_string = $DHL_extracted[1];

// 解码 Base64 字符串以获取原始的 PDF 二进制内容
$pdf_decoded_content = base64_decode($base64_pdf_string);

// 此时,$pdf_decoded_content 变量中存储的就是 PDF 文件的原始二进制数据,无需写入磁盘。

3. 邮件附件的内存发送 (SendGrid 示例)

许多邮件发送库(如 SendGrid、PHPMailer 等)都支持直接从内存中的变量添加附件,而无需提供文件路径。

<?php
// ... (接上文代码)

// 确保 SendGrid 库已加载
require 'vendor/autoload.php';
use SendGrid\Mail\Mail;

// 假设 $submission_id, $EMAIL 等变量已定义
$envoi = $submission_id . '_envoi';
$filename2 = $envoi . '.pdf'; // 附件的建议文件名

$email = new Mail();
$email->setFrom("sender@example.com", "Sender Name"); // 替换为你的发件人信息
$email->addTo($EMAIL, "Recipient Name"); // 替换为收件人信息
$email->setSubject("您的 DHL 标签");
$email->addContent("text/html", "<strong>您好!您的 DHL 标签已附在邮件中。</strong>");

// 将原始 PDF 二进制内容再次 Base64 编码,以符合 SendGrid addAttachment 的要求
// SendGrid 的 addAttachment 方法通常期望附件内容是 Base64 编码的字符串
$file_encoded_for_attachment = base64_encode($pdf_decoded_content);

// 添加附件,直接使用内存中的 Base64 编码数据
$email->addAttachment(
    $file_encoded_for_attachment, // Base64 编码的附件内容
    "application/pdf",             // MIME 类型
    $filename2,                    // 附件文件名
    "attachment"                   // 内容处置方式
);

$sendgrid = new \SendGrid('SG.YOUR_SENDGRID_API_KEY'); // 替换为你的 SendGrid API Key

try {
    $response = $sendgrid->send($email);
    // 可以在此处检查 $response->statusCode() 进行错误处理或日志记录
    // 例如:if ($response->statusCode() >= 200 && $response->statusCode() < 300) { echo "邮件发送成功"; }
} catch (Exception $e) {
    error_log('发送邮件时捕获到异常: ' . $e->getMessage());
    // 根据实际需求进行错误处理
}

注意:addAttachment 方法的具体参数和期望的数据格式可能因不同的邮件库而异。请查阅你所使用的库的文档。SendGrid 期望附件内容本身是 Base64 编码的。

4. 数据库中存储附件数据

将附件的 Base64 编码字符串直接存储到数据库中是一种常见的做法。这避免了文件存储的复杂性,并允许将附件内容与相关记录一同管理。

<?php
// ... (接上文代码)

// 假设 $base64_pdf_string (从 XML 中提取的原始 Base64 字符串)
// 以及 $submission_id, $formID, $NOM, $ADRESSE, $TELEPHONE, $COMMERCIAL_INVOICE 等变量已定义

try {
    // 连接数据库
    // 替换为你的数据库连接信息
    $db = new PDO('mysql:dbname=jotform; host=localhost', 'user', 'password');
    $db->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION); // 设置错误模式为抛出异常

    // 准备 SQL 插入语句
    // 注意:`label_envoi` 列应能存储较长的字符串,例如 `LONGTEXT` 类型
    $sql = 'INSERT INTO DHL (submission_id, formID, identite, email, adresse, telephone, label_envoi, commercial_invoice)
            VALUES (:submission_id, :formID, :NOM, :EMAIL, :ADRESSE, :TELEPHONE, :label_envoi, :COMMERCIAL_INVOICE)';

    // 预处理查询
    $query = $db->prepare($sql);

    // 执行查询,绑定参数
    $query->execute(array(
        ':submission_

php免费学习视频:立即学习
踏上前端学习之旅,开启通往精通之路!从前端基础到项目实战,循序渐进,一步一个脚印,迈向巅峰!

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