Golang压缩Jpeg图片和PNG图片的操作
作者:员力 发布时间:2024-05-09 14:55:43
博主一直在维护一个导出PDF的服务,但是这个服务导出的PDF文件是真的巨大,动辄就上百MB。这里面主要是图片占据了大多数体积,所以考虑在导出前压缩一下图片。
Jpeg的图片压缩是很好做的,因为jpeg这个协议本身就支持调整图片质量的。在golang中,我们只需要使用标准库的image/jpeg,将图片从二进制数据解码后,降低质量再编码为二进制数据即可实现压缩。而且质量和压缩比例相对而言还不错。
func compressImageResource(data []byte) []byte {
img, _, err := image.Decode(bytes.NewReader(data))
if err != nil {
return data
}
buf := bytes.Buffer{}
err = jpeg.Encode(&buf, img, &jpeg.Options{Quality: 40})
if err != nil {
return data
}
if buf.Len() > len(data) {
return data
}
return buf.Bytes()
}
比较麻烦的是压缩PNG图片,在网上找了很多相关的库,感觉都没什么即可以保持质量,又可以尽可能压缩的办法。
//下面这两个库都比较偏重于转换图片大小,在保持宽高不变的情况下,压缩比例很一般
https://github.com/discord/lilliput //这个库是一家海外公司基于C语言的一个开源图片处理库,但是封装的很好,不需要安装额外依赖
https://github.com/disintegration/imaging
//下面这个库可以对PNG图片进行较大的压缩,可惜压缩比例过大时会严重失真
https://github.com/foobaz/lossypng/
后来,借鉴一篇博客的做法,还是先把PNG图片转换为Jpeg图片,然后再将jpeg图片的质量降低。相对上边这些库,压缩比例和质量都比较令人满意
func compressImageResource(data []byte) []byte {
imgSrc, _, err := image.Decode(bytes.NewReader(data))
if err != nil {
return data
}
newImg := image.NewRGBA(imgSrc.Bounds())
draw.Draw(newImg, newImg.Bounds(), &image.Uniform{C: color.White}, image.Point{}, draw.Src)
draw.Draw(newImg, newImg.Bounds(), imgSrc, imgSrc.Bounds().Min, draw.Over)
buf := bytes.Buffer{}
err = jpeg.Encode(&buf, newImg, &jpeg.Options{Quality: 40})
if err != nil {
return data
}
if buf.Len() > len(data) {
return data
}
return buf.Bytes()
}
最后给大家分享一个超级好用PDF处理的golang 库: https://github.com/unidoc/unipdf。一开始使用这个库将生成后的PDF压缩的,可以将一个200M的PDF(里面都是图片)直接压缩到7M左右。可惜的是这个库商用需要购买商业版权,所以最后只能采取了导出前压缩图片的做法。
这个库没有授权的情况下会在处理后的PDF中加上水印,这个想去掉也简单,fork下来改一下代码就好了。虽然我这里因为是商业的场景不能这么用,但是我还是尝试了下,仓库在这:https://github.com/lianggx6/unipdf。然后再在go.mod文件中将依赖替换即可。大家如果有个人开发实践需要的可以直接这样拿来用,商用务必购买版权。
replace (
github.com/unidoc/unipdf/v3 => github.com/lianggx6/unipdf v0.0.0-20200409043947-1c871b2c4951
)
补充:golang中image/jpeg包和image/png包用法
jpeg包实现了jpeg图片的编码和解码
func Decode(r io.Reader) (image.Image, error) //Decode读取一个jpeg文件,并将他作为image.Image返回
func DecodeConfig(r io.Reader) (image.Config, error) //无需解码整个图像,DecodeConfig变能够返回整个图像的尺寸和颜色(Config具体定义查看gif包中的定义)
func Encode(w io.Writer, m image.Image, o *Options) error //按照4:2:0的基准格式将image写入w中,如果options为空的话,则传递默认参数
type Options struct {
Quality int
}
Options是编码参数,它的取值范围是1-100,值越高质量越好
type FormatError //用来报告一个输入不是有效的jpeg格式
type FormatError string
func (e FormatError) Error() string
type Reader //不推荐使用Reader
type Reader interface {
io.ByteReader
io.Reader
}
type UnsupportedError
func (e UnsupportedError) Error() string //报告输入使用一个有效但是未实现的jpeg功能
利用程序画一条直线,代码如下:
package main
import (
"fmt"
"image"
"image/color"
"image/jpeg"
"math"
"os"
)
const (
dx = 500
dy = 300
)
type Putpixel func(x, y int)
func drawline(x0, y0, x1, y1 int, brush Putpixel) {
dx := math.Abs(float64(x1 - x0))
dy := math.Abs(float64(y1 - y0))
sx, sy := 1, 1
if x0 >= x1 {
sx = -1
}
if y0 >= y1 {
sy = -1
}
err := dx - dy
for {
brush(x0, y0)
if x0 == x1 && y0 == y1 {
return
}
e2 := err * 2
if e2 > -dy {
err -= dy
x0 += sx
}
if e2 < dx {
err += dx
y0 += sy
}
}
}
func main() {
file, err := os.Create("test.jpg")
if err != nil {
fmt.Println(err)
}
defer file.Close()
nrgba := image.NewNRGBA(image.Rect(0, 0, dx, dy))
drawline(1, 1, dx-2, dy-2, func(x, y int) {
nrgba.Set(x, y, color.RGBA{uint8(x), uint8(y), 0, 255})
})
for y := 0; y < dy; y++ {
nrgba.Set(1, y, color.White)
nrgba.Set(dx-1, y, color.White)
}
err = jpeg.Encode(file, nrgba, &jpeg.Options{100}) //图像质量值为100,是最好的图像显示
if err != nil {
fmt.Println(err)
}
}
根据已经得到的图像test.jpg,我们创建一个新的图像test1.jpg
package main
import (
"fmt"
"image/jpeg"
"os"
)
func main() {
file, err := os.Open("test.jpg")
if err != nil {
fmt.Println(err)
}
defer file.Close()
file1, err := os.Create("test1.jpg")
if err != nil {
fmt.Println(err)
}
defer file1.Close()
img, err := jpeg.Decode(file) //解码
if err != nil {
fmt.Println(err)
}
jpeg.Encode(file1, img, &jpeg.Options{5}) //编码,但是将图像质量从100改成5
}
对比图像质量为100和5的图像:
image/png包用法:
image/png实现了png图像的编码和解码
png和jpeg实现方法基本相同,都是对图像进行了编码和解码操作。
func Decode(r io.Reader) (image.Image, error) //Decode从r中读取一个图片,并返回一个image.image,返回image类型取决于png图片的内容
func DecodeConfig(r io.Reader) (image.Config, error) //无需解码整个图像变能够获取整个图片的尺寸和颜色
func Encode(w io.Writer, m image.Image) error //Encode将图片m以PNG的格式写到w中。任何图片都可以被编码,但是哪些不是image.NRGBA的图片编码可能是有损的。
type FormatError
func (e FormatError) Error() string //FormatError会提示一个输入不是有效PNG的错误。
type UnsupportedError
func (e UnsupportedError) Error() string //UnsupportedError会提示输入使用一个合法的,但是未实现的PNG特性。
利用png包实现一个png的图像,代码如下:
package main
import (
"fmt"
"image"
"image/color"
"image/png"
"os"
)
const (
dx = 256
dy = 256
)
func Pic(dx, dy int) [][]uint8 {
pic := make([][]uint8, dx)
for i := range pic {
pic[i] = make([]uint8, dy)
for j := range pic[i] {
pic[i][j] = uint8(i * j % 255)
}
}
return pic
}
func main() {
file, err := os.Create("test.png")
if err != nil {
fmt.Println(err)
}
defer file.Close()
rgba := image.NewRGBA(image.Rect(0, 0, dx, dy))
for x := 0; x < dx; x++ {
for y := 0; y < dy; y++ {
rgba.Set(x, y, color.RGBA{uint8(x * y % 255), uint8(x * y % 255), 0, 255})
}
}
err = png.Encode(file, rgba)
if err != nil {
fmt.Println(err)
}
}
图像如下:
由此可见,png和jpeg使用方法类似,只是两种不同的编码和解码方式。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持asp之家。如有错误或未考虑完全的地方,望不吝赐教。
来源:https://www.cnblogs.com/lianggx6/p/12684885.html


