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

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免费学习笔记(深入)”;

<?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 SendGridMailMail;  // 假设 $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_

© 版权声明
THE END
喜欢就支持一下吧
点赞15 分享