图片资源压缩方法
2019-09-20 本文已影响0人
一路香蕉皮
在项目开发过程中,自然而然会使用到资源压缩,因为服务器带宽很珍贵的,不能任由用户上传任意大小的文件,所以需要将静态资源压缩后再上传上去,这里面提升空间最大的就是图片了。
项目工程内图片资源
这种主要是开发阶段的图片资源压缩,这种可以直接使用别人的软件,压缩图片后,引入到项目内就可以。
这种主要推荐两款。
Squoosh
这款软件是Google公司开源的项目,使用方法很简单,访问这款软件的官网---Squoosh,把需要压缩的图片拖入或者手动上传,就可以在右下角选择要压缩的格式,点击下载就完成了。同时,它还支持重置图片大小,选择压缩级别,减少色彩等等实用功能。
不过,这款app最大的槽点是不支持批量压缩和不支持第三方API,一次只能处理一张图片,每次只能在浏览器里面访问网站操作,虽然压缩可以离线操作,但是不能在项目里面调用它,很是让人失望。
GitHub地址---https://github.com/GoogleChromeLabs/squoosh/
TinyPNG
这款软件同样支持在线压缩,不过只支持JPG和PNG图片,使用方法和上面一样简单,访问官网---TinyPNG,上传图片就可以。
主要说下区别,这款软件支持批量上传图片和第三方API调用,不过免费版一个月只能压缩500次,付费版价格又太贵了,所以不适合用在用户上传图片这类频繁压缩图片的场景。
用户上传的图片资源
开发过程中,经常会使用到对于用户的图片进行压缩,您可以用下面的这段脚本,用的是上面TinyPNG的API方式,要先去TinyPNG官网申请一个API。下面是脚本
# -*- coding: utf-8 -*-
"""脚本功能说明:使用 tinypng api,一键批量压缩指定文件(夹)所有文件"""
import os
import sys
import tinify
tinify.key = "your own key" # AppKey
def get_file_dir(file):
"""获取文件目录通用函数"""
fullpath = os.path.abspath(os.path.realpath(file))
return os.path.dirname(fullpath)
def check_suffix(file_path):
"""检查指定文件的后缀是否符合要求"""
file_path_lower = file_path.lower()
return (file_path_lower.endswith('.png')
or file_path_lower.endswith('.jpg')
or file_path_lower.endswith('.jpeg'))
def compress_by_tinypng(input_file):
"""使用 tinypng 进行压缩,中文前面的 u 是为了兼容 py2.7"""
if not check_suffix(input_file):
print(u'只支持png\\jpg\\jepg格式文件:' + input_file)
return
file_name = os.path.basename(input_file)
output_path = os.path.join(get_file_dir(input_file), 'tinypng')
output_file = os.path.join(output_path, file_name)
if not os.path.isdir(output_path):
os.makedirs(output_path)
try:
source = tinify.from_file(input_file)
source.to_file(output_file)
print(u'文件压缩成功:' + input_file)
old_size = os.path.getsize(input_file)
print(u'压缩前文件大小:%d 字节' % old_size)
new_size = os.path.getsize(output_file)
print(u'文件保存地址:%s' % output_file)
print(u'压缩后文件大小:%d 字节' % new_size)
print(u'压缩比: %d%%' % ((old_size - new_size) * 100 / old_size))
except tinify.errors.AccountError:
print(u'Key 使用量已超,请更新 Key,并使用命令[Usage] %s [filepath] [key]运行'
% os.path.basename(sys.argv[0]))
def check_path(input_path):
"""如果输入的是文件则直接压缩,如果是文件夹则先遍历"""
if os.path.isfile(input_path):
compress_by_tinypng(input_path)
elif os.path.isdir(input_path):
dirlist = os.walk(input_path)
for root, dirs, files in dirlist:
for filename in files:
compress_by_tinypng(os.path.join(root, filename))
else:
print(u'目标文件(夹)不存在,请确认后重试。')
if __name__ == '__main__':
len_param = len(sys.argv)
if len_param != 2 and len_param != 3:
print('[Usage] %s [filepath]' % os.path.basename(sys.argv[0]))
elif len_param == 3:
tinify.key = sys.argv[2]
check_path(sys.argv[1])
else:
check_path(sys.argv[1])