亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

iOS中如何實現大尺寸圖片旋轉與縮放功能

發布時間:2021-07-15 13:56:48 來源:億速云 閱讀:146 作者:小新 欄目:移動開發

這篇文章將為大家詳細講解有關iOS中如何實現大尺寸圖片旋轉與縮放功能,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

前言

由于iPhone的硬件性能限制,直到iPhone 6s開始,才將最大內存拓展到2G。

可即使是如此,也不代表一個應用可使用的空間是2G。

一張10000 x 10000的圖片,如果通過UIImageJPEGRepresentation方法將圖片轉成內存數據,會有一個峰值波動。

這里的峰值其實是圖片在解壓時產生的位圖數據所占空間,然后才轉換成我們可以操作的NSData。

其計算公式是 W x H x 4 / 1024 / 1024 也就是 10000 x 10000 x4 /1024 / 1024 = 381.4(M)。

這里會產生381M的消耗,及時會被回收,但是想一下,如果圖片尺寸很大,數量很多的時候,很容易就會發生異常了。

本文將給大家詳細介紹關于iOS大尺寸圖片旋轉與縮放的相關內容,分享出來供大家參考學習,話不多說了,接下來說下具體的操作

旋轉

我們知道如果對一個UIImage對象進行旋轉操作,相信做項目時肯定會有用到 UIImage 這個類,可以有如下的方式

通過 CGContextDrawImage 進行圖片繪制

+ (UIImage *)image:(UIImage *)image rotation:(UIImageOrientation)orientation {

 long double rotate = 0.0;
 CGRect rect;
 float translateX = 0;
 float translateY = 0;
 float scaleX = 1.0;
 float scaleY = 1.0;

 switch (orientation) {
 case UIImageOrientationLeft:
 rotate = M_PI_2;
 rect = CGRectMake(0, 0, image.size.height, image.size.width);
 translateX = 0;
 translateY = -rect.size.width;
 scaleY = rect.size.width/rect.size.height;
 scaleX = rect.size.height/rect.size.width;
 break;
 default:
 rotate = 0.0;
 rect = CGRectMake(0, 0, image.size.width, image.size.height);
 translateX = 0;
 translateY = 0;
 break;
 }

 UIGraphicsBeginImageContext(rect.size);
 CGContextRef context = UIGraphicsGetCurrentContext();
 //做CTM變換
 CGContextTranslateCTM(context, 0.0, rect.size.height);
 CGContextScaleCTM(context, 1.0, -1.0);
 CGContextRotateCTM(context, rotate);
 CGContextTranslateCTM(context, translateX, translateY);

 CGContextScaleCTM(context, scaleX, scaleY);
 //繪制圖片
 CGContextDrawImage(context, CGRectMake(0, 0, rect.size.width, rect.size.height), image.CGImage);
 UIImage *newPic = UIGraphicsGetImageFromCurrentImageContext();
 return newPic;
}

這里有一個問題是,這里會創建一個新的圖片大小空間的,然后進行重新繪制。可能會存在一個隱患,就是當圖片尺寸過大的時候,就會出現內存占用過高的情況

接下來介紹一種另辟蹊徑的解決方法--通過給圖片添加濾鏡的方式。

既然操作的對象是圖片,那么它就會各種濾鏡展示。系統給我們提供了多大一百多種濾鏡,這里的濾鏡不單只顏色等狀態發生變化。

這其中就有我們需要的濾鏡Key inputTransform。

+ (UIImage *)getRotationImage:(UIImage *)image rotation:(CGFloat)rotation {

 CIImage *ciImage = [[CIImage alloc] initWithImage:image];
 CIFilter *filter = [CIFilter filterWithName:@"CIAffineTransform" keysAndValues:kCIInputImageKey, ciImage, nil];

 [filter setDefaults];
 CGAffineTransform transform = CATransform3DGetAffineTransform([self rotateTransform:CATransform3DIdentity clockwise:NO angle:rotation]);
 [filter setValue:[NSValue valueWithBytes:&transform objCType:@encode(CGAffineTransform)] forKey:@"inputTransform"];

 //根據濾鏡設置圖片
 CIContext *context = [CIContext contextWithOptions:@{kCIContextUseSoftwareRenderer : @(NO)}];
 CIImage *outputImage = [filter outputImage];
 CGImageRef cgImage = [context createCGImage:outputImage fromRect:[outputImage extent]];

 UIImage *result = [UIImage imageWithCGImage:cgImage];

 CGImageRelease(cgImage);

 return result;
}
+ (CATransform3D)rotateTransform:(CATransform3D)initialTransform clockwise:(BOOL)clockwise angle:(CGFloat)angle {

 CGFloat arg = angle*M_PI / 180.0f;
 if(!clockwise){
 arg *= -1;
 }
 //進行形變
 CATransform3D transform = initialTransform;
 transform = CATransform3DRotate(transform, arg, 0, 0, 1);
 CGFloat _flipState1 = 0;
 CGFloat _flipState2 = 0;
 transform = CATransform3DRotate(transform, _flipState1*M_PI, 0, 1, 0);
 transform = CATransform3DRotate(transform, _flipState2*M_PI, 1, 0, 0);

 return transform;
}

通過這種操作,可以利用GPU來進行圖片操作,可以一定程度的降低消耗,節約資源。

縮放

既然圖片很大,那么我們可以通過縮放的方式,來減小圖片的尺寸,減少內存消耗,進而降低異常風險。

