亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > iOS > 正文

iOS實現相冊和網絡圖片的存取

2020-07-26 02:50:22
字體:
來源:轉載
供稿:網友

保存 UIImage 到相冊

UIKit

UIKit 中一個古老的方法,Objective-C 的形式

復制代碼 代碼如下:

void UIImageWriteToSavedPhotosAlbum(UIImage *image, id completionTarget, SEL completionSelector, void *contextInfo);

保存完成后,會調用 completionTarget 的 completionSelector。如果 completionTarget 不為空,completionTarget 必須實現以下方法

復制代碼 代碼如下:

- (void)image:(UIImage *)image didFinishSavingWithError:(NSError *)error contextInfo:(void *)contextInfo;

Objective-C 的寫法

- (void)saveImage:(UIImage *)image {  UIImageWriteToSavedPhotosAlbum(image, self, @selector(image:didFinishSavingWithError:contextInfo:), nil); }- (void)image:(UIImage *)image didFinishSavingWithError:(NSError *)error contextInfo:(void *)contextInfo {  if (error) {    // Fail  } else {    // Success  }}

Swift 的寫法

func saveImage(_ image: UIImage) {  UIImageWriteToSavedPhotosAlbum(image, self, #selector(image(_:didFinishSavingWithError:contextInfo:)), nil)}func image(_ image: UIImage, didFinishSavingWithError error: NSError?, contextInfo: AnyObject) {  if error == nil {    // Success  } else {    // Fail  }}

Photos framework

iOS 8 開始,可以用 Photos framework。PHAssetChangeRequest 的類方法可以保存 UIImage

復制代碼 代碼如下:

class func creationRequestForAsset(from image: UIImage) -> Self

編輯相冊需要在 PHPhotoLibrary 的閉包中進行,有兩種方法

復制代碼 代碼如下:

func performChanges(_ changeBlock: @escaping () -> Void, completionHandler: ((Bool, Error?) -> Void)? = nil)

復制代碼 代碼如下:

func performChangesAndWait(_ changeBlock: @escaping () -> Void) throws

以上兩種方法,分別是異步和同步執行。一般用第一種異步執行的方法,不會阻塞主線程。

func saveImage(_ image: UIImage) {  PHPhotoLibrary.shared().performChanges({     PHAssetChangeRequest.creationRequestForAsset(from: image)  }, completionHandler: { (success, error) in    // NOT on main thread    if success {      // Success    } else if let error = error {      // Handle error    }  })}

編輯相冊的閉包 changeBlock 和完成的閉包 completionHandler,是在 serial queue 中執行,不在主線程。需要更新 UI 的話,要切換到主線程中執行。

保存圖片的 Data 到相冊

如果有圖片的數據(Data 或 NSData),可以用 Photos framework 的方法保存到相冊。從 iOS 9 開始,可以使用 PHAssetCreationRequest 的方法

復制代碼 代碼如下:

func addResource(with type: PHAssetResourceType, data: Data, options: PHAssetResourceCreationOptions?)

iOS 8 比較麻煩,需要把數據寫入臨時文件,用臨時文件的 URL 作為參數,調用 PHAssetChangeRequest 的類方法

復制代碼 代碼如下:

class func creationRequestForAssetFromImage(atFileURL fileURL: URL) -> Self?

以下是兼容 iOS 8 的寫法

func saveImageData(_ data: Data) {  if #available(iOS 9.0, *) {    PHPhotoLibrary.shared().performChanges({      PHAssetCreationRequest.forAsset().addResource(with: .photo, data: data, options: nil)    }, completionHandler: { (success, error) in      // NOT on main thread      if success {        // Success      } else if let error = error {        // Handle error      }    })  } else {    // Write image data to temp file    let tempPath = NSTemporaryDirectory().appending("TempImageToSaveToPhoto.image")    let tempUrl = URL(fileURLWithPath: tempPath)    try? data.write(to: tempUrl)        PHPhotoLibrary.shared().performChanges({      PHAssetChangeRequest.creationRequestForAssetFromImage(atFileURL: tempUrl)    }, completionHandler: { (success, error) in      // NOT on main thread      if success {        // Success      } else if let error = error {        // Handle error      }      // Remove temp file      try? FileManager.default.removeItem(at: tempUrl)    })  }}

SDWebImage 緩存 UIImage、Data

SDWebImage (目前版本 4.0.0) 有兩個方法可以使用。

SDWebImageManager 的方法

復制代碼 代碼如下:

- (void)saveImageToCache:(nullable UIImage *)image forURL:(nullable NSURL *)url;

SDImageCache 的方法

- (void)storeImage:(nullable UIImage *)image     imageData:(nullable NSData *)imageData      forKey:(nullable NSString *)key      toDisk:(BOOL)toDisk    completion:(nullable SDWebImageNoParamsBlock)completionBlock;

這個方法的 image、key 參數不能為空,否則直接執行 completionBlock 就返回。

從相冊獲取 UIImage、Data

UIImagePickerController 是常用的照片選取控制器。實現一個代理方法即可

復制代碼 代碼如下:

optional func imagePickerController(_ picker: UIImagePickerController, didFinishPickingMediaWithInfo info: [String : Any])

通過 info 字典,可以獲取 UIImage 等信息。這里用來查詢 info 字典的 key 有

UIImagePickerControllerOriginalImage // 原始 UIImageUIImagePickerControllerEditedImage // 編輯后的 UIImageUIImagePickerControllerReferenceURL // ALAsset 的 URL

通過 ALAsset 的 URL 可獲取 PHAsset。通過 PHImageManager 的方法可以獲得相冊圖片的 Data

復制代碼 代碼如下:

func requestImageData(for asset: PHAsset, options: PHImageRequestOptions?, resultHandler: @escaping (Data?, String?, UIImageOrientation, [AnyHashable : Any]?) -> Void) -> PHImageRequestID

以下是代碼示例

func imagePickerController(_ picker: UIImagePickerController, didFinishPickingMediaWithInfo info: [String : Any]) {  picker.dismiss(animated: true, completion: nil)    if let image = info[UIImagePickerControllerOriginalImage] as? UIImage {    // Get original image  }    if let url = info[UIImagePickerControllerReferenceURL] as? URL,    let asset = PHAsset.fetchAssets(withALAssetURLs: [url], options: nil).firstObject {    PHImageManager.default().requestImageData(for: asset, options: nil, resultHandler: { (imageData, _, _, _) in      if let data = imageData {        // Get image data      }    })  }}

從 SDWebImage 的緩存中獲取 UIImage、Data

SDWebImage 給 UIImageView 提供了方法,方便獲取、顯示網絡圖片。如果需要獲取下載的圖片(進行保存到相冊、上傳至服務器等操作),可以用以下方法

- (nullable id <SDWebImageOperation>)loadImageWithURL:(nullable NSURL *)url                       options:(SDWebImageOptions)options                       progress:(nullable SDWebImageDownloaderProgressBlock)progressBlock                      completed:(nullable SDInternalCompletionBlock)completedBlock;

Swift 的代碼示例

SDWebImageManager.shared().loadImage(with: url, options: SDWebImageOptions(rawValue: 0), progress: nil, completed: { [weak self] (cachedImage, imageData, error, _, _, _) in  guard self != nil else { return }    if let image = cachedImage {    // Get image  }  if let data = imageData {    // Get image data  }  if error != nil {    // Handle error  }})

這個方法有個問題,對于靜態圖片,可能獲取不到 Data。如果需要獲取圖片 Data 的話,不能直接這么寫。查看源碼可以找到原因。SDWebImageManager 的 loadImage: 方法會調用 SDImageCache 的 queryCacheOperationForKey: 方法

diskImageDataBySearchingAllPathsForKey: 方法用來獲取 Disk 中圖片的 Data。當圖片在 Memory 中,只有 GIF 圖片才會提供 Data,靜態圖的 Data 為空;當圖片在 Disk 中,都會提供 Data。如果能在外部直接調用 diskImageDataBySearchingAllPathsForKey: 方法就很簡單,但是不行,這是私有方法,只寫在 .m 文件里,對外不可見。

改源碼可以解決問題,將上圖第一個箭頭的 if 判斷去掉,總是調用 diskImageDataBySearchingAllPathsForKey: 方法。然而,改第三方庫源碼不好,可能會有想不到的糟糕后果。

一種方法是,根據 diskImageExistsWithKey: 方法,獲取 Disk 上的 Data。

判斷 Disk 的圖片是否存在,就是查找兩個路徑。同樣,拿到這兩個路徑的文件就可以獲得 Data。以下是 Swift 代碼示例

SDWebImageManager.shared().diskImageExists(for: imageUrl) { [weak self] (exist) in  // Always on main thread  guard self != nil else { return }  if exist {    // Find image data from disk    var data: NSData?    // Get cache key    let key = SDWebImageManager.shared().cacheKey(for: imageUrl)    // Get cache path    if let path = SDImageCache.shared().defaultCachePath(forKey: key) {      data = NSData(contentsOfFile: path)      if data == nil {        data = NSData(contentsOfFile: (path as NSString).deletingPathExtension)      }    }    if data != nil {      // Get image data    } else {      // Fail getting image data    }  } else {    // No disk image  }}

這個方法缺點在于,代碼復雜,可能會在 SDWebImage 版本升級后失效(例如,Disk 緩存路徑改變)。

推薦的方法是,將圖片緩存從 Memory 中移除,然后調用 SDWebImageManager 的 loadImage: 方法。

// Get cache keylet key = SDWebImageManager.shared().cacheKey(for: imageUrl)// Remove memory cacheSDImageCache.shared().removeImage(forKey: key, fromDisk: false, withCompletion: nil)// Load image and dataSDWebImageManager.shared().loadImage(with: imageUrl, options: SDWebImageOptions(rawValue: 0), progress: nil) { [weak self] (_, data, _, _, _, _) in  guard self != nil else { return }  if data != nil {    // Get image data  } else {    // Fail getting image data  }}

這樣寫比較簡潔。即使 SDWebImage 版本升級后改變 Disk 緩存路徑,依然有效。以上代碼執行之后,當前圖片又會存在 Memory 中。

遺留問題

將 JPG 圖片的 Data 保存至相冊,然后再取出的 Data 與保存的 Data 可能不一樣。requestImageData: 方法傳入 PHImageRequestOptions,PHImageRequestOptions 的 version 試了三種值(current、unadjusted、original)都不行。PNG、GIF 圖片還沒遇到這個問題??赡鼙4?JPG 圖片的過程會修改原始數據。如何使存取的數據一致?歡迎交流!

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国内精久久久久久久久久人| 久久久久久久久网站| 激情亚洲一区二区三区四区| 精品女同一区二区三区在线播放| 亚洲性猛交xxxxwww| 亚洲成av人片在线观看香蕉| 国产精品麻豆va在线播放| 亚洲视频自拍偷拍| 亚洲欧美国产精品专区久久| 亚洲韩国青草视频| 亚洲а∨天堂久久精品喷水| 日韩在线精品一区| 成人性生交xxxxx网站| 国产婷婷97碰碰久久人人蜜臀| 一区国产精品视频| 精品露脸国产偷人在视频| 亚洲缚视频在线观看| 国产69精品99久久久久久宅男| 国产精品久久久久久久久借妻| 日韩av在线网页| 亚洲欧美日韩网| 精品五月天久久| 97精品国产aⅴ7777| 久久精品亚洲94久久精品| 在线丨暗呦小u女国产精品| 美乳少妇欧美精品| 久久久久一本一区二区青青蜜月| 精品爽片免费看久久| 一区二区三区四区精品| 欧美亚洲国产另类| 久久综合久久美利坚合众国| 亚洲欧美日韩精品久久| 最近2019中文字幕在线高清| 欧美日韩国产在线播放| 一区二区欧美亚洲| 狠狠爱在线视频一区| 亚洲有声小说3d| 性色av一区二区三区在线观看| 国产成人精品在线播放| 国产精品va在线| 中文字幕日韩欧美在线| 国产精品人人做人人爽| 久久精品国产亚洲7777| 人九九综合九九宗合| 国产成人午夜视频网址| 国产男人精品视频| 精品日韩视频在线观看| 欧美午夜性色大片在线观看| 欧美成人精品激情在线观看| 揄拍成人国产精品视频| 一区二区三区动漫| 久久综合色影院| 国产精品嫩草视频| 日韩av电影在线播放| 久久久久久免费精品| 亚洲娇小xxxx欧美娇小| 亚洲天堂av综合网| 亚洲欧美中文日韩在线v日本| 57pao成人国产永久免费| 日韩av片免费在线观看| 国产69久久精品成人| 国产精品视频26uuu| 国产精品三级久久久久久电影| 亚洲v日韩v综合v精品v| 日韩欧美高清在线视频| 国产精品视频公开费视频| 欧美激情第三页| 国产mv久久久| 成人观看高清在线观看免费| 欧美日韩免费观看中文| 97久久久免费福利网址| 日韩精品999| 亚洲综合精品伊人久久| 中文字幕精品www乱入免费视频| 国产视频精品一区二区三区| 中文字幕在线日韩| 久久av在线播放| 亚洲高清在线观看| 国产精品久久久久久中文字| 北条麻妃一区二区三区中文字幕| 日韩免费看的电影电视剧大全| 欧美激情欧美激情| 色噜噜国产精品视频一区二区| 精品在线小视频| 中文字幕少妇一区二区三区| 亚洲精品网址在线观看| 欧美性在线观看| 欧美成人午夜剧场免费观看| 欧美丰满老妇厨房牲生活| 国产亚洲精品美女久久久久| 国内外成人免费激情在线视频网站| 亚洲最大福利视频网| 亚洲在线免费视频| 伊人伊成久久人综合网站| 亚洲缚视频在线观看| 亚洲精品女av网站| 国产一区二区三区久久精品| 久久久久久久久久久免费| 亚洲视频在线看| 久久久久免费精品国产| 亚洲欧美国产另类| 91在线免费视频| 久久999免费视频| 精品久久久国产| 欧美亚洲午夜视频在线观看| 91精品国产自产在线| 亚洲激情视频网站| 97久久伊人激情网| 欧洲美女免费图片一区| 日本精品久久久久久久| 久久精品成人欧美大片古装| 欧美高清一级大片| 高清日韩电视剧大全免费播放在线观看| 自拍视频国产精品| 亚洲天堂男人天堂女人天堂| 国产精品亚洲欧美导航| 亚洲性xxxx| 国内精品模特av私拍在线观看| 亚洲精品视频网上网址在线观看| 国产精品三级美女白浆呻吟| 91精品国产自产在线| 97涩涩爰在线观看亚洲| 精品香蕉在线观看视频一| 在线观看中文字幕亚洲| 中文字幕亚洲无线码在线一区| 97在线视频免费播放| 欧美色图在线视频| 色婷婷av一区二区三区久久| 欧美大尺度激情区在线播放| 久久国产精品影视| 亚洲成人激情图| 国产精品高潮呻吟久久av黑人| 久久精品免费电影| 久久激情五月丁香伊人| 国产精品999999| 97久久精品视频| 国产精品视频男人的天堂| 欧美精品福利视频| 亚洲一区第一页| 午夜免费日韩视频| 91精品久久久久久| 欧美激情伊人电影| 国产精品美女呻吟| 97久久精品人人澡人人爽缅北| 国产精品美女久久久久久免费| www欧美日韩| 青青久久av北条麻妃海外网| 欲色天天网综合久久| 国产精品视频一区二区高潮| 国产精品美女www| 亚洲一区二区日本| 日韩欧美一区二区三区久久| 秋霞午夜一区二区| 欧美精品999| 日韩欧美亚洲国产一区| 亚洲欧美精品一区| 精品久久久久久亚洲精品| 亚洲va国产va天堂va久久| 亚洲国产天堂久久综合网| 国产欧美精品一区二区三区介绍| 午夜精品视频在线| 亚洲视频在线视频| 亚洲美女自拍视频| 国产一区二区日韩精品欧美精品|