猜你喜欢
- 正在看的ORACLE教程是:Oracle常见错误代码的分析与解决。在使用ORACLE的过程过,我们会经常遇到一些ORACLE产生的错误,对于
- 虽然Python被说成是一种解释型语言,但是实际上,Python源程序要先经过编译,然后才能运行。与Java语言类似,Python源程序编译
- 安装模块如果使用的是Linux系统,并且安装了pip,可以直接使用pip安装xlrd, xlwt:pip install xlwtpip i
- 最近在无忧脚本混了一阵子,回复了一些贴子,自己却没有做出什么东东让大家看看,心里有些不安,于是写了下边的一点东西,本来应该发在类封装区的,考
- 不喜欢Python的人经常会吐嘈Python运行太慢。但是,事实并非如此。尝试以下六个窍门,来为你的Python应用提速。窍门一:关键代码使
- 方法一:函数添加1 import sys2 查看sys.path3 添加sys.path.append("c:\\")方
- 在利用python进行flask等开发过程中经常需要配置虚拟环境以方便针对不同的项目需求配置不同的生产环境。在python3.3之前,需要利
- 相信很多初学python的小伙伴都会遇到这样的坑:环境变量配置不好,无法正常启动python。那么环境变量究竟是个什么东西呢?为什么要设置它
- 最近认识了一个做Python语音识别的朋友,聊天时候说到,未来五到十年,Python人工智能会在国内掀起一股狂潮,对各种应用的冲击,不下于淘
- 1. 调试pythonipdb是用来python中用以交互式debug的模块,可以直接利用pip安装;其功能类似于pycharm中pytho
- 网上有很多免费的ip地址,都是可以使用的,但是如果手动来获取太麻烦,这里通过Python自动抓取,可以批量获取。代码如下:# -*- cod
- 下面是我已经证实可用的自动备份的方法. 1、打开企业管理器->管理->sql server代理 2、新建一个作业,作业名称随便取
- 本文介绍了python selenium UI自动化解决验证码的4种方法,分享给大家,具体如下:测试环境windows7+firefox50
- 目录前期准备界面编写截图功能实现OCR实现内容显示总结前期准备在这个阶段主要准备整个小程序的结构,既然要实现ocr,那么输入就是一张图片,而
- --1. 创建表,添加测试数据 CREATE TABLE tb(id int, [value] varchar(10)) INSERT tb
- 首先,自学Python是能够找到相关工作的。Python语言在近几年的上升趋势非常明显,语言生态也越来越健全,在Web开发、大数据开发、人工
- 一、说明 关于matplotlib的scatter函数有许多活动参数,如果不专门注解,是
- 在批评Python的讨论中,常常说起Python多线程是多么的难用。还有人对 global interpreter lock(也被亲切的称为
- 目录什么是虚拟 dom?为什么需要虚拟dom?虚拟dom是如何转换为真实dom的?模板和虚拟dom的关系注入挂载完整流程总结什么是虚拟 do
- 本文为大家分享了Python多线程聊天室,是一个Socket,两个线程,一个是服务器,一个是客户端。 最近公司培训,要写个大富翁的小程序,准