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

PHP用imagestring加文本水印要注意坐标和字体大小
文本水印本质是把字符串画到图像上,核心函数是 imagestring(GD库),但它只支持内置的 5 种位图字体(1–5),不支持 TTF 字体。如果直接传入中文或自定义字号,会显示乱码或完全不出现。
实操建议:
-
imagestring的坐标是左下角起点,不是左上角,y 值需预留字体高度(比如字体大小为 5 时,实际高度约 13px) - 中文必须换用
imagettftext,且要确保 PHP 编译时启用了--with-freetype,并传入真实存在的 .ttf 文件路径 - 文字颜色要用
imagecolorallocate创建,不能直接传 RGB 数组;半透明需配合imagealphablending和imagesavealpha - 示例:在右下角加白色小字水印
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');,且确保无任何输出(包括空格、BOM、echo) - JPEG 质量控制用
imagejpeg($im, null, 90),第三个参数范围 0–100,低于 75 会明显失真
复杂点在于:同一套逻辑要兼容 JPG/PNG/WebP 输入,就得先用 getimagesize 判断类型,再选对应 imagecreatefromxxx 函数——漏掉一种,就卡死在 false 返回上。











