偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

iOS相冊Moment功能的優(yōu)化方案

移動開發(fā)
最近在開發(fā)公司產(chǎn)品Perfect365的Gallery模塊, 包括按日期排序的Moment以及Album這兩個模塊. Moment功能和系統(tǒng)相冊類似, 就是根據(jù)圖片的日期信息進行排序,然后按照不同日期分section顯示.

最近在開發(fā)公司產(chǎn)品Perfect365的Gallery模塊, 包括按日期排序的Moment以及Album這兩個模塊. Moment功能和系統(tǒng)相冊類似, 就是根據(jù)圖片的日期信息進行排序, 然后按照不同日期分section顯示.

 

18.jpg

Moment的實現(xiàn)思路很簡單: 先遍歷系統(tǒng)的所有相冊, 然后獲取每個相冊內(nèi)圖片的日期信息, 根據(jù)日期進行分類和排序, ***把枚舉完的所有數(shù)據(jù)放到界面上來顯示。示例代碼如下:

 


  1. NSSortDescriptor *sort = [NSSortDescriptor sortDescriptorWithKey:@"date" ascending:NO]; 
  2.  
  3. [objects sortUsingDescriptors:@[sort]]; 
  4.  
  5. MomentCollection *lastGroup = nil; NSMutableArray *ds = [[NSMutableArray alloc] init]; 
  6.  
  7. for (ALAsset *asset in objects) 
  8.     @autoreleasepool 
  9.     { 
  10.         NSDateComponents *components = [[NSCalendar currentCalendar] components:NSDayCalendarUnit   | 
  11.                                                                                 NSMonthCalendarUnit | 
  12.                                                                                 NSYearCalendarUnit 
  13.                                                                        fromDate:[asset date]]; 
  14.         NSUInteger month = [components month]; 
  15.         NSUInteger year  = [components year]; 
  16.         NSUInteger day   = [components day]; 
  17.          
  18.         if (!lastGroup || lastGroup.year!=year || lastGroup.month!=month || lastGroup.day!=day) 
  19.         { 
  20.             lastGroup = [MomentCollection new]; [ds addObject:lastGroup]; 
  21.              
  22.             lastGroup.month = month; lastGroup.year = year; lastGroup.day = day; 
  23.         } 
  24.          
  25.         ALAsset *lPhoto    = [lastGroup.assetObjs lastObject]; 
  26.         NSURL   *lPhotoURL = [lPhoto valueForProperty:ALAssetPropertyAssetURL]; 
  27.         NSURL   *photoURL  = [asset  valueForProperty:ALAssetPropertyAssetURL]; 
  28.         if (![lPhotoURL isEqual:photoURL]) 
  29.         { 
  30.             [lastGroup.assetObjs addObject:asset]; 
  31.         } 
  32.     } 

So far so good, 接下來創(chuàng)建UICollectionView, 設(shè)置好dataSource就可以顯示moment圖片了. 起初我也是這么認為的, 但是對于開發(fā)一款擁有6500萬用戶的App來說everything is possible. 版本發(fā)布之后, 很多用戶反饋打開相冊后App直接freeze掉, what the hell is this? QA測試時一切OK的呀. 好吧, 繼續(xù)騷擾用戶詢問到底是神馬情況, 用戶回復(fù): 我手機里面放了30k+圖片, 占了20G+的存儲空間. OH MY GOD!!!

優(yōu)化方案

對于Moment功能, 肯定需要遍歷完系統(tǒng)內(nèi)的所有相冊圖片, 然后再按日期排序后顯示給用戶, 那優(yōu)化就只能在枚舉和排序這兩部分來壓榨了. 經(jīng)過2天的苦思冥想決定采用分批加載+取尾排序的方案來優(yōu)化. 具體思路為: 如果用戶設(shè)備內(nèi)的圖片比較多, 不是等所有圖片都枚舉排序完了再顯示, 而是枚舉每隔一定數(shù)量的圖片(e.g. 50張)后就拋出去(放到NSOperationQueue里)按日期分類并排序, 再顯示給用戶, 這樣讓用戶看到我們動態(tài)加載圖片的過程, 讓他知道我們的程序still alive, 并且在不斷的加載圖片. 但是一般情況下排序的耗時會大于圖片的枚舉, 也就是***個50張排完序后, 前面枚舉放到Queue里面等待排序的已經(jīng)有好幾批了, 那么我們只對***一批的圖片再排序(也就是取尾)并清空當前的Queue, 因為中間的幾批數(shù)據(jù)已經(jīng)makes no sense了. 方案詳細流程圖如下:

19.jpg

曲線流程圖

