0

0

如何在 PHP 中正确下载并保存 Blob 图像(如 QR 码)

霞舞

霞舞

发布时间:2026-02-27 15:50:03

|

896人浏览过

|

来源于php中文网

原创

如何在 PHP 中正确下载并保存 Blob 图像(如 QR 码)

本文详解如何使用 curl 从 api 获取二进制图像响应(如 jpg 格式 qr 码),安全写入服务器本地文件,并通过前端触发浏览器下载,解决因文件模式错误、路径缺失或响应处理不当导致的图像损坏问题。

本文详解如何使用 curl 从 api 获取二进制图像响应(如 jpg 格式 qr 码),安全写入服务器本地文件,并通过前端触发浏览器下载,解决因文件模式错误、路径缺失或响应处理不当导致的图像损坏问题。

在 PHP 中调用返回图像二进制流(Blob)的 REST API(例如生成 QR 码的服务)时,常见错误是将响应体直接以文本模式写入文件,导致图像数据被意外截断或编码污染。核心问题在于:fopen(..., "w") 默认以文本模式打开文件,而图像属于二进制数据,必须使用 "wb"(write binary)模式;此外,还需确保目标目录存在、权限合理,并妥善处理文件名与路径。

以下是一个完整、健壮的实现方案:

HyperWrite
HyperWrite

AI写作助手帮助你创作内容更自信

下载

✅ 正确步骤与关键修复点

  1. 启用二进制安全写入:使用 fopen($path, "wb") 替代 "w";
  2. 预清理输出目录(可选但推荐):避免旧文件堆积,降低存储成本;
  3. 验证 cURL 响应有效性:检查 HTTP 状态码与响应长度,防止空/错误内容写入;
  4. 服务端保存 + 前端触发下载:分离「服务器存储」与「用户下载」逻辑,提升安全性与可控性。

? 完整示例代码

<?php
function cleanQrCodeImagesFolder($directoryUrl) {
    if (is_dir($directoryUrl)) {
        $files = glob($directoryUrl . "*");
        foreach ($files as $file) {
            if (is_file($file)) {
                unlink($file);
            }
        }
    }
}

function generateQrCode($message, $dataId) {
    // ✅ 1. 配置输出路径(确保目录存在)
    $directoryOutput = __DIR__ . "/../img/qrcode";
    if (!is_dir($directoryOutput)) {
        mkdir($directoryOutput, 0755, true);
    }

    $urlOutput = $directoryOutput . "/image_{$dataId}.jpg";

    // ✅ 2. 发起 cURL 请求(含错误处理)
    $url = 'https://backend.com/qr/create-image';
    $data = ['message' => $message];
    $ch = curl_init($url);
    curl_setopt($ch, CURLOPT_POST, 1);
    curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode($data));
    curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
    curl_setopt($ch, CURLOPT_HTTPHEADER, ['Content-Type: application/json']);
    curl_setopt($ch, CURLOPT_HEADER, 0); // 不返回响应头
    curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true);
    curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false); // 生产环境请启用证书验证

    $result = curl_exec($ch);
    $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
    curl_close($ch);

    // ✅ 3. 校验响应
    if ($httpCode !== 200 || empty($result)) {
        throw new RuntimeException("API request failed: HTTP {$httpCode}, response empty.");
    }

    // ✅ 4. 以二进制模式写入文件
    $file = fopen($urlOutput, "wb");
    if (!$file) {
        throw new RuntimeException("Failed to open file for writing: {$urlOutput}");
    }
    fwrite($file, $result);
    fclose($file);

    // ✅ 5. 前端触发下载(纯前端逻辑,不依赖 PHP 输出)
    $relativePath = str_replace($_SERVER['DOCUMENT_ROOT'], '', $urlOutput);
    ?>
    <script type="text/javascript">
        const a = document.createElement('a');
        a.href = <?= json_encode($relativePath, JSON_UNESCAPED_SLASHES | JSON_HEX_TAG | JSON_HEX_AMP) ?>;
        a.download = 'qrcode_<?= htmlspecialchars($dataId) ?>.jpg';
        document.body.appendChild(a);
        a.click();
        document.body.removeChild(a);
    </script>
    <?php
}
?>

