質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
Realm

RealmとはSQLiteやCore Dataに代わるモバイルデータベースです。iOSとAndroidの両方でサポートされています。

iOS

iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

Swift

Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

Q&A

解決済

1回答

740閲覧

マルチスレッドでのCollectionview読み込みがうまくいかない

退会済みユーザー

退会済みユーザー

総合スコア0

Realm

RealmとはSQLiteやCore Dataに代わるモバイルデータベースです。iOSとAndroidの両方でサポートされています。

iOS

iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

Swift

Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

0グッド

0クリップ

投稿2019/02/10 19:13

Realmに保存している写真のlocalIdentifierから写真をCollectionViewに表示できるようにしたのですが

スクロールの速度を上げるとカクついてしまうため、マルチスレッドで読み込みを早くしようとしたら、カクつきはしなくなりましたが
その代わり読み込まれる画像がおかしくなってしまいます

もしかしたら他スレッドでの処理が終わっていないのにcellへの画像設定が呼び出されているから、おかしいのかなと思いましたが、解決策がわかりません

解決方法をご存知の方いましたらご教授願えませんでしょうか

よろしくお願いいたします

CollectionViewDelegate

swift

1func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell { 2 let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "test", for: indexPath) as! collectionViewCell 3 _getPhotoImage(index: indexPath.row) 4 cell.photoView.image = self.image 5 return cell 6}

swift

1 2private var image: UIImage? 3 4private func _getPhotoImage(index: Int){ 5 let realm = try! Realm() 6 //Realmはバラバラに保存されるため、順番を保持するための配列 7 var photoCollection = UserDefaults.standard.array(forKey: "PHOTO_COLLECTION") 8 9 if let photoObject = realm.object(ofType: Photo.self, forPrimaryKey: (photoCollection?[index])){ 10 let _photo = ThreadSafeReference(to: photoObject) 11 DispatchQueue.global(qos: .userInteractive).async { 12 let realm = try! Realm() 13 guard let photo = realm.resolve(_photo) else { 14 return 15 } 16 let localIdentifier: [String] = [photo.photoURL!] 17 let photoAsset = PHAsset.fetchAssets(withLocalIdentifiers: localIdentifier, options: nil) 18  //PHAssetをUIImageに変換 19 self.image = getUIImage(asset:photoAsset.object(at: 0)) 20 } 21 } 22}

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

マルチスレッドで動いているなら完了する順番がわかりませんから self.imageに一旦保存するようなことはしてはダメです。
画像が読み込み終わったら、DispatchQueue.mainでメインスレッドに戻してセルが画面内にあれば(cellForItemが返ってくれば)そのセルに画像を設定します。

Swift

1let image = getUIImage(asset:photoAsset.object(at: 0)) 2DispatchQueue.main.async{ 3 // cellは適当に自分のセルの型に変換してください 4 let cell = self.collectionView.cellForItem(at:index) 5 cell?.photoView.image = image 6}

cellForItemAt の cell.photoView.image = self.image は不要です。
あと、現状だとセルが表示されるたびに読み込みが発生するので、多少キャッシュするような処理が必要になると思います。

投稿2019/02/11 02:22

編集2019/02/11 02:25
toki_td

総合スコア2850

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

退会済みユーザー

退会済みユーザー

2019/02/11 03:09

回答ありがとうございます 試してみたところ動きがよくなりました! 最後に一つだけお聞きしたいのですが、カク付きはなくなったのですが、読み込みが間に合っていない時があります、それはおっしゃっていた表示されるたびに読み込みが発生しているから、処理が間に合っていないという考え方で大丈夫でしょうか?
toki_td

2019/02/11 07:41

間に合っていない、というのは空の領域が表示されてしばらくしたら画像が出る、ということですか? それならその通りです。素早くスクロールした場合などは大量の読み込みが発生するので、DispatchQueueに要求がどんどん溜まっていき、(すでにスクロールが行き過ぎて不要になった画像も含めて)順次読み込まれていくのでどうしても遅延が発生します。 対応方法はいくつかあります。 ・DispatchQueue.globalの実行ブロックでそのインデックスが表示範囲内か確認し、不要なら画像を読み込まない ・スクロールの範囲から必要な前後をある程度先読みする ・一度読んだ画像はある程度キャッシュする 完全な対処は難しいですが、これらを組み合わせれば必要十分なレベルになると思います。
退会済みユーザー

退会済みユーザー

2019/02/12 00:59

勉強になりました! ありがとうございました!!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問