我們通常采用UIImage提供的系統方法drawInRect 及其一系列的方法,來進行圖片縮放。

可是這種操作的缺陷和最開始介紹的旋轉一樣,其實質都是進行圖片的重新繪制。

通過繪制圖片的方式進行圖片縮放

+ (UIImage *)image:(UIImage *)image transformtoSize:(CGSize)Newsize {
 // 創建一個bitmap的context
 UIGraphicsBeginImageContext(Newsize);
 // 繪制改變大小的圖片
 [image drawInRect:CGRectMake(0, 0, Newsize.width, Newsize.height)];
 // 從當前context中創建一個改變大小后的圖片
 UIImage *TransformedImg=UIGraphicsGetImageFromCurrentImageContext();
 // 使當前的context出堆棧
 UIGraphicsEndImageContext();
 // 返回新的改變大小后的圖片
 return TransformedImg;
}

這里是內存消耗。通過看圖可以發現,針對大圖,在進行縮放的時候,內存消耗的峰值能達到426M,耗時在1.5s左右
由于我們使用的手機是iPhone X,在更低端的設備上,這是多么大的損耗,很容易發生異常

iOS中如何實現大尺寸圖片旋轉與縮放功能

既然上面的方法損耗很大,我們來看下另外的一種方式。

先看下內存消耗

iOS中如何實現大尺寸圖片旋轉與縮放功能

通過圖上可以看出,在進行圖片縮放的時候,內存有小幅增加,產生的消耗在18M,耗時也在1.5s左右。

這樣的效果是非常顯著的。下面來看代碼

+(UIImage *)resizeImage:(UIImage *)image toSize:(CGSize)size {

 CIImage *ciImage = [[CIImage alloc] initWithImage:image];
 //創建一個input image類型的濾鏡
 CIFilter *filter = [CIFilter filterWithName:@"CIAffineTransform" keysAndValues:kCIInputImageKey, ciImage, nil];
 //設置默認的濾鏡效果
 [filter setDefaults];

 //設置縮放比例
 CGFloat scale = 1;
 if (size.width != CGFLOAT_MAX) {
 scale = (CGFloat) size.width / image.size.width;
 } else if (size.height != CGFLOAT_MAX) {
 scale = (CGFloat) size.height / image.size.height;
 }

 //進行賦值
 CGAffineTransform transform = CGAffineTransformMakeScale(scale, scale);
 [filter setValue:[NSValue valueWithBytes:&transform objCType:@encode(CGAffineTransform)] forKey:@"inputTransform"];

 //通過GPU的方式來進行處理
 CIContext *context = [CIContext contextWithOptions:@{kCIContextUseSoftwareRenderer : @(NO)}];
 //根據濾鏡輸出圖片
 CIImage *outputImage = [filter outputImage];
 CGImageRef cgImage = [context createCGImage:outputImage fromRect:[outputImage extent]];
 //創建UIImage 對象,并釋放資源
 UIImage *result = [UIImage imageWithCGImage:cgImage];

 CGImageRelease(cgImage);

 return result;
}

可以發現我們這里使用的和旋轉是同樣的方式。通過給圖片添加濾鏡能夠很安全的實現我們的需求。

總結

1.針對巨幅圖片操作,可以采用這種思路:先生成一個尺寸小的縮略圖,然后在進行各種操作,可以降低資源消耗;

2.通過CoreImage.framework來進行圖片處理。

3.之前一直對CoreImage.framework的理解,只是其能夠對圖片和視頻添加那種可見的濾鏡,未曾想過這種濾鏡也支持縮放和旋轉。

? 為什么CoreImage.framework的方式能夠很安全呢?

該框架從iOS 5開始投入使用,通過對CoreGraphics.framework、CoreVideo.framework、Image I/O.framework進行數據處理,

可以自由在CPU和GPU之間切換運算方式,

可以最大限度的利用GPU來進行計算,降低內存消耗,

甚至可以對視頻進行實時濾鏡處理。

針對不能通過原生對UIView進行transform操作的時候,CoreImage.framework會是你的朋友。

最直接的來自文檔

Core Image is an image processing and analysis technology designed to provide near real-time processing for still and video images. It operates on image data types from the Core Graphics, Core Video, and Image I/O frameworks, using either a GPU or CPU rendering path. Core Image hides the details of low-level graphics processing by providing an easy-to-use application programming interface (API). You don't need to know the details of OpenGL, OpenGL ES, or Metal to leverage the power of the GPU, nor do you need to know anything about Grand Central Dispatch (GCD) to get the benefit of multicore processing. Core Image handles the details for you.

它已經幫你把所有東西都處理好了

關于“iOS中如何實現大尺寸圖片旋轉與縮放功能”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,使各位可以學到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

ios
AI

泊头市| 易门县| 大厂| 同仁县| 潮安县| 离岛区| 黄龙县| 广德县| 绿春县| 江门市| 永靖县| 华容县| 北宁市| 明溪县| 丹凤县| 青铜峡市| 赤城县| 尼勒克县| 凤山市| 田阳县| 抚松县| 大渡口区| 安图县| 玉门市| 虎林市| 丹寨县| 芜湖市| 合肥市| 舞钢市| 都昌县| 延津县| 化州市| 德格县| 甘肃省| 泾阳县| 泰顺县| 永顺县| 确山县| 佳木斯市| 达尔| 江油市|