app后端设计(12)--图片的处理
app上线后,不断接受用户的反馈,于是,反馈非常差的情况下,都会有app的改版。
一旦app的改版,都会有比较大的UI改动,一改动UI,那么图片的尺寸也就必须要改变。
在app后端设计(1)—api(http://blog.csdn.net/newjueqi/article/details/14053733)这篇文章中,我提到过app后台图片处理的一个基本原则,数据库中只保存原图的路径。对于同一张图片来说,针对不同机型,不同app版本所需要的不同尺寸,使用动态生成的策略,大体思路如下:
(1) 在图片的url末尾加上参数,声明需要生成的图片的新的尺寸,例如:户端需要图片(http://www.baidu.com/img/bdlogo.gif)的80*80的尺寸,则在图片的路径加上宽和高的参数(类似于CDN的机制) http://www.baidu.com/img/bdlogo.gif?w=80&h=80
(2) 服务器接收到图片的请求,先在缓存中查找这个尺寸的图片是否已经生成,如果已经在缓存中有记录,则不用重新生成。
(3) 如果该尺寸的图片还没生成,则生成新的图片尺寸,并把新生成的图片路径放在缓存中。
在app整个系统架构中,图片应该有两层缓存:
(1) app本地的图片缓存,当app中没有该图片时,才去服务取
(2) 服务器的图片缓存,记录图片不同尺寸的保存路径
我的建议是,如果不差钱,直接使用七牛的云存储的服务吧,云存储不但可以加速图片的下载上传,也能实现图片的大量操作。要知道,速度才是用户体验最直接的部分。
如果真的要自己实现图片的裁切,那么要考虑到图片操作是非常消耗CPU,内存,和大量的磁盘IO,所以在选择图片处理工具要慎重!!!
推荐使用GraphicsMagick,一个久经考验的图片处理软件,支持多个平台,而且支持多种语言的客服端。GraphicsMagick是ImageMagick的一个分支,相对于ImageMagick而言,TA处理速度更快,消耗资源更少,并且大的图片处理网站,如 Flickr and Etsy 已经在使用TA了。
使用GraphicsMagick时,最折腾的是怎么配GraphicsMagick环境,查阅了大量的文章,都注明在Linux下不能使用cmd.setSearchPath(path); ,但经过我实验,是可以的,而且配了这个的话,可以让linux和win下都运行同一段代码,只要把path放在配置文件中就好了。
下面我写的GraphicsMagick+Im4java图片裁剪的工具类,
- /**
- *
- */
- package com.bmob.worker.image;
- import java.awt.image.BufferedImage;
- import java.io.FileInputStream;
- import java.io.FileNotFoundException;
- import java.io.IOException;
- import java.io.InputStream;
- import java.util.concurrent.ExecutionException;
- /**
- */
- public class Image {
- /*
- 1、指定宽,高自适应,等比例缩放;
- 2、指定高, 宽自适应,等比例缩放;
- 3、指定最长边,短边自适应,等比例缩放;
- 4、指定最短边,长边自适应,等比例缩放;
- 5、指定最大宽高, 等比例缩放;
- 6、固定宽高, 居中裁剪)
- */
- public static int DefineWidth=1;
- public static int DefineHeight=2;
- public static int DefineLong=3;
- public static int DefineShort=4;
- public static int MaxWidthHeight=5;
- public static int DefineWidthHeight=6;
- /**
- * 图片缩放的方法
- *
- * @param mode
- 1、指定宽,高自适应,等比例缩放;
- 2、指定高, 宽自适应,等比例缩放;
- 3、指定最长边,短边自适应,等比例缩放;
- 4、指定最短边,长边自适应,等比例缩放;
- 5、指定最大宽高, 等比例缩放;
- 6、固定宽高, 居中裁剪)
- * @param src 源文件路径
- * @param desc 目标文件路径
- * @param width 指定宽
- * @param height 指定高
- * @param maxFrame 指定最长边
- * @param minFrame 指定最短边
- * @return
- * @throws Exception
- */
- public String resize(int mode, String src,String desc, int width, int height, int maxFrame, int minFrame) throws Exception {
- String str="";
- BHPApplication.init();
- // create command
- ConvertCmd cmd = this.getCmd();
- IMOperation op =null;
- if( mode==Image.DefineWidth ){
- op=this.resizeDefineWidth( src,desc, width, height);
- }else if( mode==Image.DefineHeight ){
- op=this.resizeDefineHeight( src,desc, width, height);
- }else if( mode==Image.DefineLong ){
- op=this.resizeDefineLong( src,desc, maxFrame);
- }else if( mode==Image.DefineShort ){
- op=this.resizeDefineShort( src,desc, minFrame);
- }else if( mode==Image.MaxWidthHeight ){
- op=this.resizeMaxWidthHeight( src,desc, width, height);
- }else if( mode==Image.DefineWidthHeight ){
- op=this.resizeDefineWidthHeight( src,desc, width, height);
- }
- cmd.run(op);
- return str;
- }
- //指定宽,高自适应,等比例缩放;
- public IMOperation resizeDefineWidth(String src,String desc, int width, int height){
- IMOperation op = new IMOperation();
- op.addImage(src);
- op.resize(width,null);
- op.addImage(desc);
- return op;
- }
- //指定高, 宽自适应,等比例缩放;
- public IMOperation resizeDefineHeight(String src,String desc, int width, int height){
- IMOperation op = new IMOperation();
- op.addImage(src);
- op.resize(null,height);
- op.addImage(desc);
- return op;
- }
- //指定最长边,短边自适应,等比例缩放;
- public IMOperation resizeDefineLong(String src,String desc, int maxFrame) throws Exception{
- InputStream is = new FileInputStream(src);//通过文件名称读取
- BufferedImage buff = ImageIO.read(is);
- int srcWidth=buff.getWidth();//得到图片的宽度
- int srcHeight=buff.getHeight(); //得到图片的高度
- is.close(); //关闭Stream
- IMOperation op = new IMOperation();
- op.addImage(src);
- if( srcWidth>srcHeight ){
- op.resize(maxFrame,null);
- }else{
- op.resize(null,maxFrame);
- }
- op.addImage(desc);
- return op;
- }
- //指定最短边,长边自适应,等比例缩放;
- public IMOperation resizeDefineShort(String src,String desc, int minFrame) throws Exception {
- InputStream is = new FileInputStream(src);//通过文件名称读取
- BufferedImage buff = ImageIO.read(is);
- int srcWidth=buff.getWidth();//得到图片的宽度
- int srcHeight=buff.getHeight(); //得到图片的高度
- is.close(); //关闭Stream
- IMOperation op = new IMOperation();
- op.addImage(src);
- if( srcWidth<srcHeight ){
- op.resize(minFrame,null);
- }else{
- op.resize(null,minFrame);
- }
- op.addImage(desc);
- return op;
- }
- //指定最大宽高, 等比例缩放;
- public IMOperation resizeMaxWidthHeight(String src,String desc, int width, int height){
- IMOperation op = new IMOperation();
- op.addImage(src);
- op.resize(width,height,'!');
- op.addImage(desc);
- return op;
- }
- //固定宽高, 居中裁剪
- public IMOperation resizeDefineWidthHeight(String src,String desc, int width, int height){
- IMOperation op = new IMOperation();
- op.addImage(src);
- op.gravity("center").extent(width, height);
- op.addImage(desc);
- return op;
- }
- public ConvertCmd getCmd(){
- ConvertCmd cmd = new ConvertCmd(true); //set true, use GraphicsMagick
- String path = "/usr/local/GraphicsMagick/bin"; //GraphicsMagick安装路径
- cmd.setSearchPath(path);
- return cmd;
- }
- }
如果您觉得这系列的文章对你有所帮助,欢迎打赏。
支付宝账号:190678908@qq.com 收款人:曾健生
新建了“app后端技术” 交流qq群:254659220
app后端设计(12)--图片的处理的更多相关文章
- app后端设计--总目录 (转)
特此说明,我转载的!!! app后端设计(1)--api app后端设计(2)--xmpp的使用 app后端设计(3)--短信,邮件,推送服务 app后端设计(4)-- 通讯的安全性 app后端设计( ...
- app后端设计--总目录
做了3年app相关的系统架构,api设计,先后在3个创业公司中工作,经历过手机网页端,android客户端,iphone客户端,现就职于app云后端平台bmob(想了解bmob点击这里).其中的乐与苦 ...
- [置顶] app后端设计--总目录
版权声明:本文为博主原创文章,未经博主允许不得转载. 做了3年app相关的系统架构,api设计,先后在3个创业公司中工作,经历过手机网页端,Android客户端,iphone客户端,现就职于app云后 ...
- app后端设计(php)
来源:http://blog.csdn.net/column/details/mobilebackend.html?page=1 做了3年app相关的系统架构,api设计,先后在3个创业公司中工作,经 ...
- app后端设计(0)--总文件夹
原文:http://blog.csdn.net/newjueqi/article/details/19003775 做了接近两年app相关的系统架构,api设计,先后在两个创业公司中工作,经历过手机网 ...
- app后端设计(0)--总目录(转)
原文:http://blog.csdn.net/newjueqi/article/details/19003775 做了接近两年app相关的系统架构,api设计,先后在两个创业公司中工作,经历过手机网 ...
- app后端设计(11)-- 系统架构(2014.12.05更新)
个人认为,在小型的创业团队中,特别是以应用产品为主,在架构后台的时候,需要集中精力解决自身业务上的问题,不是花时间解决第三方已经解决的问题,简单点来说,就是能用第三方服务就使用第三方的服务.基于这个原 ...
- app后端设计(3)--短信,邮件,推送服务(2014.12.05更新)
在app的后端设计中,免不了消息的推送,短信,邮件等服务,下面就个人的开发经验谈谈这方面. (1)最重要的是,各种推送一定要放在队列系统中处理,不然会严重影响api的响应时间. (2)短信方面 以前我 ...
- app后端设计(7)-- 项目管理
移动互联网行业是个快速发展的行业,需求不断变化,产品更新快.基于移动互联网的以上特点,在开发产品的过程中,我们放弃了传统的瀑布流开发模型,引入了精益的理念和scrum这个敏捷开发框架,下面谈谈实施过程 ...
随机推荐
- 对contentoffset的理解
今天遇到一个问题,在写瀑布流时,竖屏的时候可以正常实现,在手机变成横屏后,总是显示不全. 最终查了两个小时,查到了导致这个的原因,是自己的判断cell是否在当前显示区域的方法写错了. 根本原因是没有很 ...
- UpdatePanel里的Repeater和DropDownList
在updatepanel里使用dropdownlist的AutoPostBack,正常情况下都可以局部刷新. 但是,如果updatepanel下是Repeater,repeater里绑定dropdow ...
- Flask + WSGI + Nginx 云部署
这几天学着用flask写一些rest api,然后部署到云上.这个过程虽然网上有很多的教程,但还是遇到不少的问题! 采用flask的原因是因为它比较容易上手吧.用flask有专门restful api ...
- javaweb学习总结 servlet开发(一)
转载:http://www.cnblogs.com/xdp-gacl/p/3760336.html 这里主要是将其加入自己的理解过一遍. 这里的代码全在eclipse java ee中执行的. 一.s ...
- business knowledge
Finance knowledge Trading---At the core of our business model is Trading, which involves the buying ...
- PHPnow在win8下安装失败的解决办法
提示: 安装服务[ Apache_pn ]失败,可能原因如下:1.服务名已存在,请卸载或使用不同服务名.2.非管理员权限,不能操作Window NT服务. 解决方案: 搜索:命令提示符 , 右键以 ...
- PowerDesigner怎样才能在修改表的字段Name的时候Code不自动跟着变
怎样才能在修改表的字段Name的时候,Code不自动跟着变 tools-> General Options-> Dialog:Operation Modes: 去掉 NameToC ...
- Python 基礎 - 文件的操作
在來我們來玩一下文件操作,這個在未來工作上,也是會很常用到的功能 Python2.7中,可以用file()來打開文件,而在Python3中,一律都是用open(),接下來在當前目錄下,先建立一個空文件 ...
- 利用 ipset 封禁大量 IP
使用 iptables 封 IP,是一种比较简单的应对网络攻击的方式,也算是比较常见.有时候可能会封禁成千上万个 IP,如果添加成千上万条规则,在一台注重性能的服务器或者本身性能就很差的设备上,这就是 ...
- Struts2学习笔记(拦截器配置添加)
一.拦截器工作原理: 根据Struts2的工作原理图,拦截器在action执行前进行顺序调用,之后执行Action并返回结果字符串,再逆序调用拦截器.(结构类似递归方式...)大部分时候,拦截器方法都 ...