為了***程度的減輕動態(tài)加載后刷新顯示對用戶造成的突兀感, 在顯示之前需要判斷用戶是否在滑動頁面, 只有頁面靜止的時候刷新顯示. 但對于全部圖片枚舉完成后的***一批數(shù)據(jù)則要暫時保存住(否則就木有東東顯示了), 待用戶停止滑動后reloadData.

分批加載

Moment需要按日期分類顯示(***的顯示在最前面), 所以在枚舉相冊的時候可以先從camera roll開始(一般用戶拍攝的照片相對導(dǎo)入的圖片會早一點). 加載到50的倍數(shù)張后就拋到queue里面等待排序, 一個相冊枚舉完后再繼續(xù)遍歷其余的相冊...

 


  1. - (void)getPhotosWithGroupTypes:(ALAssetsGroupType)types 
  2.                     batchReturn:(BOOL)batch 
  3.                      completion:(void (^)(BOOL ret, id obj))completion 
  4.     self.batchBlock        = completion; 
  5.     NSMutableArray *tmpArr = [[NSMutableArray alloc] init]; 
  6.      
  7.     [self.assetLibary enumerateGroupsWithTypes:types 
  8.      
  9.                                     usingBlock:^(ALAssetsGroup *group, BOOL *stop) 
  10.     { 
  11.         if (self.stopEnumeratePhoto) {*stop = YES; return;} 
  12.         NSInteger gType = [[group valueForProperty:ALAssetsGroupPropertyType] integerValue]; 
  13.         if (group && (gType != ALAssetsGroupPhotoStream)) 
  14.         { 
  15.             [group setAssetsFilter:[ALAssetsFilter allPhotos]]; 
  16.              
  17.             [group enumerateAssetsWithOptions:NSEnumerationReverse 
  18.                                    usingBlock:^(ALAsset *result, NSUInteger index, BOOL *stop) 
  19.             { 
  20.                 if (self.stopEnumeratePhoto) {*stop = YES; return;} 
  21.                  
  22.                 if (result) [tmpArr addObject:result]; 
  23.                  
  24.                 if (batch && !([tmpArr count]%50)) [self addQueueWithData:tmpArr final:NO]; 
  25.             }]; 
  26.         } 
  27.         else if (nil == group) 
  28.         { 
  29.             [self addQueueWithData:tmpArr final:YES]; 
  30.         } 
  31.     }failureBlock:nil]; 

取尾排序

