PHP创建文件如何添加水印_文本水印与图片水印实现【详解】

2次阅读

php图像水印需分文本与图片两类:文本用imagestring(仅支持5种位图字体)或imagettftext(中文需freetype及正确.ttf路径);图片用imagecopymerge(支持透明度)并注意gd扩展、路径、内存释放与header设置。

PHP创建文件如何添加水印_文本水印与图片水印实现【详解】

PHP用imagestring加文本水印要注意坐标和字体大小

文本水印本质是把字符串画到图像上,核心函数是 imagestringGD库),但它只支持内置的 5 种位图字体(1–5),不支持 TTF 字体。如果直接传入中文或自定义字号,会显示乱码或完全不出现。

实操建议:

  • imagestring 的坐标是左下角起点,不是左上角,y 值需预留字体高度(比如字体大小为 5 时,实际高度约 13px)
  • 中文必须换用 imagettftext,且要确保 PHP 编译时启用了 --with-freetype,并传入真实存在的 .ttf 文件路径
  • 文字颜色要用 imagecolorallocate 创建,不能直接传 RGB 数组;半透明需配合 imagealphablendingimagesavealpha
  • 示例:在右下角加白色小字水印
font $font = 'simhei.ttf'; // 中文需绝对路径或 web 可读路径 $size = 12; $angle = 0; $x = imagesx($im) - 80; $y = imagesy($im) - 10; $color = imagecolorallocate($im, 255, 255, 255); imagettftext($im, $size, $angle, $x, $y, $color, $font, '© example.com');

PHP用imagecopy叠加图片水印要控制透明度和位置

图片水印本质是把一张小图(如 logo.png)复制粘贴到原图上,关键函数是 imagecopy 或更灵活的 imagecopymerge(支持透明度)。

常见错误现象:

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

  • 水印发黑或全白——没调用 imagealphablending($watermark, false)imagesavealpha($watermark, true) 保留 PNG 透明通道
  • 水印被裁剪——目标图尺寸小于水印图,或 imagecopy 的宽高参数写反了(顺序是:目标图、源图、目标x、目标y、源x、源y、宽、高)
  • 位置算错——例如想右下角对齐,却用 imagesx($im) - imagesx($watermark),但忘了减去边距

推荐用 imagecopymerge($im, $watermark, $dst_x, $dst_y, 0, 0, $w, $h, 30),最后一个参数是透明度(0–100),比 imagecopy 更直观。

GD扩展未启用或字体路径错误导致水印完全不显示

这是最常被忽略的环境问题:代码逻辑没问题,但水印就是不出来。根本原因往往不是 PHP 写错了,而是 GD 库或字体资源不可用。

排查步骤:

  • 运行 phpinfo(),确认 “gd” 模块已加载,且 “FreeType Support” 显示为 enabled
  • 检查字体文件路径是否可读:file_exists($font) && is_readable($font),Web 路径不能直接用于 imagettftext,必须是服务器本地路径
  • 临时关闭错误报告(error_reporting(0))可能掩盖致命警告,建议开启 display_errors=On 并查看 PHP 错误日志
  • GD 不支持 WebP 输入?是的——若原图是 WebP,imagecreatefromwebp 需 PHP ≥ 7.1 且 GD 编译时带 WebP 支持,否则会返回 false 导致后续操作失败

生成带水印文件后记得释放内存和设置正确 header

GD 图像资源不释放会导致内存泄漏,尤其批量处理时容易崩溃;而输出前没设 header,浏览器会把二进制当乱码文本显示。

关键动作:

  • 每张图处理完必须调用 imagedestroy($im)imagedestroy($watermark)
  • 保存到文件用 imagepng($im, $output_path),注意 PNG 默认不压缩透明通道,加 imagesavealpha($im, true) 才能保真
  • 如果是即时输出(非保存),务必在 imagepng($im) 前加:header('Content-Type: image/png');,且确保无任何输出(包括空格、bomecho
  • JPEG 质量控制用 imagejpeg($im, NULL, 90),第三个参数范围 0–100,低于 75 会明显失真

复杂点在于:同一套逻辑要兼容 JPG/PNG/WebP 输入,就得先用 getimagesize 判断类型,再选对应 imagecreatefromxxx 函数——漏掉一种,就卡死在 false 返回上。

text=ZqhQzanResources