美文网首页IOS开发iOS开发指南iOS开发
iOS 图片压缩限制大小最优解

iOS 图片压缩限制大小最优解

作者: leonardni | 来源:发表于2017-06-27 20:40 被阅读7527次

概要:

  1. 图片的两种压缩方法

1.1 压缩图片质量
1.2 压缩图片尺寸

  1. 压缩图片使图片文件小于指定大小

2.1 压缩图片质量
2.2 压缩图片尺寸
2.3 两种图片压缩方法结合

文章更新记录


2017-12-17日

1. 修改实现方法(改为UIImage的分类实现)更新demo工程

一、两种图片压缩方法


两种压缩图片的方法:压缩图片质量(Quality),压缩图片尺寸(Size)。

1.1 压缩图片质量

NSData *data = UIImageJPEGRepresentation(image, compression);
UIImage *resultImage = [UIImage imageWithData:data];

通过UIImageNSData的相互转化,减小 JPEG 图片的质量来压缩图片。UIImageJPEGRepresentation::第二个参数compression 取值 0.0~1.0,值越小表示图片质量越低,图片文件自然越小。

1.2 压缩图片尺寸

UIGraphicsBeginImageContext(size);
[image drawInRect:CGRectMake(0, 0, size.width, size.height)];
resultImage = UIGraphicsGetImageFromCurrentImageContext();
UIGraphicsEndImageContext();

给定所需的图片尺寸 size,resultImage 即为原图 image 绘制为 size 大小的图片。

二、 压缩图片使图片文件小于指定大小


如果对图片清晰度要求不高,要求图片的上传、下载速度快的话,上传图片前需要压缩图片。压缩到什么程度要看具体情况,但一般会设定一个图片文件最大值,例如100 KB。可以用上诉两种方法来压缩图片。假设图片转化来的 NSData对象为data,通过data.length即可得到图片的字节大小。

2.1 压缩图片质量

比较容易想到的方法是,通过循环来逐渐减小图片质量,直到图片稍小于指定大小(maxLength)

UIImage+WLCompress.m
- (NSData *)compressQualityWithMaxLength:(NSInteger)maxLength {
    CGFloat compression = 1;
    NSData *data = UIImageJPEGRepresentation(self, compression);
    while (data.length > maxLength && compression > 0) {
        compression -= 0.02;
        data = UIImageJPEGRepresentation(self, compression); // When compression less than a value, this code dose not work
    }
    return data;
}

这样循环次数多,效率低,耗时长。

可以通过二分法来优化。

UIImage+WLCompress.m
- (NSData *)compressQualityWithMaxLength:(NSInteger)maxLength {
   CGFloat compression = 1;
    NSData *data = UIImageJPEGRepresentation(self, compression);
    if (data.length < maxLength) return data;
    CGFloat max = 1;
    CGFloat min = 0;
    for (int i = 0; i < 6; ++i) {
        compression = (max + min) / 2;
        data = UIImageJPEGRepresentation(self, compression);
        if (data.length < maxLength * 0.9) {
            min = compression;
        } else if (data.length > maxLength) {
            max = compression;
        } else {
            break;
        }
    }
    return data;
}

当图片大小小于 maxLength,大于 maxLength * 0.9 时,不再继续压缩。最多压缩 6 次,1/(2^6) = 0.015625 < 0.02,也能达到每次循环 compression减小 0.02 的效果。这样的压缩次数比循环减小 compression少,耗时短。需要注意的是,当图片质量低于一定程度时,继续压缩没有效果。也就是说,compression继续减小,data 也不再继续减小。

压缩图片质量的优点在于,尽可能保留图片清晰度,图片不会明显模糊;
缺点在于,不能保证图片压缩后小于指定大小。

2.2 压缩图片尺寸

与之前类似,比较容易想到的方法是,通过循环逐渐减小图片尺寸,直到图片稍小于指定大小(maxLength)。具体代码省略。同样的问题是循环次数多,效率低,耗时长。可以用二分法来提高效率,具体代码省略。这里介绍另外一种方法,比二分法更好,压缩次数少,而且可以使图片压缩后刚好小于指定大小(不只是 < maxLength, > maxLength * 0.9)。

objectiveC:

-(NSData *)compressBySizeWithMaxLength:(NSUInteger)maxLength{
    UIImage *resultImage = self;
    NSData *data = UIImageJPEGRepresentation(resultImage, 1);
    NSUInteger lastDataLength = 0;
    while (data.length > maxLength && data.length != lastDataLength) {
        lastDataLength = data.length;
        CGFloat ratio = (CGFloat)maxLength / data.length;
        CGSize size = CGSizeMake((NSUInteger)(resultImage.size.width * sqrtf(ratio)),
                                 (NSUInteger)(resultImage.size.height * sqrtf(ratio))); // Use NSUInteger to prevent white blank
        UIGraphicsBeginImageContext(size);
        // Use image to draw (drawInRect:), image is larger but more compression time
        // Use result image to draw, image is smaller but less compression time
        [resultImage drawInRect:CGRectMake(0, 0, size.width, size.height)];
        resultImage = UIGraphicsGetImageFromCurrentImageContext();
        UIGraphicsEndImageContext();
        data = UIImageJPEGRepresentation(resultImage, 1);
    }
    return data;
}

