Java 图片处理——如何生成高清晰度而占有磁盘小的缩略图
现在的web项目,图片越来越多,图片大小也越来越大,随便就能达到1M,2M,甚至更大。用户上传的图片,一般是无法直接使用的。一般要生成两三种对应的缩略图,分别适配不同的终端,不同的场景。比如PC,手机,平板等等不同的终端;在比如图片列表和图片详情,肯定一个要使用缩略图,一个要使用高清图。
一般图片优化的第一步,就是在适当的地方使用缩略图,尽量不要在web端使用CSS缩放高清原始图片。下面分析了Java中如何生成不同的缩略图的技术。
常见的图片格式有: ".*\\.(?i)(jpg|jpeg|gif|bmp|png)"
这其中有分为了两种,png 和 gif 是一种,其它格式是一种,因为 png 和 gif 存在透明度的问题,如果按照jpg一样处理,就会导致生成黑色背景的图片。
1. 指定高度的等比例 缩放图片:
- /**
- * 按指定高度 等比例缩放图片
- *
- * @param imageFile
- * @param newPath
- * @param newWidth 新图的宽度
- * @throws IOException
- */
- public static void zoomImageScale(File imageFile, String newPath, int newWidth) throws IOException {
- if(!imageFile.canRead())
- return;
- BufferedImage bufferedImage = ImageIO.read(imageFile);
- if (null == bufferedImage)
- return;
- int originalWidth = bufferedImage.getWidth();
- int originalHeight = bufferedImage.getHeight();
- double scale = (double)originalWidth / (double)newWidth; // 缩放的比例
- int newHeight = (int)(originalHeight / scale);
- zoomImageUtils(imageFile, newPath, bufferedImage, newWidth, newHeight);
- }
- private static void zoomImageUtils(File imageFile, String newPath, BufferedImage bufferedImage, int width, int height)
- throws IOException{
- String suffix = StringUtils.substringAfterLast(imageFile.getName(), ".");
- // 处理 png 背景变黑的问题
- if(suffix != null && (suffix.trim().toLowerCase().endsWith("png") || suffix.trim().toLowerCase().endsWith("gif"))){
- BufferedImage to= new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
- Graphics2D g2d = to.createGraphics();
- to = g2d.getDeviceConfiguration().createCompatibleImage(width, height, Transparency.TRANSLUCENT);
- g2d.dispose();
- g2d = to.createGraphics();
- Image from = bufferedImage.getScaledInstance(width, height, Image.SCALE_AREA_AVERAGING);
- g2d.drawImage(from, 0, 0, null);
- g2d.dispose();
- ImageIO.write(to, suffix, new File(newPath));
- }else{
- // 高质量压缩,其实对清晰度而言没有太多的帮助
- // BufferedImage tag = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
- // tag.getGraphics().drawImage(bufferedImage, 0, 0, width, height, null);
- //
- // FileOutputStream out = new FileOutputStream(newPath); // 将图片写入 newPath
- // JPEGImageEncoder encoder = JPEGCodec.createJPEGEncoder(out);
- // JPEGEncodeParam jep = JPEGCodec.getDefaultJPEGEncodeParam(tag);
- // jep.setQuality(1f, true); //压缩质量, 1 是最高值
- // encoder.encode(tag, jep);
- // out.close();
- BufferedImage newImage = new BufferedImage(width, height, bufferedImage.getType());
- Graphics g = newImage.getGraphics();
- g.drawImage(bufferedImage, 0, 0, width, height, null);
- g.dispose();
- ImageIO.write(newImage, suffix, new File(newPath));
- }
- }
上面中 zoomImageScale可以指定生成图片的高度,然后宽度按照原始图的 高宽比 计算出新图片的宽度;同理也可以 指定生成图片的宽度,来等比例生成新图片。zoomImageUtils 方法中涉及到了三种图片处理方法:
1)图片的按照指定高度,宽度 进行普通的重绘:
- BufferedImage newImage = new BufferedImage(width, height, bufferedImage.getType());
- Graphics g = newImage.getGraphics();
- g.drawImage(bufferedImage, 0, 0, width, height, null);
- g.dispose();
- ImageIO.write(newImage, suffix, new File(newPath));
2)利用JPEGImageEncoder生成所谓的“高质量”的图片:
- // 高质量压缩,其实对清晰度而言没有太多的帮助
- // BufferedImage tag = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
- // tag.getGraphics().drawImage(bufferedImage, 0, 0, width, height, null);
- //
- // FileOutputStream out = new FileOutputStream(newPath); // 将图片写入 newPath
- // JPEGImageEncoder encoder = JPEGCodec.createJPEGEncoder(out);
- // JPEGEncodeParam jep = JPEGCodec.getDefaultJPEGEncodeParam(tag);
- // jep.setQuality(1f, true); //压缩质量, 1 是最高值
- // encoder.encode(tag, jep);
- // out.close();
这种方法,其实仅仅是生成的图片所占硬盘更大而已,但是实际上,对图片的清晰度而已,没有实际的作用。
3)png 和 gif 图片不能采用上面说到的 图片处理方法,因为会导致生成的图片背景变成黑色,要另外处理(指定透明处理):
- BufferedImage to = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
- Graphics2D g2d = to.createGraphics();
- to = g2d.getDeviceConfiguration().createCompatibleImage(width,height, Transparency.TRANSLUCENT);
- g2d.dispose();
- g2d = to.createGraphics();
- Image from = bufferedImage.getScaledInstance(width, height, Image.SCALE_AREA_AVERAGING);
- g2d.drawImage(from, 0, 0, null);
- g2d.dispose();
- ImageIO.write(to, suffix, new File(newPath));
4)线性处理,本方法也不能处理 png 图片:
- /**
- * 等比例改变图片尺寸
- * @param nw 新图片的宽度
- * @param oldImage 原图片
- * @throws IOException
- */
- public static void constrainProportios(int nw, String oldImage) throws IOException {
- AffineTransform transform = new AffineTransform();
- BufferedImage bis = ImageIO.read(new File(oldImage));
- int w = bis.getWidth();
- int h = bis.getHeight();
- int nh = (nw * h) / w;
- double sx = (double) nw / w;
- double sy = (double) nh / h;
- transform.setToScale(sx, sy);
- AffineTransformOp ato = new AffineTransformOp(transform, null);
- BufferedImage bid = new BufferedImage(nw, nh, BufferedImage.TYPE_3BYTE_BGR);
- ato.filter(bis, bid);
- String newPath = StringUtils.substringBeforeLast(oldImage,".")+"_3."+StringUtils.substringAfterLast(oldImage,".");
- ImageIO.write(bid, "jpeg", new File(newPath));
- // ImageIO.write(bid, "jpeg", response.getOutputStream());
- }
上面有4中图片的生成方法,除了png需要另外处理之外,其它几种图片的处理方法,用实际生成的图片的清晰度比较而言,实际上是差别不大,基本没有明显的差别。实际的测试发现,如果要生成的图片和原始图片,在清晰度上要达到用肉眼不能明显区分它们的效果的话,关键的不是使用哪种图片生成方法,关键的是不要让生成的图片的 width 和 height 太小!这个才是关键,实际测试发现,生成的图片的 width 和 height 最好不要小于500,一定不要小于 400。
2. 按照指定的高度和宽度生成图片:
- /**
- * 按尺寸缩放图片
- *
- * @param imageFile
- * @param newPath
- * @param times
- * @throws IOException
- */
- public static void zoomImage(File imageFile, String newPath, int width, int height) throws IOException {
- if (imageFile != null && !imageFile.canRead())
- return;
- BufferedImage bufferedImage = ImageIO.read(imageFile);
- if (null == bufferedImage)
- return;
- zoomImageUtils(imageFile, newPath, bufferedImage, width, height);
- }
这里没有按照原始图片的 高宽比 来生成图片,而是按照指定的 高度和宽度来生成图片。一般而言,最好不要选择这种处理方法,因为图片会被压缩或者拉伸,图片会变得比较难看。
3. 实际效果比较:
1)原始图片:600 x 800, 220k
2) 按照指定高度等比例生成的图片,指定高度为448, 33.7k
3) 宽度和高度小于400的普通重绘的图片,224 x 300, 12.9k
4) 宽度和高度小于400的 JPEGImageEncoder生成所谓的“高质量”的图片,224 x 300, 63.4k
5)上面四种图片在网页中,按照同样的指定的图片CSS下面的表现:
- img {
- height: auto;
- max-height: 300px;
- max-width: 224px;
- width: auto;
- }
可以比较明显的看出,第一张和第二张图片的清晰度是极其接近的,你甚至无法说出那张图片更加清晰;而第三张和第四种图片,从额前的头发而言,就明显的比第一张和第二张要差一些。
实际的图片分别为:
第一张:按照指定高度等比例生成的图片,指定高度为 448, 33.7k
第二张:原始图片:600 x 800, 220k
第三张:宽度和高度小于400的普通重绘的图片,224 x 300, 12.9k
第四张:宽度和高度小于400的 JPEGImageEncoder生成所谓的“高质量”的图片,224 x 300, 63.4k
结论:
生成的图片的清晰度,主要取决于生成的图片的高度和宽度的大小,而不是取决于图片生成的算法;
如果要想达到肉眼无法分别的效果,生成图片的高宽最好不要小于500,一定不要小于400;
-----
第四张图片达到了63.4K,但是清晰度明显差于第一张大小只有33.7K的图片。图片的宽度和高度的大小才是清晰度的决定因数。
第三张图片只有 12.9K,而清晰度和第四张63.4K的图片清晰度几乎一样,因为它们的尺寸是一样的,都是224 x 300,尽管生成图片的算法不一样。
-----------------------------------------------------------------------
图片处理时,可能会遇到的两个问题:
1)上面的代码处理 png 图片时,如果png 图片的宽度或者高度,数值特别大时,可能会导致内存溢出,比如我在处理一张 320 x 16606 的png时,
jvm抛出了内存溢出,所以如何在上面的代码中加入一个判断,如果图片的宽度或者高度,数值特别大,跳过就行了。
2)JPG图片可能会遇到:javax.imageio.IIOException: Unsupported Image Type 错误,
- javax.imageio.IIOException: Unsupported Image Type
- at com.sun.imageio.plugins.jpeg.JPEGImageReader.readInternal(JPEGImageReader.java:1068)
- at com.sun.imageio.plugins.jpeg.JPEGImageReader.read(JPEGImageReader.java:1039)
- at javax.imageio.ImageIO.read(ImageIO.java:1448)
- at javax.imageio.ImageIO.read(ImageIO.java:1308)
该错误的原因是 图片的模式错了,PS保存图片为jpg格式时,默认的模式是CMYK模式(注意,这是给印刷机用的)。所以我们应该讲模式改成:RGB. 在图像-->模式中改为RGB模式才是显示器用的。
参考:http://blog.sina.com.cn/s/blog_600ff075010153wn.html
http://iaiai.iteye.com/blog/1461370
http://zhangmingji.iteye.com/blog/1969693
也可以使用 imageMagick软件的命令来处理:
C:\Users\Administrator>mogrify -colorspace RGB -quality 100 F:\upload\images\20150728\2015072822arrsOH.jpg
Java 图片处理——如何生成高清晰度而占有磁盘小的缩略图的更多相关文章
- java 图片与文字生成PDF
1.jar包:iText-2.1.5.jar 2.code: import java.awt.Color; import java.io.File; import java.io.FileNotFou ...
- 【java】Freemarker 动态生成word(带图片表格)
1.添加freemarker.jar 到java项目. 2.新建word文档. 3.将文档另存为xml 格式. 4.将xml格式化后打开编辑(最好用notepad,有格式),找到需要替换的内容,将内容 ...
- java图片操作--生成与原图对称的图片
java图片操作--生成与原图对称的图片 package com.pay.common.util; import java.awt.image.BufferedImage; import java.i ...
- java后台中处理图片辅助类汇总(上传图片到服务器,从服务器下载图片保存到本地,缩放图片,copy图片,往图片添加水印图片或者文字,生成二维码,删除图片等)
最近工作中处理小程序宝箱活动,需要java画海报,所以把这块都快百度遍了,记录一下处理的方法,百度博客上面也有不少坑! 获取本地图片路径: String bgPath = Thread.current ...
- java图片处理(加水印、生成缩略图)等之Thumbnailator库
Thumbnailator 是一个为Java界面更流畅的缩略图生成库.从API提供现有的图像文件和图像对象的缩略图中简化了缩略过程,两三行代码就能够从现有图片生成缩略图,且允许微调缩略图生成,同时保持 ...
- java 二维码生成(可带图片)springboot版
本文(2019年6月29日 飞快的蜗牛博客) 有时候,男人和女人是两个完全不同的世界,男人的玩笑和女人的玩笑也完全是两码事,爱的人完全不了解你,你也不要指望一个女人了解你,所以男的不是要求别人怎么样, ...
- Thumbnailator java图片压缩,加水印,批量生成缩略图
地址:http://code.google.com/p/thumbnailator/ 1.指定大小进行缩放 //size(宽度, 高度) /* * 若图片横比200小,高比300小,不变 * 若图片横 ...
- Java 图片提取RGB数组 RGBOfCharMaps (整理)
package demo; /** * Java 图片提取RGB数组 RGBOfCharMaps (整理) * 声明: * 和ImageCombining配合使用的工具,这里是提取图片的R.G.B生成 ...
- Java 图片转换为字符图 CharMaps (整理)
/* * Java 图片转换成字符图 CharMaps (整理) * * 2016-1-2 深圳 南山平山村 曾剑锋 * * @(#)CharMaps.java 2014/1/16 * 1.这个一 ...
随机推荐
- IOS开发UI基础UISwitch属性
UISwitch属性1. onTintColor 处于on时switch 的颜色 switchImage.onTintColor = [UIColor grayColor];2.tintC ...
- SeaJS 模块化加载框架使用
SeaJS 是一个遵循 CMD 规范的模块化加载框架 CommonJS,CMD,AMD等规范后文会提到,这里主要先了解如何在代码中使用. 如果你有使用过nodejs ,那么理解起来就容易多了. 我们通 ...
- ASP.NET的路由
之前在探讨ASP.NET MVC的路由时,无意发现原本ASP.NET也有路由机制的.在学习MVC的路由时觉得这部分的资料不太多,不怎么充实(也许是我不懂得去看微软的官方文档).后来也尝试一下ASP. ...
- Java编码规范
1. Java命名约定 除了以下几个特例之外,命名时应始终采用完整的英文描述符.此外,一般应采用小写字母,但类名.接口名以及任何非初始单词的第一个字母要大写.1.1 一般概念 n 尽量使用完整 ...
- 【JS复习笔记】06 方法
数组的方法: array.concat 一个数组去连接另一个数组,返回一个合成数组.var arrC=arrA.concat(arrB,'asd','sad',true,1.5); array.j ...
- js 倒计时 跳转
1. setTimeout() 方法用于在指定的毫秒数后调用函数或计算表达式. setTimeout() 只执行 code 一次.如果要多次调用,请使用 setInterval() 或者让 code ...
- .NET Core Roadmap
This post was written by Scott Hunter. It has been about two weeks since we shipped .NET Core / ASP. ...
- [翻译]:SQL死锁-阻塞探测
到了这篇,才是真正动手解决问题的时候,有了死锁之后就要分析死锁的原因,具体就是需要定位到具体的SQL语句上.那么如何发现产生死锁的问题本质呢?下面这篇讲的非常细了,还提到了不少实用的SQL,但对我个人 ...
- 从" ThinkPHP 开发规范 "看 PHP 的命名规范和开发建议
稍稍水一篇博客,摘抄自Think PHP 的开发规范,很有引导性,我们可以将这些规范实践到原生 PHP 中. 命名规范 使用ThinkPHP开发的过程中应该尽量遵循下列命名规范: 类文件都是以.cla ...
- (九)play之yabe项目【发表博文】
(九)play之yabe项目[发表博文] 博客分类: 框架@play framework 发表一篇博文 填充管理页面 从主页链接到管理页面时,只简单显示了登陆用户的名称 现在对显示的内容加以丰富 ...