demo下載地址在最下方
一、AssetsLibrary
iOS提供的AssetsLibrary是用來讀取和操作本地相冊的,相當于app與系統相冊之間的橋梁,該框架提供了以下的幾個類:ALAssetsLibrary,ALAssetsGroup,ALAsset,ALAssetsFilter,ALAssetRepresentation
ALAssetsLibrary:本地資源庫對象
ALAssetsGroup:本地相冊分組對象,本地有多少個相薄就會有幾個ALAssetsGroup類型的對象。
ALAsset:從本地相冊獲取的圖片或者視頻的對象都是ALAsset類型的對象。
ALAssetsFilter:對本地所有資源的過濾篩選,可以選擇只獲取照片,只獲取視頻,或者獲取所有的資源。
ALAssetRepresentation:獲取視頻或者圖片的url,文件名,二進制數據,封面縮略圖等。
在這里先介紹一些比較常用的方法:
ALAssetsLibrary
1、獲得相冊的組別
- (void)enumerateGroupsWithTypes:(ALAssetsGroupType)types usingBlock:(ALAssetsLibraryGroupsEnumerationResultsBlock)enumerationBlock failureBlock:(ALAssetsLibraryAccessFailureBlock)failureBlock
在這個方法可以獲取分組的列表,包括名稱、封面圖片等,其中有以下幾個type:ALAssetsGroupLibrary,ALAssetsGroupAlbum,ALAssetsGroupEvent,ALAssetsGroupFaces,ALAssetsGroupSavedPhotos,ALAssetsGroupPhotoStream,ALAssetsGroupAll。
2、返回一個ALAsset對象
- (void)assetForURL:(NSURL *)assetURL resultBlock:(ALAssetsLibraryAssetForURLResultBlock)resultBlock failureBlock:(ALAssetsLibraryAccessFailureBlock)failureBlock
3、返回一個ALAssetsGroup對象
- (void)groupForURL:(NSURL *)groupURL resultBlock:(ALAssetsLibraryGroupResultBlock)resultBlock failureBlock:(ALAssetsLibraryAccessFailureBlock)failureBlock
ALAssetsGroup
ALAssetsGroup是獲取相冊組的類,可以通過這個類提供的方法來獲取相冊組的信息
1、獲得名稱、類型、ID、NSURL
- (id)valueForProperty:(NSString *)property;
property可以是:
ALAssetsGroupPropertyName:名稱
ALAssetsGroupPropertyType:類型
ALAssetsGroupPropertyPersistentID:ID
ALAssetsGroupPropertyURL :NSURL
2、獲得相冊分組的封面圖片
- (CGImageRef)posterImage;
3、過濾器
- (void)setAssetsFilter:(ALAssetsFilter *)filter
ALAssetsFilter一下這些方法是過濾類型:
+ (ALAssetsFilter *)allPhotos;
+ (ALAssetsFilter *)allVideos;
+ (ALAssetsFilter *)allAssets;
4、獲得相冊的組的數量
- (NSInteger)numberOfAssets;
5. 通過相冊組獲取里面的圖片
- (void)enumerateAssetsUsingBlock:(ALAssetsGroupEnumerationResultsBlock)enumerationBlock?
- (void)enumerateAssetsWithOptions:(NSEnumerationOptions)options usingBlock:(ALAssetsGroupEnumerationResultsBlock)enumerationBlock?
- (void)enumerateAssetsAtIndexes:(NSIndexSet *)indexSet options:(NSEnumerationOptions)options usingBlock:(ALAssetsGroupEnumerationResultsBlock)enumerationBlock
ALAssetRepresentation
ALAssetRepresentation對象封裝了一個給定對象ALAsset對象的描述
具體參考:
blog.csdn.net/meegomeego/article/details/21165145
ALAsset
- (id)valueForProperty:(NSString *)property;
有以下幾個類型:
ALAssetPropertyType 資源的類型(照片,視頻)
ALAssetPropertyLocation 資源地理位置(無位置信息返回null)
ALAssetPropertyDuation 播放時長(照片返回ALErorInvalidProperty)
ALAssetPropertyOrientation 方向(共有8個方向,參見:ALAssetOrientation)
ALAssetPropertyDate 拍攝時間(包含了年與日時分秒)
ALAssetPropertyRepresentations 描述(打印看了下,只有帶后綴的名稱)
ALAssetPropertyURLs(返回一個字典,鍵值分別是文件名和文件的url)
ALAssetPropertyAssetURL 文件的url
//得到縮略圖
- (CGImageRef)thumbnail;
以上是對AssetsLibrary的總結,下面是對photokit框架的總結
二、photokit
photokit是iOS8出的一個新的圖片選擇框架,用來取代上述AssetsLibrary框架,相比較AssetsLibrary來說photokit更加靈活,而在iOS9中,AssetsLibrary已經被棄用,也不支持livephoto,并且使用AssetsLibrary時,圖片的分辨率會變低,接下來就介紹一下photokit。
Photos中有以下幾個類:
PHAdjustmentData:當用戶編輯資源,照片與修改后的圖像或視頻數據保存在一個PHAdjustmentData對象中
PHAsset:代表照片庫中的一個資源實體,可以理解為一張照片,在打印的時候,可以清楚的看見里面包含了照片的時間、標題等信息
PHAssetChangeRequest:使用PHAssetChangeRequest對象來創建、刪除和修改相片庫里面的PHAsset對象
PHAssetCollectionChangeRequest:使用PHAssetCollectionChangeRequest對象來創建、刪除和修改相片庫里面的PHAssetCollection對象
PHAssetCreationRequest:使用PHAssetCreationRequest對象構造一個新的照片或視頻,并將其添加到照片庫
PHAssetResource:一個PHAssetResource對象表示在照片庫中的照片或視頻資源(也就是一個PHAsset對象)相關聯的基礎數據資源之一,使用PHAssetResourceManager類獲取它
PHAssetResourceManager:PHAssetResourceManager對象提供了訪問與照片資源相關的資源基礎數據存儲方法
PHChange:PHChange對象來通知的照片庫中的資源的一些變法,比如增加和刪除
PHCollection:PHAssetCollection對象表示一組照片或視頻資源。包括時刻和相冊中可見照片應用,以及特殊的集合,如共享照片流。
PHCollectionListChangeRequest:使用PHCollectionListChangeRequest對象在一個照片庫中創建、刪除或修改PHCollectionList對象。
PHContentEditingInput:PHContentEditingInput對象描述資產用于編輯元數據
PHContentEditingOutput:PHContentEditingOutput對象表示編輯照片資產的照片或視頻內容的結果
PHFetchOptions:獲取資源時的檢索參數,可以傳 nil,即使用系統默認值
PHFetchResult:通過PHFetchResult來獲取照片實體的有序列表
PHImageManager:PHImageManager對象提供用于加載與PHAsset對象相關聯的圖像或視頻數據的方法。使用這些方法來獲取全尺寸照片資產或縮略圖,或者檢索AV Foundation對象播放,導出和操縱的視頻資產。
PHLivePhoto:用來獲取LivePhoto實體
PHObject:照片等實體的抽象類
PHPhotoLibrary:PHPhotoLibrary對象表示用戶的照片庫整套資源和收藏的照片,包括存儲在本地設備上在iCloud的照片對象。
PhotosTypes:與Photos框架相關的類型的枚舉
使用photokit最直觀明了的地方就是一下這個關系:
PHCollectionList,它表示一組 PHCollection,它本身也是一個 PHCollection,因此 PHCollection 作為一個集合,可以包含其他集合,這使得 PhotoKit 的組成比 ALAssetLibrary 要復雜一些。另外與 ALAssetLibrary 相似,一個 PHAsset 可以同時屬于多個不同的 PHAssetCollection,最常見的例子就是剛剛拍攝的照片,至少同時屬于“最近添加”、“相機膠卷”以及“照片 - 精選”這三個 PHAssetCollection,他們的關系如下圖:
photokit的機制:
采用“獲取”的方式拉取資源,從 PHAssetCollection 中獲取到的可以是相冊也可以是資源,但無論是哪種內容,都統一使用 PHFetchResult 對象封裝起來,因此雖然 PHAssetCollection 獲取到的結果可能是多樣的,但通過 PHFetchResult 就可以使用統一的方法去處理這些內容(即遍歷 PHFetchResult)
// 列出所有相冊智能相冊
PHFetchResult *smartAlbums = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeSmartAlbum subtype:PHAssetCollectionSubtypeAlbumRegular options:nil];
// 這時 smartAlbums 中保存的應該是各個智能相冊對應的 PHAssetCollection
for (NSInteger i = 0; i < fetchResult.count; i++) {
// 獲取一個相冊(PHAssetCollection)
PHCollection *collection = fetchResult[i];
if ([collection isKindOfClass:[PHAssetCollection class]]) {
PHAssetCollection *assetCollection = (PHAssetCollection *)collection;
// 從一個相冊中獲取的PHFetchResult中包含的才是PHAsset
PHFetchResult *fetchResult = [PHAsset fetchAssetsInAssetCollection:assetCollection options:fetchOptions];
else {
NSAssert(NO, @"Fetch collection not PHCollection: %@", collection);
}
// 獲取所有資源的集合,并按資源的創建時間排序
PHFetchOptions *options = [[PHFetchOptions alloc] init];
options.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"creationDate" ascending:YES]];
PHFetchResult *assetsFetchResults = [PHAsset fetchAssetsWithOptions:options];
// 這時 assetsFetchResults 中包含的,應該就是各個資源(PHAsset)
for (NSInteger i = 0; i < fetchResult.count; i++) {
// 獲取一個資源(PHAsset)
PHAsset *asset = fetchResult[i];
}
到了這里已經成功取出了PHAsset,接下來做的操作就是獲取圖片
可以首先進行一下設置
PHImageRequestOptions *reques = [[PHImageRequestOptions alloc]init];
reques.synchronous = NO;//獲取圖片方式是否開啟異步
reques.resizeMode = PHImageRequestOptionsResizeModeFast;
reques.deliveryMode = PHImageRequestOptionsDeliveryModeFastFormat;
此時可以進行圖片獲取了:
- (PHImageRequestID)requestImageForAsset:(PHAsset *)asset
targetSize:(CGSize)targetSize
contentMode:(PHImageContentMode)contentMode
options:(nullable PHImageRequestOptions *)options
resultHandler:(void (^)(UIImage *__nullable result, NSDictionary *__nullable info))resultHandler;
resizeMode對應以下三個屬性:
1、PHImageRequestOptionsResizeModeNone,
2、PHImageRequestOptionsResizeModeFast, //根據傳入的size,迅速加載大小相匹配(略大于或略小于)的圖像
3、PHImageRequestOptionsResizeModeExact //精確的加載與傳入size相匹配的圖像
PHImageRequestOptionsDeliveryMode
PHImageRequestOptionsDeliveryMode是一個enum類型的,它囊括了非常復雜的操作,有三個值分別對應:
PHImageRequestOptionsDeliveryModeOpportunistic 當選用此項時,Photos會在你請求時給你提供一個或者多個結果,這就意味著resultHandler block可能會執行一次或多次,例如
Photos會先給你一個低分辨率的圖片讓你暫時顯示,然后加載出高質量的圖片后再次給你。如果PHImageManager已經pre-cache了圖片,那result handler便只會執行一次。另外,如果synchronous屬性為NO,此選項是不起作用的。
PHImageRequestOptionsDeliveryModeHighQualityFormat 當請求image時 不管請求需要多長時間完成,Photos只會提供高質量的image。另外,當synhronous屬性為YES 或者使用requestImageDataForAsset:options:resultHandler: 方法時,這個選項是默認且唯一的。
PHImageRequestOptionsDeliveryModeFastFormat 當請求image時,高質量的圖片或低質量的圖片,誰加載的快 便顯示誰。另外Photos也能通過檢查info字典(resultHandler中的參數)里的PHImageResultIsDegradedKey值來判定傳高質量或者低質量的圖片。
targetSize
需要獲取的圖像的尺寸,如果輸入的尺寸大于資源原圖的尺寸,只返回原圖
contentMode
圖像的裁剪方式,與UIView的contentMode參數相似,控制圖片是按比例縮放還是按比例填充的方式放在展示的控件中,但如果傳入的targetSize是PHImageManagerMaximumSize,contentMode不管傳什么值都是PHImageContentModeDefault;
這樣整個調取圖片的過程就已介紹完畢,最后可以得到一個包含資源對于圖像的UIImage和一個包含圖像信息的NSDictionary,而在整個過程中,這個block會被多次調用,通俗來說就是在開始的時候給你放一張低質量的圖上去,等它拿完之后給你呈現出高清圖,具體的效果可以查看微信聊天發圖片時大圖預覽,滑動時會先出現一張模糊的圖片,然后馬上呈現高清圖片。
以上就是我對AssetsLibrary以及photokit的大致總結,這段時間通過在項目中進行實際的使用操作后發現了兩者之間的區別:
首先就是二者的機制:
1、在 ALAssetLibrary 中獲取數據,無論是相冊,還是資源,本質上都是使用枚舉的方式,遍歷照片庫取得相應的數據,并且數據是從 ALAssetLibrary(照片庫) - ALAssetGroup(相冊)- ALAsset(資源)這一路徑逐層獲取,即使有直接從 ALAssetLibrary 這一層獲取 ALAsset 的接口,本質上也是枚舉 ALAssetLibrary 所得,并不是直接獲取,這樣的好處很明顯,就是非常符合實際應用中資源的顯示路徑:照片庫 - 相冊 - 圖片或視頻,但由于采用枚舉的方式獲取資源,效率低而且不靈活。而在 PhotoKit 中,則是采用“獲取”的方式拉取資源,通過上文的機制可以得知。
2、由于二者的機制以及層次關系問題,二者獲取到的資源也不一樣,所以iOS會在iOS9中開始廢棄ALAssetLibrary而采用photokit。ALAssetLibrary中獲取所有相冊資源使用的ALAssetsGroupAll屬性,而到了iOS9之后,對于個人收藏等一張圖片可以同時存在于多個相簿的資源來說,就會取不到這些公用相簿,而photokit的PHCollectionList的關系就可以取到這些資源,從而可以方便并且高效的達到我們需要的效果。
3、從獲取資源的方式來看,photokit比ALAssetLibrary更加高效并且更加靈活。
以上是我最近實踐發現的一些兩者的區別。如果還有其他的,希望大家可以在評論區留言給我我會再更新補充。接下來是我最近項目使用photokit框架所遇到的問題,希望有遇到該問題的朋友能夠和我一起探討,如果能有解決辦法,也希望可以給予我一些幫助。
在最新的photokit框架中上文已經提到獲取圖片的block會調用多次,而在我們進行項目測試時,發現在1000張圖片之內,本身是很快并且很高效的。
這里貼上demo地址https://github.com/Archerry/WMHAlbumSelectOC.git(之前會崩潰的已經替換掉了,現在換上最近重構優化的)
上面是OC版本,這個是swift版本github.com/Archerry/swift-photokit