iOS实现图片压缩的两种方法及图片压缩上传功能
两种压缩图片的方法:压缩图片质量(Quality),压缩图片尺寸(Size)。
压缩图片质量
NSData*data=UIImageJPEGRepresentation(image,compression); UIImage*resultImage=[UIImageimageWithData:data];
通过UIImage和NSData的相互转化,减小JPEG图片的质量来压缩图片。UIImageJPEGRepresentation::第二个参数compression取值0.0~1.0,值越小表示图片质量越低,图片文件自然越小。
压缩图片尺寸
UIGraphicsBeginImageContext(size); [imagedrawInRect:CGRectMake(0,0,size.width,size.height)]; resultImage=UIGraphicsGetImageFromCurrentImageContext(); UIGraphicsEndImageContext();
给定所需的图片尺寸size,resultImage即为原图image绘制为size大小的图片。
压缩图片使图片文件小于指定大小
如果对图片清晰度要求不高,要求图片的上传、下载速度快的话,上传图片前需要压缩图片。压缩到什么程度要看具体情况,但一般会设定一个图片文件最大值,例如100KB。可以用上诉两种方法来压缩图片。假设图片转化来的NSData对象为data,通过data.length即可得到图片的字节大小。
压缩图片质量
比较容易想到的方法是,通过循环来逐渐减小图片质量,直到图片稍小于指定大小(maxLength)。
+(UIImage*)compressImageQuality:(UIImage*)imagetoByte:(NSInteger)maxLength{ CGFloatcompression=1; NSData*data=UIImageJPEGRepresentation(image,compression); while(data.length>maxLength&&compression>0){ compression-=0.02; data=UIImageJPEGRepresentation(image,compression);//Whencompressionlessthanavalue,thiscodedosenotwork } UIImage*resultImage=[UIImageimageWithData:data]; returnresultImage; }
这样循环次数多,效率低,耗时长。
可以通过二分法来优化。
+(UIImage*)compressImageQuality:(UIImage*)imagetoByte:(NSInteger)maxLength{ CGFloatcompression=1; NSData*data=UIImageJPEGRepresentation(image,compression); if(data.length<maxLength)returnimage; CGFloatmax=1; CGFloatmin=0; for(inti=0;i<6;++i){ compression=(max+min)/2; data=UIImageJPEGRepresentation(image,compression); if(data.length<maxLength*0.9){ min=compression; }elseif(data.length>maxLength){ max=compression; }else{ break; } } UIImage*resultImage=[UIImageimageWithData:data]; returnresultImage; }
当图片大小小于maxLength,大于maxLength*0.9时,不再继续压缩。最多压缩6次,1/(2^6)=0.015625<0.02,也能达到每次循环compression减小0.02的效果。这样的压缩次数比循环减小compression少,耗时短。需要注意的是,当图片质量低于一定程度时,继续压缩没有效果。也就是说,compression继续减小,data也不再继续减小。压缩图片质量的优点在于,尽可能保留图片清晰度,图片不会明显模糊;缺点在于,不能保证图片压缩后小于指定大小。
压缩图片尺寸
与之前类似,比较容易想到的方法是,通过循环逐渐减小图片尺寸,直到图片稍小于指定大小(maxLength)。具体代码省略。同样的问题是循环次数多,效率低,耗时长。可以用二分法来提高效率,具体代码省略。这里介绍另外一种方法,比二分法更好,压缩次数少,而且可以使图片压缩后刚好小于指定大小(不只是<maxLength,>maxLength*0.9)。
+(UIImage*)compressImageSize:(UIImage*)imagetoByte:(NSUInteger)maxLength{ UIImage*resultImage=image; NSData*data=UIImageJPEGRepresentation(resultImage,1); NSUIntegerlastDataLength=0; while(data.length>maxLength&&data.length!=lastDataLength){ lastDataLength=data.length; CGFloatratio=(CGFloat)maxLength/data.length; CGSizesize=CGSizeMake((NSUInteger)(resultImage.size.width*sqrtf(ratio)),(NSUInteger)(resultImage.size.height*sqrtf(ratio)));//UseNSUIntegertopreventwhiteblank UIGraphicsBeginImageContext(size); //Useimagetodraw(drawInRect:),imageislargerbutmorecompressiontime //Useresultimagetodraw,imageissmallerbutlesscompressiontime [resultImagedrawInRect:CGRectMake(0,0,size.width,size.height)]; resultImage=UIGraphicsGetImageFromCurrentImageContext(); UIGraphicsEndImageContext(); data=UIImageJPEGRepresentation(resultImage,1); } returnresultImage; }
[resultImagedrawInRect:CGRectMake(0,0,size.width,size.height)];是用新图resultImage绘制,也可以用原图image来绘制。用原图绘制,压缩后图片更接近指定大小,但是压缩次数较多,耗时较长。一张大小为6064KB的图片,压缩图片尺寸,原图绘制与新图绘制结果如下
两种绘制方法压缩后大小很接近,与指定大小也很接近,但原图绘制压缩次数可达到新图绘制压缩次数的两倍。建议使用新图绘制,减少压缩次数。压缩后图片明显比压缩质量模糊。
需要注意的是绘制尺寸的代码
CGSizesize=CGSizeMake((NSUInteger)(resultImage.size.width*sqrtf(ratio)),(NSUInteger)(resultImage.size.height*sqrtf(ratio)));,每次绘制的尺寸size,要把宽width和高height转换为整数,防止绘制出的图片有白边。
压缩图片尺寸可以使图片小于指定大小,但会使图片明显模糊(比压缩图片质量模糊)。
两种图片压缩方法结合
如果要保证图片清晰度,建议选择压缩图片质量。如果要使图片一定小于指定大小,压缩图片尺寸可以满足。对于后一种需求,还可以先压缩图片质量,如果已经小于指定大小,就可得到清晰的图片,否则再压缩图片尺寸。
+(UIImage*)compressImage:(UIImage*)imagetoByte:(NSUInteger)maxLength{ //Compressbyquality CGFloatcompression=1; NSData*data=UIImageJPEGRepresentation(image,compression); if(data.length<maxLength)returnimage; CGFloatmax=1; CGFloatmin=0; for(inti=0;i<6;++i){ compression=(max+min)/2; data=UIImageJPEGRepresentation(image,compression); if(data.length<maxLength*0.9){ min=compression; }elseif(data.length>maxLength){ max=compression; }else{ break; } } UIImage*resultImage=[UIImageimageWithData:data]; if(data.length<maxLength)returnresultImage; //Compressbysize NSUIntegerlastDataLength=0; while(data.length>maxLength&&data.length!=lastDataLength){ lastDataLength=data.length; CGFloatratio=(CGFloat)maxLength/data.length; CGSizesize=CGSizeMake((NSUInteger)(resultImage.size.width*sqrtf(ratio)),(NSUInteger)(resultImage.size.height*sqrtf(ratio)));//UseNSUIntegertopreventwhiteblank UIGraphicsBeginImageContext(size); [resultImagedrawInRect:CGRectMake(0,0,size.width,size.height)]; resultImage=UIGraphicsGetImageFromCurrentImageContext(); UIGraphicsEndImageContext(); data=UIImageJPEGRepresentation(resultImage,compression); } returnresultImage; }
下面看下iOS图片压缩上传的实现方法
需求
很多时候我们上传图片经常遇到一些问题,要不就是图片质量变差,要不就是图片太大等等问题。这里,我找到了一个算是目前比较符合需求的解决方案。在原有基础上增加了动态压缩系数,改写成Swift版本。
实现思路
先调整分辨率,分辨率可以自己设定一个值,大于的就缩小到这分辨率,小余的就保持原本分辨率。然后再根据图片最终大小来设置压缩比,比如传入maxSize=30KB,最终计算大概这个大小的压缩比。基本上最终出来的图片数据根据当前分辨率能保持差不多的大小同时不至于太模糊,跟微信,微博最终效果应该是差不多的,代码仍然有待优化!
实现代码
Swift3.0之前旧版本压缩模式
//MARK:-降低质量 funcresetSizeOfImageData(source_image:UIImage,maxSize:Int)->NSData{ //先调整分辨率 varnewSize=CGSize(width:source_image.size.width,height:source_image.size.height) lettempHeight=newSize.height/1024 lettempWidth=newSize.width/1024 iftempWidth>1.0&&tempWidth>tempHeight{ newSize=CGSize(width:source_image.size.width/tempWidth,height:source_image.size.height/tempWidth) } elseiftempHeight>1.0&&tempWidth<tempHeight{ newSize=CGSize(width:source_image.size.width/tempHeight,height:source_image.size.height/tempHeight) } UIGraphicsBeginImageContext(newSize) source_image.drawAsPatternInRect(CGRect(x:0,y:0,width:newSize.width,height:newSize.height)) letnewImage=UIGraphicsGetImageFromCurrentImageContext() UIGraphicsEndImageContext() //先判断当前质量是否满足要求,不满足再进行压缩 varfinallImageData=UIImageJPEGRepresentation(newImage,1.0) letsizeOrigin=Int64((finallImageData?.length)!) letsizeOriginKB=Int(sizeOrigin/1024) ifsizeOriginKB<=maxSize{ returnfinallImageData! } //保存压缩系数 letcompressionQualityArr=NSMutableArray() letavg=CGFloat(1.0/250) varvalue=avg forvari=250;i>=1;i--{ value=CGFloat(i)*avg compressionQualityArr.addObject(value) } //调整大小 //说明:压缩系数数组compressionQualityArr是从大到小存储。 //思路:折半计算,如果中间压缩系数仍然降不到目标值maxSize,则从后半部分开始寻找压缩系数;反之从前半部分寻找压缩系数 finallImageData=UIImageJPEGRepresentation(newImage,CGFloat(compressionQualityArr[125]as!NSNumber)) ifInt(Int64((UIImageJPEGRepresentation(newImage,CGFloat(compressionQualityArr[125]as!NSNumber))?.length)!)/1024)>maxSize{ //拿到最初的大小 finallImageData=UIImageJPEGRepresentation(newImage,1.0) //从后半部分开始 foridxin126..<250{ letvalue=compressionQualityArr[idx] letsizeOrigin=Int64((finallImageData?.length)!) letsizeOriginKB=Int(sizeOrigin/1024) print("当前降到的质量:\(sizeOriginKB)") ifsizeOriginKB>maxSize{ print("\(idx)----\(value)") finallImageData=UIImageJPEGRepresentation(newImage,CGFloat(valueas!NSNumber)) }else{ break } } }else{ //拿到最初的大小 finallImageData=UIImageJPEGRepresentation(newImage,1.0) //从前半部分开始 foridxin0..<125{ letvalue=compressionQualityArr[idx] letsizeOrigin=Int64((finallImageData?.length)!) letsizeOriginKB=Int(sizeOrigin/1024) print("当前降到的质量:\(sizeOriginKB)") ifsizeOriginKB>maxSize{ print("\(idx)----\(value)") finallImageData=UIImageJPEGRepresentation(newImage,CGFloat(valueas!NSNumber)) }else{ break } } } returnfinallImageData! }
Swift3.0版本二分法压缩模式
//MARK:-降低质量 funcresetSizeOfImageData(source_image:UIImage!,maxSize:Int)->NSData{ //先判断当前质量是否满足要求,不满足再进行压缩 varfinallImageData=UIImageJPEGRepresentation(source_image,1.0) letsizeOrigin=finallImageData?.count letsizeOriginKB=sizeOrigin!/1024 ifsizeOriginKB<=maxSize{ returnfinallImageData!asNSData } //先调整分辨率 vardefaultSize=CGSize(width:1024,height:1024) letnewImage=self.newSizeImage(size:defaultSize,source_image:source_image) finallImageData=UIImageJPEGRepresentation(newImage,1.0); //保存压缩系数 letcompressionQualityArr=NSMutableArray() letavg=CGFloat(1.0/250) varvalue=avg vari=250 repeat{ i-=1 value=CGFloat(i)*avg compressionQualityArr.add(value) }whilei>=1 /* 调整大小 说明:压缩系数数组compressionQualityArr是从大到小存储。 */ //思路:使用二分法搜索 finallImageData=self.halfFuntion(arr:compressionQualityArr.copy()as![CGFloat],image:newImage,sourceData:finallImageData!,maxSize:maxSize) //如果还是未能压缩到指定大小,则进行降分辨率 whilefinallImageData?.count==0{ //每次降100分辨率 ifdefaultSize.width-100<=0||defaultSize.height-100<=0{ break } defaultSize=CGSize(width:defaultSize.width-100,height:defaultSize.height-100) letimage=self.newSizeImage(size:defaultSize,source_image:UIImage.init(data:UIImageJPEGRepresentation(newImage,compressionQualityArr.lastObjectas!CGFloat)!)!) finallImageData=self.halfFuntion(arr:compressionQualityArr.copy()as![CGFloat],image:image,sourceData:UIImageJPEGRepresentation(image,1.0)!,maxSize:maxSize) } returnfinallImageData!asNSData } //MARK:-调整图片分辨率/尺寸(等比例缩放) funcnewSizeImage(size:CGSize,source_image:UIImage)->UIImage{ varnewSize=CGSize(width:source_image.size.width,height:source_image.size.height) lettempHeight=newSize.height/size.height lettempWidth=newSize.width/size.width iftempWidth>1.0&&tempWidth>tempHeight{ newSize=CGSize(width:source_image.size.width/tempWidth,height:source_image.size.height/tempWidth) }elseiftempHeight>1.0&&tempWidth<tempHeight{ newSize=CGSize(width:source_image.size.width/tempHeight,height:source_image.size.height/tempHeight) } UIGraphicsBeginImageContext(newSize) source_image.draw(in:CGRect(x:0,y:0,width:newSize.width,height:newSize.height)) letnewImage=UIGraphicsGetImageFromCurrentImageContext() UIGraphicsEndImageContext() returnnewImage! } //MARK:-二分法 funchalfFuntion(arr:[CGFloat],image:UIImage,sourceDatafinallImageData:Data,maxSize:Int)->Data?{ vartempFinallImageData=finallImageData vartempData=Data.init() varstart=0 varend=arr.count-1 varindex=0 vardifference=Int.max whilestart<=end{ index=start+(end-start)/2 tempFinallImageData=UIImageJPEGRepresentation(image,arr[index])! letsizeOrigin=tempFinallImageData.count letsizeOriginKB=sizeOrigin/1024 print("当前降到的质量:\(sizeOriginKB)\n\(index)----\(arr[index])") ifsizeOriginKB>maxSize{ start=index+1 }elseifsizeOriginKB<maxSize{ ifmaxSize-sizeOriginKB<difference{ difference=maxSize-sizeOriginKB tempData=tempFinallImageData } end=index-1 }else{ break } } returntempData }
以上所述是小编给大家介绍的iOS实现图片压缩的两种方法,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的,在此也非常感谢大家对毛票票网站的支持!