每組批次的圖片都加到一個串行queue隊列里面等待排序, 某個批次的排序完成之后取當前queue***一個(也就是***過來的枚舉圖片)繼續(xù)執(zhí)行排序, 并清空當前的queue. 也就是在下面的sortMomentWithDate:final:函數(shù)里面調(diào)用cleanQueueAfterRoundOperation.

 


  1. - (void)addQueueWithData:(NSMutableArray *)data final:(BOOL)final 
  2.     NSMutableArray *rawData = [NSMutableArray arrayWithArray:data]; 
  3.      
  4.     NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^ 
  5.     { 
  6.         [self sortMomentWithDate:rawData final:final]; 
  7.     }]; 
  8.      
  9.     [self.operQueue addOperation:op]; 
  10.  
  11. - (void)cleanQueueAfterRoundOperation 
  12.     if (self.operQueue == nil) return
  13.      
  14.     if (self.operQueue.operationCount > 1
  15.     { 
  16.         NSArray *queueArr = self.operQueue.operations; 
  17.         NSMutableArray *opArr = [NSMutableArray arrayWithArray:queueArr]; 
  18.          
  19.         [opArr removeLastObject]; [opArr removeLastObject]; 
  20.         [opArr makeObjectsPerformSelector:@selector(cancel)]; 
  21.     } 

刷新CollectionView顯示圖片

中間批次按日期分類過的數(shù)據(jù)ready后, 在reloadData之前先判斷一下當前用戶是否在滑動collectionView, 如果是非scroll狀態(tài)則刷新顯示, 否則直接drop掉, 但是對于***一批數(shù)據(jù)需要先存儲著, 并在scrollViewDidEndDragging和scrollViewDidEndDecelerating里面判斷, 一旦用戶停止滑動了就立即刷新到collectionView上.

 


  1. [[ImageDataAPI sharedInstance] getMomentsWithBatchReturn:YES 
  2.                                                ascending:NO 
  3.                                               completion:^(BOOL done, id obj) 
  4.     NSMutableArray *dArr = (NSMutableArray *)obj; 
  5.      
  6.     if (dArr != nil && [dArr count]) 
  7.     { 
  8.         if (!self.momentView.dragging && !self.momentView.decelerating) 
  9.         { 
  10.             dispatch_async(dispatch_get_main_queue(), ^ 
  11.             { 
  12.                 [self reloadWithData:dArr]; 
  13.             }); 
  14.         } 
  15.         else 
  16.         { 
  17.             if (done) {self.backupArr = dArr} 
  18.         } 
  19.     } 
  20. }]; 
  21.  
  22. - (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate 
  23.     if (!decelerate && self.backupArr) 
  24.     { 
  25.         dispatch_async(dispatch_get_main_queue(), ^{ 
  26.             [self reloadWithData:self.backupArr]; 
  27.             self.backupArr = nil; // done refresh 
  28.         }); 
  29.     } 
  30.  
  31. - (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView 
  32.     if (self.backupArr) 
  33.     { 
  34.         dispatch_async(dispatch_get_main_queue(), ^{ 
  35.             [self reloadWithData:self.backupArr]; 
  36.             self.backupArr = nil; // done refresh 
  37.         }); 
  38.     } 

后續(xù)改進思路

按上述方案不管設(shè)備圖片有多少, 基本可以正常打開相冊并加載圖片. 但還是有很多需要繼續(xù)改進的地方: e.g.

  • 中間批次的數(shù)據(jù)ready后也可以先存儲著, 待用戶停止滑動后在reload上去, 而不是簡單的drop掉.

  • 排序還需要再優(yōu)化. 現(xiàn)在***批50張圖片排序后, 第二批進入排序的200張圖片又需要重新分類排序, 中間批次數(shù)據(jù)只是為了先顯示給用戶看. 是不是第200張圖片可以只對后面的150張進行排序, 也就是后面150張有新的日期, 則新建section, 相同日期直接insert到前面去. 這個還需要后面再研究...

以上只是自己的一些優(yōu)化思路, 如果有更好的方案, 歡迎留言交流~~

Photos.framework

iOS 8新引入了全新的PhotoKit API, 用來替代AssetsLibrary框架, PhotoKit提供了直接訪問Moment數(shù)據(jù)的接口+ (PHFetchResult *)fetchMomentsWithOptions:(nullable PHFetchOptions *)options該函數(shù)直接返回按日期分類的圖片集合數(shù)據(jù), 且速度非???猜想是不是Apple在用戶拍攝圖片或者導(dǎo)入圖片時已mark日期信息并分類排序). 因此在iOS 8以上的系統(tǒng)可以直接采用PhotoKit框架來實現(xiàn)moment功能.

 


  1. PHFetchOptions *options  = [[PHFetchOptions alloc] init]; 
  2. options.sortDescriptors  = @[[NSSortDescriptor sortDescriptorWithKey:@"endDate" 
  3.                                                            ascending:ascending]]; 
  4. PHFetchResult  *momentRes = [PHAssetCollection fetchMomentsWithOptions:options]; 
  5. NSMutableArray *momArray  = [[NSMutableArray alloc] init]; 
  6. for (PHAssetCollection *collection in momentRes) 
  7.     NSDateComponents *components = [[NSCalendar currentCalendar] components:NSDayCalendarUnit   | 
  8.                                                                             NSMonthCalendarUnit | 
  9.                                                                             NSYearCalendarUnit 
  10.                                                                    fromDate:collection.endDate]; 
  11.     NSUInteger month = [components month]; 
  12.     NSUInteger year  = [components year]; 
  13.     NSUInteger day   = [components day]; 
  14.     MomentCollection *moment = [MomentCollection new]; 
  15.     moment.month = month; moment.year = year; moment.day = day; 
  16.     PHFetchOptions *option  = [[PHFetchOptions alloc] init]; 
  17.     option.predicate = [NSPredicate predicateWithFormat:@"mediaType = %d", PHAssetMediaTypeImage]; 
  18.     moment.assetObjs = [PHAsset fetchAssetsInAssetCollection:collection 
  19.                                                      options:option]; 
  20.     if ([moment.assetObjs count]) [momArray addObject:moment]; 

So, 我們可以對外統(tǒng)一moment接口, 在內(nèi)部(Gallery Model類)區(qū)分系統(tǒng)實現(xiàn): iOS 7系統(tǒng)采用AssetsLibrary并使用上文的優(yōu)化方案, iOS 8系統(tǒng)則直接調(diào)用Photos.framework的Moment接口.

但是這里面有個問題, AssetsLibrary的類型是ALAssetsGroup, 而PhotoKit的類型是PHFetchResult, 怎么在使用的時候統(tǒng)一呢? 難道還需要在外部調(diào)用的時候再區(qū)分一下系統(tǒng)么?

解決方法很簡單, 定義自己的數(shù)據(jù)類, 在數(shù)據(jù)結(jié)構(gòu)內(nèi)部再區(qū)分, 外部調(diào)用時使用的都是自己定義的數(shù)據(jù)類型:

e.g. 定義MomentCollection, 包括年月日信息, 對外的屬性assetObjs則在內(nèi)部區(qū)分系統(tǒng)并返回或設(shè)定相應(yīng)的類型:

 


  1. @interface MomentCollection : NSObject 
  2.  
  3. @property (nonatomic, readwrite) NSUInteger     month; 
  4. @property (nonatomic, readwrite) NSUInteger     year; 
  5. @property (nonatomic, readwrite) NSUInteger     day; 
  6. @property (nonatomic, strong) id assetObjs; 
  7.  
  8. @end 
  9.  
  10. @property (nonatomic, strong) NSMutableArray *items; 
  11. @property (nonatomic, strong) PHFetchResult  *assets; 
  12.  
  13. - (id)assetObjs 
  14.     return IS_IOS_8 ? self.assets : self.items; 
  15.  
  16. - (void)setAssetObjs:(id)assetObjs 
  17.     if (IS_IOS_8) 
  18.     { 
  19.         self.assets = (PHFetchResult *)assetObjs; 
  20.     } 
  21.     else 
  22.     { 
  23.         self.items  = (NSMutableArray *)assetObjs; 
  24.     } 

對于相冊或者某個具體的圖片也是類似的處理方法, 定義AlbumObj和PhotoObj數(shù)據(jù)類型. 這樣外界(調(diào)用者)就不用管數(shù)據(jù)類型了, 所有的邏輯都在內(nèi)部handle了...

另外, 對于對于其他功能, 比如相冊的枚舉, 相冊Poster圖片的獲取, 圖片URL的獲取, 某個相冊內(nèi)所有thumbnail的獲取等等都可以對外統(tǒng)一接口, 內(nèi)部再區(qū)分是使用PhotoKit還是AssetsLibrary.

 


  1. - (void)getMomentsWithBatchReturn:(BOOL)batch // batch for iOS 7 only 
  2.                         ascending:(BOOL)ascending 
  3.                        completion:(void (^)(BOOL done, id obj))completion; 
  4.                         
  5. - (void)getThumbnailForAssetObj:(id)asset 
  6.                        withSize:(CGSize)size  // size for iOS 8 only 
  7.                      completion:(void (^)(BOOL ret, UIImage *image))completion; 
  8.                       
  9. - (void)getURLForAssetObj:(id)asset 
  10.                 /*usingPH:(BOOL)usingPH*/ 
  11.                completion:(void (^)(BOOL ret, NSURL *URL))completion; 
  12.                 
  13. - (void)getAlbumsWithCompletion:(void (^)(BOOL ret, id obj))completion; 
  14.  
  15. - (void)getPosterImageForAlbumObj:(id)album 
  16.                        completion:(void (^)(BOOL ret, id obj))completion; 
  17.                         
  18. - (void)getPhotosWithGroup:(id)group 
  19.                 completion:(void (^)(BOOL ret, id obj))completion; 
  20.                  
  21. - (void)getImageForPhotoObj:(id)asset 
  22.                    withSize:(CGSize)size 
  23.                  completion:(void (^)(BOOL ret, UIImage *image))completion; 

完整的moment優(yōu)化方案和PhotoKit/AssetsLibrary集成接口實現(xiàn)代碼(RJPhotoGallery)已經(jīng)上傳到GitHub, 有興趣的童鞋可以參考一下. 程序內(nèi)封裝的ImageDataAPI是圖片加載的model類, 實現(xiàn)了Moment/Album功能, 有需要的可以直接copy過去使用.

P.S. 歡迎各路童鞋大神吐槽和交流~~

責任編輯:倪明 來源: RylanJIN投稿
相關(guān)推薦

2013-10-16 15:36:53

iOS優(yōu)化

2022-09-05 15:29:52

Windows 11微軟功能

2021-10-11 17:22:18

微信iOS騰訊

2015-05-28 10:20:34

js相冊翻頁

2013-09-17 10:17:39

Android布局

2022-11-11 17:23:49

Windows 11微軟資源管理器

2022-12-01 07:11:23

蘋果iOS

2009-06-17 08:47:00

Hibernate優(yōu)化

2010-05-07 11:00:25

Oracle多表查詢

2010-07-01 14:23:25

SQL Server查

2015-07-28 10:03:19

Java性能優(yōu)化

2015-09-22 09:46:46

iOS9漏洞

2009-07-14 10:13:38

MyEclipse優(yōu)化

2013-07-30 18:52:11

RiverbedWAN廣域網(wǎng)優(yōu)化

2010-03-22 15:58:08

Python模塊功能

2010-04-20 10:58:54

2010-05-14 17:56:16

SQL優(yōu)化索引

2023-04-25 08:01:23

JavaQuarkusKubernetes

2021-11-04 08:53:00

if-else代碼Java

2013-10-16 16:17:15

iOS開發(fā)優(yōu)化方案
點贊
收藏

51CTO技術(shù)棧公眾號