QQ登录

只需要一步,快速开始

APP扫码登录

只需要一步,快速开始

手机号码,快捷登录

手机号码,快捷登录

查看: 1516|回复: 0

[PHP] 图片无损压缩

[复制链接]

等级头衔

积分成就    金币 : 2857
   泡泡 : 1516
   精华 : 6
   在线时间 : 1317 小时
   最后登录 : 2025-4-23

丰功伟绩

优秀达人突出贡献荣誉管理论坛元老活跃会员

联系方式
发表于 2023-4-23 12:49:49 | 显示全部楼层 |阅读模式
想想一下,假设服务器的宽带是100M,一张图片5M,20张图片就可以将服务器的宽带挤爆,就更别提api请求返回数据了。而且图片比较占用磁盘空间,对磁盘也会造成不小的压力。* N8 m) y1 ^' \' {6 }
那么针对图片有没有什么优化的方法呢?
* @; a* Z: N$ m. k主要的优化有两个方向,第一个就是nginx启用gzip压缩,nginx返回图片数据给前端时先将图片压缩再返回。这种方式确实可以最大化减少宽带的压力,但是也增加了cpu的压力。  p% g) J1 Z/ z3 u: ~' A2 B
还有一种办法,存储图片时先将图片进行无损压缩。比如一张5M的图片经过无损压缩后大小只有不到500kb。既可以解决宽带压力还可以减少磁盘空间存储的压力。9 i9 i/ \1 P. e
当然大家一般都是同时使用上面两种方案,图片无损压缩后存储到服务器,nginx再使用gzip压缩,客户端再进行解压缩。实际到达客户端的就只有几十kb了' N9 @5 d* J/ l5 i& k) o
无损压缩% I& O" `2 _3 s  H
在不改变图片尺寸大小,清晰度的前提下将图片压缩到更小,这就是无损压缩。$ X* P# U1 t. ]! u$ o. h
PHP代码实现无损压缩
/ }& r. r; j6 r+ i+ c5 [在PHP中能实现图片无损压缩的有GD和Imagick这两个扩展。PHP7.4之后官方已经建议大家使用Imagick,因为这个扩展功能更加强大,我们这里也使用Imagick进行演示。
: z, m  G2 b+ k. U7 K; \& Z5 l$ ]
//创建imagick对象并将读取上传的临时文件
$imgick = new \Imagick($_FILES['file']['tmp_name']);

//设置图片压缩的质量(必须是整数)
$imgick->setImageCompressionQuality(70);

//获取原图的后缀(jpg,jpeg,png,webp等)
$format = pathinfo($_FILES['file']['name'])['extension'];

//设置图片压缩后的格式,此处一定要设置(有些图片可能是webp强转png,有可能导致错误)
$imgick->setFormat($format);

//获取压缩后的二进制图片
$imgBlob = $imgick->getImageBlob();

//二进制图片转base64格式
$base64 = 'data:image/'.$format.';base64,'.base64_encode($imgBlob);

//获取压缩后的大小,单位kb
$size = ceil($imgick->getImageLength()/1024);
如果你遇到这个错误:413 Request Entity Too Large: P7 @) m: ^9 E7 d
那是你的nginx或者apache限制了body最大值,可以在nginx的server中添加如下信息,将限制改为20M:client_max_body_size 20m;- ]1 k/ Y' X9 d8 {4 }2 Z  c* A0 e
当然在PHP配置文件中也有限制,需要将PHP对上传文件的大小限制改大一些,如下所示:
5 f5 ~1 R' V# g- k* o0 v& {' B
# 允许最大上传文件的大小改为20兆
. ^% \% ~7 j  M! o6 b) Gupload_max_filesize = 20M

# O: R# G' H0 n
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|手机版|小黑屋|paopaomj.COM ( 渝ICP备18007172号|渝公网安备50010502503914号 )

GMT+8, 2025-4-25 09:00

Powered by paopaomj X3.5 © 2016-2025 sitemap

快速回复 返回顶部 返回列表