[resultImage drawInRect:CGRectMake(0, 0, size.width, size.height)];
是用新图 resultImage 绘制,也可以用原图 image 来绘制。
用原图绘制,压缩后图片更接近指定大小,但是压缩次数较多,耗时较长。一张大小为 6064 KB 的图片,压缩图片尺寸,原图绘制与新图绘制结果如下


两种绘制方法压缩后大小很接近,与指定大小也很接近,但原图绘制压缩次数可达到新图绘制压缩次数的两倍。建议使用新图绘制,减少压缩次数。压缩后图片明显比压缩质量模糊。

需要注意的是绘制尺寸的代码CGSize size = CGSizeMake((NSUInteger)(resultImage.size.width * sqrtf(ratio)), (NSUInteger)(resultImage.size.height * sqrtf(ratio)));,每次绘制的尺寸 size,要把宽 width 和 高 height 转换为整数,防止绘制出的图片有白边。

压缩图片尺寸可以使图片小于指定大小,但会使图片明显模糊(比压缩图片质量模糊)。

2.3 两种图片压缩方法结合

如果要保证图片清晰度,建议选择压缩图片质量。如果要使图片一定小于指定大小,压缩图片尺寸可以满足。对于后一种需求,还可以先压缩图片质量,如果已经小于指定大小,就可得到清晰的图片,否则再压缩图片尺寸。

UIImage+WLCompress.m
-(NSData *)compressWithMaxLength:(NSUInteger)maxLength{
    // Compress by quality
    CGFloat compression = 1;
    NSData *data = UIImageJPEGRepresentation(self, compression);
    //NSLog(@"Before compressing quality, image size = %ld KB",data.length/1024);
    if (data.length < maxLength) return data;
    
    CGFloat max = 1;
    CGFloat min = 0;
    for (int i = 0; i < 6; ++i) {
        compression = (max + min) / 2;
        data = UIImageJPEGRepresentation(self, compression);
        //NSLog(@"Compression = %.1f", compression);
        //NSLog(@"In compressing quality loop, image size = %ld KB", data.length / 1024);
        if (data.length < maxLength * 0.9) {
            min = compression;
        } else if (data.length > maxLength) {
            max = compression;
        } else {
            break;
        }
    }
    //NSLog(@"After compressing quality, image size = %ld KB", data.length / 1024);
    if (data.length < maxLength) return data;
    UIImage *resultImage = [UIImage imageWithData:data];
    // Compress by size
    NSUInteger lastDataLength = 0;
    while (data.length > maxLength && data.length != lastDataLength) {
        lastDataLength = data.length;
        CGFloat ratio = (CGFloat)maxLength / data.length;
        //NSLog(@"Ratio = %.1f", ratio);
        CGSize size = CGSizeMake((NSUInteger)(resultImage.size.width * sqrtf(ratio)),
                                 (NSUInteger)(resultImage.size.height * sqrtf(ratio))); // Use NSUInteger to prevent white blank
        UIGraphicsBeginImageContext(size);
        [resultImage drawInRect:CGRectMake(0, 0, size.width, size.height)];
        resultImage = UIGraphicsGetImageFromCurrentImageContext();
        UIGraphicsEndImageContext();
        data = UIImageJPEGRepresentation(resultImage, compression);
        //NSLog(@"In compressing size loop, image size = %ld KB", data.length / 1024);
    }
    //NSLog(@"After compressing size loop, image size = %ld KB", data.length / 1024);
    return data;
}

演示demo地址:ImagePickDemo
转自于
iOS 图片压缩方法

