如何使用 DispatchGroup 查找照片并将其存储在 UIImage 数组中
How can I use DispatchGroup to look up photos and store in UIImage array
我看过很多使用调度组的例子,但我似乎无法让它工作。
我正在为我的项目使用 Google 地图,并在附近的商家处放置了标记。单击标记时,我想显示另一个包含该地点图像的视图控制器。
func mapView(_ mapView: GMSMapView, didTap marker: GMSMarker) -> Bool {
self.loadPhotosForSalonByPlaceId(placeID: poiItem.placeId)
photoDispatchGroup.wait()
let salonInfoViewController = self.addPullUpController() //show view controller
salonInfoViewController.ImageCarouselView.images = self.salonImages
salonInfoViewController.salonName.text = poiItem.name
salonInfoViewController.salonAddress.text = poiItem.address
salonInfoViewController.openAppointmentSlotArray = self.openAppointmentSlots
self.isSalonInfoViewPresented = true
return isSalonInfoViewPresented
}
这是我的 loadPhotosForSalonByPlaceId 的样子:
func loadPhotosForSalonByPlaceId(placeID: String){
var countToLimitToFiveImages = 0
GMSPlacesClient.shared().lookUpPhotos(forPlaceID: placeID) { (photos, error) -> Void in
if let error = error {
// TODO: handle the error.
print("Error: \(error.localizedDescription)")
} else {
for photoMetadata in (photos?.results)! {
if countToLimitToFiveImages == 5 {
break;
}
self.photoDispatchGroup.enter()
self.loadImageForMetadata(photoMetadata: photoMetadata)
self.photoDispatchGroup.leave()
countToLimitToFiveImages += 1
}
}
}
}
我是否错误地使用了进入和离开?或者我应该在 lookUpPhotos 完成后通知主线程继续吗?因为现在,当我想显示视图控制器时,UIImages 数组是空的。
提前致谢!
下面的代码是我在 loadPhotosForSalonByPlaceId 函数中调用的。它将 PhotoMetaData 转换为 UIImage,我将其附加到我的数组。据我了解,查找照片和 loadplacephoto 是异步调用。在完成这两项任务后,我如何使用 DispatchGroup 来显示我的视图控制器。
func loadImageForMetadata(photoMetadata: GMSPlacePhotoMetadata) {
var image = UIImage()
GMSPlacesClient.shared().loadPlacePhoto(photoMetadata, callback: {
(photo, error) -> Void in
if let error = error {
// TODO: handle the error.
print("Error: \(error.localizedDescription)")
} else {
image = photo!
self.salonImages.append(image)
}
})
}
你目前对DispatchGroup的使用都是错误的。您应该在异步进程开始之前调用 enter
并在异步进程完成时调用 leave
。永远不要在主队列上调用 wait
。你会想要使用 notify
。同样,在这种情况下不要使用调度组。
您在这里有两个级别的异步调用。第一个在 loadImageForMetadata
中。这需要重构以提供完成处理程序。第二个在 loadPhotosForSalonByPlaceId
中,也需要重构以提供完成处理程序。您还需要重做如何使用调度组。
这是你的重构 loadImageForMetadata
:
func loadImageForMetadata(photoMetadata: GMSPlacePhotoMetadata, completion: (Bool) -> Void) {
var image = UIImage()
GMSPlacesClient.shared().loadPlacePhoto(photoMetadata, callback: {
(photo, error) -> Void in
if let error = error {
// TODO: handle the error.
print("Error: \(error.localizedDescription)")
completion(false)
} else {
image = photo!
self.salonImages.append(image)
completion(true)
}
})
}
这是重构后的 loadPhotosForSalonByPlaceId
,带有完成处理程序和正确使用 DispatchGroup:
func loadPhotosForSalonByPlaceId(placeID: String, completion: (Bool) -> Void) {
var countToLimitToFiveImages = 0
GMSPlacesClient.shared().lookUpPhotos(forPlaceID: placeID) { (photos, error) -> Void in
if let error = error {
// TODO: handle the error.
print("Error: \(error.localizedDescription)")
completion(false)
} else {
let group = DispatchGroup()
for photoMetadata in (photos?.results)! {
if countToLimitToFiveImages == 5 {
break;
}
group.enter()
self.loadImageForMetadata(photoMetadata: photoMetadata) { (success) in
if success {
countToLimitToFiveImages += 1
}
group.leave()
}
}
group.notify(queue: DispatchQueue.global()) {
completion(true)
}
}
}
}
然后更新你的用法:
func mapView(_ mapView: GMSMapView, didTap marker: GMSMarker) -> Bool {
self.loadPhotosForSalonByPlaceId(placeID: poiItem.placeId) { (success) in
if success {
DispatchQueue.main.async {
let salonInfoViewController = self.addPullUpController() //show view controller
salonInfoViewController.ImageCarouselView.images = self.salonImages
salonInfoViewController.salonName.text = poiItem.name
salonInfoViewController.salonAddress.text = poiItem.address
salonInfoViewController.openAppointmentSlotArray = self.openAppointmentSlots
}
}
}
self.isSalonInfoViewPresented = true
return isSalonInfoViewPresented
}
我看过很多使用调度组的例子,但我似乎无法让它工作。
我正在为我的项目使用 Google 地图,并在附近的商家处放置了标记。单击标记时,我想显示另一个包含该地点图像的视图控制器。
func mapView(_ mapView: GMSMapView, didTap marker: GMSMarker) -> Bool {
self.loadPhotosForSalonByPlaceId(placeID: poiItem.placeId)
photoDispatchGroup.wait()
let salonInfoViewController = self.addPullUpController() //show view controller
salonInfoViewController.ImageCarouselView.images = self.salonImages
salonInfoViewController.salonName.text = poiItem.name
salonInfoViewController.salonAddress.text = poiItem.address
salonInfoViewController.openAppointmentSlotArray = self.openAppointmentSlots
self.isSalonInfoViewPresented = true
return isSalonInfoViewPresented
}
这是我的 loadPhotosForSalonByPlaceId 的样子:
func loadPhotosForSalonByPlaceId(placeID: String){
var countToLimitToFiveImages = 0
GMSPlacesClient.shared().lookUpPhotos(forPlaceID: placeID) { (photos, error) -> Void in
if let error = error {
// TODO: handle the error.
print("Error: \(error.localizedDescription)")
} else {
for photoMetadata in (photos?.results)! {
if countToLimitToFiveImages == 5 {
break;
}
self.photoDispatchGroup.enter()
self.loadImageForMetadata(photoMetadata: photoMetadata)
self.photoDispatchGroup.leave()
countToLimitToFiveImages += 1
}
}
}
}
我是否错误地使用了进入和离开?或者我应该在 lookUpPhotos 完成后通知主线程继续吗?因为现在,当我想显示视图控制器时,UIImages 数组是空的。
提前致谢!
下面的代码是我在 loadPhotosForSalonByPlaceId 函数中调用的。它将 PhotoMetaData 转换为 UIImage,我将其附加到我的数组。据我了解,查找照片和 loadplacephoto 是异步调用。在完成这两项任务后,我如何使用 DispatchGroup 来显示我的视图控制器。
func loadImageForMetadata(photoMetadata: GMSPlacePhotoMetadata) {
var image = UIImage()
GMSPlacesClient.shared().loadPlacePhoto(photoMetadata, callback: {
(photo, error) -> Void in
if let error = error {
// TODO: handle the error.
print("Error: \(error.localizedDescription)")
} else {
image = photo!
self.salonImages.append(image)
}
})
}
你目前对DispatchGroup的使用都是错误的。您应该在异步进程开始之前调用 enter
并在异步进程完成时调用 leave
。永远不要在主队列上调用 wait
。你会想要使用 notify
。同样,在这种情况下不要使用调度组。
您在这里有两个级别的异步调用。第一个在 loadImageForMetadata
中。这需要重构以提供完成处理程序。第二个在 loadPhotosForSalonByPlaceId
中,也需要重构以提供完成处理程序。您还需要重做如何使用调度组。
这是你的重构 loadImageForMetadata
:
func loadImageForMetadata(photoMetadata: GMSPlacePhotoMetadata, completion: (Bool) -> Void) {
var image = UIImage()
GMSPlacesClient.shared().loadPlacePhoto(photoMetadata, callback: {
(photo, error) -> Void in
if let error = error {
// TODO: handle the error.
print("Error: \(error.localizedDescription)")
completion(false)
} else {
image = photo!
self.salonImages.append(image)
completion(true)
}
})
}
这是重构后的 loadPhotosForSalonByPlaceId
,带有完成处理程序和正确使用 DispatchGroup:
func loadPhotosForSalonByPlaceId(placeID: String, completion: (Bool) -> Void) {
var countToLimitToFiveImages = 0
GMSPlacesClient.shared().lookUpPhotos(forPlaceID: placeID) { (photos, error) -> Void in
if let error = error {
// TODO: handle the error.
print("Error: \(error.localizedDescription)")
completion(false)
} else {
let group = DispatchGroup()
for photoMetadata in (photos?.results)! {
if countToLimitToFiveImages == 5 {
break;
}
group.enter()
self.loadImageForMetadata(photoMetadata: photoMetadata) { (success) in
if success {
countToLimitToFiveImages += 1
}
group.leave()
}
}
group.notify(queue: DispatchQueue.global()) {
completion(true)
}
}
}
}
然后更新你的用法:
func mapView(_ mapView: GMSMapView, didTap marker: GMSMarker) -> Bool {
self.loadPhotosForSalonByPlaceId(placeID: poiItem.placeId) { (success) in
if success {
DispatchQueue.main.async {
let salonInfoViewController = self.addPullUpController() //show view controller
salonInfoViewController.ImageCarouselView.images = self.salonImages
salonInfoViewController.salonName.text = poiItem.name
salonInfoViewController.salonAddress.text = poiItem.address
salonInfoViewController.openAppointmentSlotArray = self.openAppointmentSlots
}
}
}
self.isSalonInfoViewPresented = true
return isSalonInfoViewPresented
}