当前位置: 首页 > news >正文

php 实现:给图片加文字水印,图片水印,压缩图片

演示环境:
1、windows10
2、phpstudy
3、php7.4

一、案例演示:

在这里插入图片描述
在这里插入图片描述

二、素材准备

1、准备一张原始图片
2、准备一张水印图片(透明底图的最好)
3、字体库(windows系统自带的字体库,路径在:C:\Windows\Fonts)
4、开启GD库
在这里插入图片描述
在这里插入图片描述

三、图片添加水印

1、文字水印封装类 FontWatermark.class.php

<?phpclass FontWatermark
{private $sourceImagePath;private $fontPath;private $fontSize;private $watermarkText;private $outputImagePath;public function __construct($sourceImagePath, $fontPath, $fontSize, $watermarkText, $outputImagePath){$this->sourceImagePath = $sourceImagePath;$this->fontPath = realpath($fontPath);  // 获取字体文件的绝对路径$this->fontSize = $fontSize;$this->watermarkText = $watermarkText;$this->outputImagePath = $outputImagePath;}public function addFontWatermark(){// 加载源图片$sourceImage = imagecreatefromjpeg($this->sourceImagePath);if (!$sourceImage) {die('无法加载源图片');}// 获取源图片的宽高$sourceWidth = imagesx($sourceImage);$sourceHeight = imagesy($sourceImage);// 确保字体文件存在if (!file_exists($this->fontPath)) {die('字体文件不存在: ' . $this->fontPath);}// 设置文字颜色 (白色)$white = imagecolorallocate($sourceImage, 255, 255, 255);// 设置文字阴影颜色 (黑色)$black = imagecolorallocate($sourceImage, 0, 0, 0);// 计算文字水印的边界框$textBox = imagettfbbox($this->fontSize, 0, $this->fontPath, $this->watermarkText);if ($textBox === false) {die('无法计算文字边界框');}$textWidth = $textBox[2] - $textBox[0];$textHeight = $textBox[1] - $textBox[7];// 设置文字水印的位置 (左上角)$destX = 10;$destY = $textHeight + 10;// 绘制阴影文字if (imagettftext($sourceImage, $this->fontSize, 0, $destX + 2, $destY + 2, $black, $this->fontPath, $this->watermarkText) === false) {die('无法绘制阴影文字');}// 绘制水印文字if (imagettftext($sourceImage, $this->fontSize, 0, $destX, $destY, $white, $this->fontPath, $this->watermarkText) === false) {die('无法绘制水印文字');}// 保存加水印后的图片if (!imagejpeg($sourceImage, $this->outputImagePath)) {die('无法保存加水印后的图片');}// 释放内存imagedestroy($sourceImage);echo "图片水印已添加并保存到: " . $this->outputImagePath;echo "<br>";echo "<img src='" . $this->outputImagePath . "' alt='文字水印' width='400px;' height='300px;'>";}
}
?>

2、调用文字水印类

<?php
require "FontWatermark.class.php";
// 使用示例
$watermark = new FontWatermark('1111.jpg',                // 源图片路径'msyh.ttc', // 字体路径80,                        // 字体大小'测试水印',                 // 水印文字'font_watermark.jpg' // 输出图片路径
);$watermark->addFontWatermark();?>

四、图片添加图片水印

1、图片水印封装类 WatermarkImage.class.php

<?php
class WatermarkImage
{private $sourceImagePath;private $watermarkImagePath;private $outputImagePath;private $sourceImage;private $watermarkImage;public function __construct($sourceImagePath, $watermarkImagePath, $outputImagePath){$this->sourceImagePath = $sourceImagePath;$this->watermarkImagePath = $watermarkImagePath;$this->outputImagePath = $outputImagePath;}private function loadImages(){// 加载源图片$this->sourceImage = @imagecreatefromjpeg($this->sourceImagePath);if (!$this->sourceImage) {die('无法加载源图片: ' . $this->sourceImagePath);}// 加载水印图片$this->watermarkImage = @imagecreatefrompng($this->watermarkImagePath);if (!$this->watermarkImage) {imagedestroy($this->sourceImage);die('无法加载水印图片: ' . $this->watermarkImagePath);}}private function addWatermark(){// 获取源图片和水印图片的宽高$sourceWidth = imagesx($this->sourceImage);$sourceHeight = imagesy($this->sourceImage);$watermarkWidth = imagesx($this->watermarkImage);$watermarkHeight = imagesy($this->watermarkImage);// 计算水印位置(右下角)$destX = $sourceWidth - $watermarkWidth - 10; // 右边距10像素$destY = $sourceHeight - $watermarkHeight - 10; // 底边距10像素// 将水印图片合并到源图片上imagecopy($this->sourceImage, $this->watermarkImage, $destX, $destY, 0, 0, $watermarkWidth, $watermarkHeight);}private function saveImage(){// 保存加水印后的图片if (!imagejpeg($this->sourceImage, $this->outputImagePath)) {imagedestroy($this->sourceImage);imagedestroy($this->watermarkImage);die('无法保存加水印后的图片: ' . $this->outputImagePath);}}private function cleanup(){// 释放内存imagedestroy($this->sourceImage);imagedestroy($this->watermarkImage);}public function applyWatermark(){$this->loadImages();$this->addWatermark();$this->saveImage();$this->cleanup();echo "图片水印已添加并保存到: " . $this->outputImagePath;echo "<br>";echo "<img src='" . $this->outputImagePath . "' alt='图片水印' width='400px;' height='300px;'>";}
}?>

2、调用图片水印类

<?php
require "WatermarkImage.class.php";
// 使用示例
$sourceImagePath = '1111.jpg';
$watermarkImagePath = 'shuiyin.png';
$outputImagePath = 'image_watermark.jpg';$watermarker = new WatermarkImage($sourceImagePath, $watermarkImagePath, $outputImagePath);
$watermarker->applyWatermark();?>

五、压缩图片

1、缩略图封装类 ImageResizer

<?phpclass ImageResizer
{private $sourceImage;private $sourceWidth;private $sourceHeight;public function __construct($sourceImagePath){$this->loadImage($sourceImagePath);}private function loadImage($path){if (!file_exists($path)) {throw new Exception("File does not exist: $path");}$this->sourceImage = imagecreatefromjpeg($path);if (!$this->sourceImage) {throw new Exception("Failed to load image: $path");}$this->sourceWidth = imagesx($this->sourceImage);$this->sourceHeight = imagesy($this->sourceImage);}public function resize($width, $height, $outputPath){$thumb = imagecreatetruecolor($width, $height);imagecopyresampled($thumb,$this->sourceImage,0,0,0,0,$width,$height,$this->sourceWidth,$this->sourceHeight);if (!imagejpeg($thumb, $outputPath)) {throw new Exception("Failed to save thumbnail to: $outputPath");}imagedestroy($thumb);}public function __destruct(){imagedestroy($this->sourceImage);}
}?>

2、调用缩略图类

<?php
require "Resizer.class.php";// 使用示例
try {$sourceImagePath = 'E:\test\Resizer\222.jpg';// $outputImagePath = 'E:\test\Resizer\222_resier.jpg';$outputImagePath = '222_resier.jpg';$width = 200;$height = 200;$resizer = new ImageResizer($sourceImagePath);$resizer->resize($width, $height, $outputImagePath);echo "缩略图创建在网站目录下,图片名称: $outputImagePath";echo "<br>";echo "<img src='$outputImagePath' alt='缩略图'>";
} catch (Exception $e) {echo "错误: " . $e->getMessage();
}
?>

说明:更多复杂的水印功能可以基于此脚本扩展

相关文章:

php 实现:给图片加文字水印,图片水印,压缩图片

演示环境&#xff1a; 1、windows10 2、phpstudy 3、php7.4 一、案例演示&#xff1a; 二、素材准备 1、准备一张原始图片 2、准备一张水印图片&#xff08;透明底图的最好&#xff09; 3、字体库&#xff08;windows系统自带的字体库&#xff0c;路径在&#xff1a;C:\Window…...

免费实现网站HTTPS访问

HTTPS&#xff08;Hypertext Transfer Protocol Secure&#xff09;是一种基于SSL协议的HTTP安全协议&#xff0c;旨在为客户端&#xff08;浏览器&#xff09;与服务器之间的通信提供加密通道&#xff0c;确保数据在传输过程中的保密性、完整性和身份验证。与传统的HTTP相比&a…...

vue3使用vue3-print-nb打印

打印效果 1.下载插件 Vue2.0版本安装方法 npm install vue-print-nb --saveVue3.0版本安装方法&#xff1a; npm install vue3-print-nb --save2.main.js引入 vue2引入 import Print from vue-print-nb Vue.use(Print)vue3引入 import print from vue3-print-nb // 打印…...

R语言ggplot2包绘制网络地图

重要提示&#xff1a;数据和代码获取&#xff1a;请查看主页个人信息&#xff01;&#xff01;&#xff01; 载入R包 rm(listls()) pacman::p_load(tidyverse,assertthat,igraph,purrr,ggraph,ggmap) 网络节点和边数据 nodes <- read.csv(nodes.csv, row.names 1) edges…...

php获取今天凌晨零点的时间

不废话直接上代码 //使用strtotime $midnightToday strtotime("today midnight"); //输出&#xff1a;1716998400 //如果是明天 $midnightToday 86400 //后天 $midnightToday 86400*2//ORM中比对使用 $row ModelVisit::where(uid,$this->uid)->where(visi…...

CATIA进阶操作——创成式曲面设计入门(1)线架设计,三维点、直线、平面、曲线

目录 引出三维空间点生成三维直线三维平面三维曲线总结异形弹簧新建几何体草图编辑&#xff0c;画一条样条线进行扫掠&#xff0c;圆心和半径画出曲面上的螺旋线再次选择扫掠&#xff0c;圆心和半径 其他自定义信号和槽1.自定义信号2.自定义槽3.建立连接4.进行触发 自定义信号重…...

thinkphp6中怎么查看ThinkPHP版本号

<?php namespace app\controller; use app\BaseController; use think\app; //这句 class Index extends BaseController { public function test() { echo App::VERSION; //还有这句 } }...

第十二章 创建Web客户端

文章目录 第十二章 创建Web客户端SOAP向导的概述使用SOAP向导 第十二章 创建Web客户端 web客户端是访问web服务的软件。web客户端提供了一组代理方法&#xff0c;每个方法对应于web服务的一个方法。代理方法使用与它所对应的web服务方法相同的签名&#xff0c;并在被请求时调用…...

调试记录-RK平台用指令开启ADB功能

需求 嵌入式Linux系统调试过程中&#xff0c;为了方便&#xff0c;我们会借鉴Android调试的方法&#xff0c;在Linux系统添加adb功能&#xff0c;主要功能是通过USB线连接开发板和PC&#xff0c;实现两者之间传输文件&#xff0c;在PC上执行指令操作开发板。 实现 前提&…...

奇安信_NAC终端安全准入系统(相关问题整理)

奇安信终端安全准入系统 ,下称NAC 一、入网控制方式 1.IP流量控制 2.802.1X 准入 需要NAC、交换机、终端 以802.1X 3.DHCP 准入 将NAC作为DHCP服务器&#xff0c;为客户端分配地址&#xff0c;并对分配地址的客户端进行入网管控。 &#xff08;*&#xff09;可选 强制入网…...

在iPhone上恢复已删除的Safari历史记录的最佳方法

您是否正在寻找恢复 iPhone 上已删除的 Safari 历史记录的最佳方法&#xff1f;好吧&#xff0c;这篇文章提供了 4 种在有/无备份的情况下恢复 iPhone 上已删除的 Safari 历史记录的最佳方法。现在按照分步指南进行操作。 iPhone 上的 Safari 历史记录会被永久删除吗&#xff1…...

【设计模式深度剖析】【7】【结构型】【享元模式】| 以高脚杯重复使用、GUI中的按钮为例说明,并对比Java类库设计加深理解

&#x1f448;️上一篇:外观模式 | 下一篇:结构型设计模式对比&#x1f449;️ 设计模式-专栏&#x1f448;️ 目录 享元模式定义英文原话直译如何理解&#xff1f;字面理解例子&#xff1a;高脚杯的重复使用例子&#xff1a;GUI中的按钮传统方式使用享元模式 4个角色1. …...

OceanBase 内存研究(OceanBase 3.2.4.5)

内存结构 从官网的结构图可以看出&#xff0c;一台observer可使用的总内存(memory_limit)包括 系统内存(system_memory) 和 租户内存(sys租户与普通租户) 系统内存 系统内存system_memory 属于 observer 的内部内存&#xff0c;允许其它租户共享使用该内存资源 (root10.0.0.…...

麒麟系统 安装xrdp 远程桌面方法记录

一、安装环境 麒麟V10 2107 ft2000 麒麟V10 2107 x86_64 二、安装准备 使用《Kylin-Desktop-V10-Release-2107-arm64.iso》镜像 做好U盘启动系统后&#xff0c;需要安装一个远程桌面工具&#xff0c;可以多用户在windows上使用远程桌面访问麒麟系统。 目前在linux系统上较…...

解析Java中1000个常用类:SafeVarargs类,你学会了吗?

在 Java 编程中,泛型和可变参数(varargs)的结合使用可能会导致一些类型安全的问题。为了解决这些问题,Java 提供了 @SafeVarargs 注解。本文将详细介绍 @SafeVarargs 注解的定义、使用方法、应用场景以及其背后的原理,帮助读者深入理解并掌握这一重要特性。 什么是 @Safe…...

【数据挖掘】3σ原则识别数据中的异常值(附代码)

写在前面&#xff1a; 首先感谢兄弟们的订阅&#xff0c;让我有创作的动力&#xff0c;在创作过程我会尽最大能力&#xff0c;保证作品的质量&#xff0c;如果有问题&#xff0c;可以私信我&#xff0c;让我们携手共进&#xff0c;共创辉煌。 路虽远&#xff0c;行则将至&#…...

人眼是如何看到物体的

我在试图理解人眼如何观察到物体的&#xff0c;发现没有解释。本来我想这应该跟照相机照相的结果一样&#xff0c;但是发现&#xff0c;照相机也不对劲&#xff0c;没有理由能成像啊。 因为物体在散射光的时候&#xff0c;假设散射的光在局部是平行光&#xff0c;那么物体散射…...

vue打包时报错文件包过大

1.问题&#xff1a;npm run build 之后出现 2. 翻译之后意思就是某块过大 3. 解决办法&#xff1a;在vite.config.ts文件上添加 build: { chunkSizeWarningLimit: 1600, }, 4.最终打包...

预编码算法(个人总结)

引言 预编码算法是现代无线通信系统中的关键技术&#xff0c;特别是在多输入多输出&#xff08;MIMO&#xff09;系统中。它们通过在发送端对信号进行处理&#xff0c;减少干扰并提高信道容量。这种技术广泛应用于5G、Wi-Fi和卫星通信系统中。本教程将详细介绍预编码算法的背景…...

【重学C语言】十七、预处理指令

【重学C语言】十七、预处理指令 预处理指令预定义宏`#define` 宏定义示例注意事项特殊符号条件编译头文件包含`#pragma`预处理指令 C语言中的预处理指令(Preprocessor Directives)是一种特殊的指令,它们在编译过程的早期阶段(即实际编译之前)被预处理器(Preprocessor)处…...

C++初阶-list的底层

目录 1.std::list实现的所有代码 2.list的简单介绍 2.1实现list的类 2.2_list_iterator的实现 2.2.1_list_iterator实现的原因和好处 2.2.2_list_iterator实现 2.3_list_node的实现 2.3.1. 避免递归的模板依赖 2.3.2. 内存布局一致性 2.3.3. 类型安全的替代方案 2.3.…...

label-studio的使用教程(导入本地路径)

文章目录 1. 准备环境2. 脚本启动2.1 Windows2.2 Linux 3. 安装label-studio机器学习后端3.1 pip安装(推荐)3.2 GitHub仓库安装 4. 后端配置4.1 yolo环境4.2 引入后端模型4.3 修改脚本4.4 启动后端 5. 标注工程5.1 创建工程5.2 配置图片路径5.3 配置工程类型标签5.4 配置模型5.…...

(十)学生端搭建

本次旨在将之前的已完成的部分功能进行拼装到学生端&#xff0c;同时完善学生端的构建。本次工作主要包括&#xff1a; 1.学生端整体界面布局 2.模拟考场与部分个人画像流程的串联 3.整体学生端逻辑 一、学生端 在主界面可以选择自己的用户角色 选择学生则进入学生登录界面…...

React Native在HarmonyOS 5.0阅读类应用开发中的实践

一、技术选型背景 随着HarmonyOS 5.0对Web兼容层的增强&#xff0c;React Native作为跨平台框架可通过重新编译ArkTS组件实现85%以上的代码复用率。阅读类应用具有UI复杂度低、数据流清晰的特点。 二、核心实现方案 1. 环境配置 &#xff08;1&#xff09;使用React Native…...

大模型多显卡多服务器并行计算方法与实践指南

一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...

【开发技术】.Net使用FFmpeg视频特定帧上绘制内容

目录 一、目的 二、解决方案 2.1 什么是FFmpeg 2.2 FFmpeg主要功能 2.3 使用Xabe.FFmpeg调用FFmpeg功能 2.4 使用 FFmpeg 的 drawbox 滤镜来绘制 ROI 三、总结 一、目的 当前市场上有很多目标检测智能识别的相关算法&#xff0c;当前调用一个医疗行业的AI识别算法后返回…...

网站指纹识别

网站指纹识别 网站的最基本组成&#xff1a;服务器&#xff08;操作系统&#xff09;、中间件&#xff08;web容器&#xff09;、脚本语言、数据厍 为什么要了解这些&#xff1f;举个例子&#xff1a;发现了一个文件读取漏洞&#xff0c;我们需要读/etc/passwd&#xff0c;如…...

Razor编程中@Html的方法使用大全

文章目录 1. 基础HTML辅助方法1.1 Html.ActionLink()1.2 Html.RouteLink()1.3 Html.Display() / Html.DisplayFor()1.4 Html.Editor() / Html.EditorFor()1.5 Html.Label() / Html.LabelFor()1.6 Html.TextBox() / Html.TextBoxFor() 2. 表单相关辅助方法2.1 Html.BeginForm() …...

【LeetCode】3309. 连接二进制表示可形成的最大数值(递归|回溯|位运算)

LeetCode 3309. 连接二进制表示可形成的最大数值&#xff08;中等&#xff09; 题目描述解题思路Java代码 题目描述 题目链接&#xff1a;LeetCode 3309. 连接二进制表示可形成的最大数值&#xff08;中等&#xff09; 给你一个长度为 3 的整数数组 nums。 现以某种顺序 连接…...

Java并发编程实战 Day 11:并发设计模式

【Java并发编程实战 Day 11】并发设计模式 开篇 这是"Java并发编程实战"系列的第11天&#xff0c;今天我们聚焦于并发设计模式。并发设计模式是解决多线程环境下常见问题的经典解决方案&#xff0c;它们不仅提供了优雅的设计思路&#xff0c;还能显著提升系统的性能…...