相关文章

  • iOS 图片压缩限制大小最优解

    iOS 图片压缩限制大小最优解 图片的两种压缩方法 1.1 压缩图片质量 1.2 压缩图片尺寸 压缩图片使图片文件...

  • iOS 图片压缩限制大小最优解

    概要: 图片的两种压缩方法1.1 压缩图片质量1.2 压缩图片尺寸压缩图片使图片文件小于指定大小2.1 压缩图片质...

  • iOS 图片压缩限制大小最优解--转载

    本文章转自 作者 ITCodeShare:https://www.jianshu.com/p/99c3e6a6c0...

  • iOS 图片压缩限制大小

    概要: 1、图片的两种压缩方法 1.1 压缩图片质量 1.2 压缩图片尺寸2、压缩图片使图片文件小于指定大小 ...

  • iOS 图片压缩限制大小

    一、两种图片压缩方法 两种压缩图片的方法:压缩图片质量(Quality),压缩图片尺寸(Size)。 压缩图片质量...

  • iOS-图片限制

    //先限制图片大小,一般用了节省流量iOS和安卓端统一宽度640或者720; 图片大小超过1M,就要循环压缩图片...

  • 判断 png 图片色彩位数(深度)

    缘由 Android 项目 Apk 大小限制,对历史项目中的未压缩图片进行筛检并处理。图片压缩(如: TinyPn...

  • iOS 图片压缩 (尺寸大小)

    /// 压缩图片 /// - Parameters: /// - size: 目标尺寸 /// - image...

  • iOS 图片压缩方法

    iOS 图片压缩方法 更多图片处理方法见图片组件 BBWebImage iOS 图片压缩方法 两种图片压缩方法 两...

  • 宾馆管理系统

    图片存储表设计(Sys_Image) 说明:限制图片大小,后台不压缩图片 脚本 宾馆房间表(Hotel_Room)...

网友评论

  • 最讨厌梅雨天啦:使用阶梯式判断压缩倍数,性能上来说怎么样?就比如,if(data.length>maxLength*1.5),则compresssion=(max+min)/2 ;
    最讨厌梅雨天啦:先用阶梯式压缩一遍,再用二分法
  • zcc_ios:压缩尺寸只有在datalength = 0.9*maxlength才执行吗..感觉好怪...
    默默装作不知道:他是保证break的图片在maxlength的0.9-1之间
  • 旧夏2014:gif图可以压缩质量吗
    leonardni:原理差不多,gif压缩现有的代码可能需要更改下
  • _小沫:最多压缩6次是自己限定的?
    _小沫:@ITCodeShare 哦 了解 就是要接近于0.02
    leonardni:这里是有个计算公式:二分法的精度范围 范围/2^n 这里范围0~1 1/2^6 = 0.015625
    leonardni:你画个图看下,0~1 二分法运行六次,看下可以做到多大的精度。这个可以根据自己需要调整。
  • b8e6682d009f:首先谢谢楼主的分享,方法很好。有些问题问一下:1、UIImage *image = [UIImage imageNamed:@"test2"];
    NSData *data = UIImageJPEGRepresentation(image, 1.0);我用ps处理得到的最优画质的jpg图片,用JPEG得到的为啥还是比原图大呢,想问下这个偏差是怎么来的。2、尺寸压缩ratio是data的length的比例,为什么size用的sqrtf(ratio)来算呢,data的length的比例和size之间是有什么关系吗。
    leonardni:能想出这样的方法的人我觉得真的很厉害啊,最大的保证图像压缩质量的同时又能保证压缩算法运行的次数,也就是我们说的压缩速度,简直太厉害了
    leonardni:第二个问题是一个算法来的,ratio: CGFloat = CGFloat(maxLength) / CGFloat(data.count) 假如我要的是一个1M的JPEG照片 原图为5M radio = 0.2 < 1, 开方后 = 0.44 ,进行一次循环得到的值是原图尺寸的0.44倍,比你直接*0.2 来的尺寸要大,应该是为了保证图像质量考虑的,当你对一个数无限次开平方根的时候,这个数会无限接近于1。也就是这个算法最终达到无法压缩尺寸退出的条件,毕竟不能让程序死在循环中吧
    leonardni:第一个问题
    你提的第一个问题,要解答的话只能从JPEG图片压缩算法入手,机器能识别的只能是0101这样的语言,从显示的角度看要实现能显示真彩色至少是24位显示一个像素点,也就是red 8bit yellow 8bit green 8bit。这就是为什么我们平时写色值的时候要/255.0的原因了,[UIColor colorWithRed:100.0f/255.0f green:100.0f/255.0f blue:100.0f/255.0f alpha:0.8]; 一个颜色8bit 刚好是255 。JPEG算法还原,还原出的机器数据(UIImage大小)会有一定的偏差。
  • 翻滚的炒勺2013:是我不会用吗,图片根本没有压缩啊
    leonardni:直接返回nsdata 回来,这里返回uiimage 只是方便看压缩还原后的图片质量所以返回uiimage。你看到的数据没有压缩是因为你把uiimage 直接用UIImageJPEGRepresentation(image,1.0) 变成nadata所致,相当于这里你用压缩方法JPEG反复压缩,然后又用1.0还原了。详细的原理你可以看下JPEG图像压缩算法
    leonardni:你是不是直接考了函数直接用了,返回uiimage 后又用 uiimage - > data 上传?
  • __阳阳:在压缩图片质量的时候, for循环里边的判断条件指定大小 maxLength * 0.9 是什么意思 ?
    __阳阳:@ITCodeShare 恩恩, 明白了, 谢谢
    leonardni:你画个集合图就清楚了,确保数据大小在规定大小的90%~100%之间

本文标题:iOS 图片压缩限制大小最优解

本文链接:https://www.haomeiwen.com/subject/tmzecxtx.html