⚠️ 注意事项与最佳实践

  • 目录权限:确保 ../img/qrcode/ 目录对 Web 服务器用户(如 www-data 或 apache)具有读写权限(推荐 0755);
  • SSL 安全:生产环境中务必移除 CURLOPT_SSL_VERIFYPEER => false,并配置正确的 CA 证书;
  • 大文件处理:若图像较大(>2MB),需检查 upload_max_filesize、post_max_size 和 memory_limit 配置;
  • 前端兼容性 在部分旧版 Safari 或跨域场景下可能失效,此时建议改用 fetch() + Blob URL 方案;
  • 替代方案(纯服务端下载):若需直接向客户端输出图像(不存服务器),可省略 fopen/fwrite,改用:
    header('Content-Type: image/jpeg');
    header('Content-Disposition: attachment; filename="qrcode.jpg"');
    echo $result;
    exit;

通过以上结构化实现,你不仅能稳定获取并保存 API 返回的二进制图像,还能兼顾可维护性、安全性与用户体验。

相关文章

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不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
curl_exec
curl_exec

curl_exec函数是PHP cURL函数列表中的一种,它的功能是执行一个cURL会话。给大家总结了一下php curl_exec函数的一些用法实例,这个函数应该在初始化一个cURL会话并且全部的选项都被设置后被调用。他的返回值成功时返回TRUE, 或者在失败时返回FALSE。

452

2023.06.14

linux常见下载安装工具
linux常见下载安装工具

linux常见下载安装工具有APT、YUM、DNF、Snapcraft、Flatpak、AppImage、Wget、Curl等。想了解更多linux常见下载安装工具相关内容,可以阅读本专题下面的文章。

182

2023.10.30

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

427

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

599

2023.08.10

apache是什么意思
apache是什么意思

Apache是Apache HTTP Server的简称,是一个开源的Web服务器软件。是目前全球使用最广泛的Web服务器软件之一,由Apache软件基金会开发和维护,Apache具有稳定、安全和高性能的特点,得益于其成熟的开发和广泛的应用实践,被广泛用于托管网站、搭建Web应用程序、构建Web服务和代理等场景。本专题为大家提供了Apache相关的各种文章、以及下载和课程,希望对各位有所帮助。

418

2023.08.23

apache启动失败
apache启动失败

Apache启动失败可能有多种原因。需要检查日志文件、检查配置文件等等。想了解更多apache启动的相关内容,可以阅读本专题下面的文章。

937

2024.01.16

Java 流式处理与 Apache Kafka 实战
Java 流式处理与 Apache Kafka 实战

本专题专注讲解 Java 在流式数据处理与消息队列系统中的应用,系统讲解 Apache Kafka 的基础概念、生产者与消费者模型、Kafka Streams 与 KSQL 流式处理框架、实时数据分析与监控,结合实际业务场景,帮助开发者构建 高吞吐量、低延迟的实时数据流管道,实现高效的数据流转与处理。

119

2026.02.04

http500解决方法
http500解决方法

http500解决方法有检查服务器日志、检查代码错误、检查服务器配置、检查文件和目录权限、检查资源不足、更新软件版本、重启服务器或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

482

2023.11.09

Golang 并发编程模型与工程实践:从语言特性到系统性能
Golang 并发编程模型与工程实践:从语言特性到系统性能

本专题系统讲解 Golang 并发编程模型,从语言级特性出发,深入理解 goroutine、channel 与调度机制。结合工程实践,分析并发设计模式、性能瓶颈与资源控制策略,帮助将并发能力有效转化为稳定、可扩展的系统性能优势。

2

2026.02.27

热门下载

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

精品课程

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

共137课时 | 12.6万人学习

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号