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

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

ただいまの
回答率

90.34%

  • Swift

    7644questions

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

  • Xcode

    4316questions

    Xcodeはソフトウェア開発のための、Appleの統合開発環境です。Mac OSXに付随するかたちで配布されています。

  • iOS

    4166questions

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

  • MacOS(OSX)

    2030questions

    MacOSとは、Appleの開発していたGUI(グラフィカルユーザーインターフェース)を採用したオペレーションシステム(OS)です。Macintoshと共に、市場に出てGUIの普及に大きく貢献しました。

Swiftにて、非同期なブロック(?)の終了を待ちたいです。

解決済

回答 1

投稿

  • 評価
  • クリップ 0
  • VIEW 2,382

SST8897

score 33

前提・実現したいこと

Swift3、iOSアプリ開発にて、非同期なブロック...例えばcompletionHandlerとか名前ついてるアレの中で代入したり得たりした値を、completionHandlerの終了を待ってから、その外側(?)のreturnで返したいです。

発生している問題・エラーメッセージ

エラーメッセージ:(Class PLBuildVersion is implemented in both <パス> <パス> というエラーは出てるが多分これには関係なさそう)

下のソースコードを実行すると、止まったまま動かなくなってしまいます。
completionHandlerが非同期で実行されるので、
completionHandlerの中でexifの情報を得るより先に、
一番下のreturn exifが値を返してしまうので、いつもexifはnilになってしまっているようです。

該当のソースコード

//画像のExif情報をこれで返したいです。
private func getExif(asset : PHAsset)->NSDictionary? {
        let editOptions = PHContentEditingInputRequestOptions()
        editOptions.isNetworkAccessAllowed = true

        let semaphore = DispatchSemaphore(value : 0)

        var exif:NSDictionary?

        let queue = DispatchQueue.global(qos: .default)

        //completionHandlerは別スレッドで行われる(非同期)だそう
        //なので普通にreturn exif するといつもnilが入ってる!
     //だからcompletionHandlerが終了するのを待ってreturn exifしたい!
        //ディスパッチセマフォを使って待とうと思ったけど、これを実行すると止まってしまう
        asset.requestContentEditingInput(with: editOptions, completionHandler: { (contentEditingInput, _) -> Void in
            let url = contentEditingInput?.fullSizeImageURL
            let inputImage:CIImage = CoreImage.CIImage(contentsOf: url!)!

            self.meta = inputImage.properties as NSDictionary?

            //Exif情報がexifに代入される(が、completionHandlerを待たないと、ここでexifに何か入るより先にreturn exifされるのでnilが返される)
            exif = self.meta!["{Exif}"] as? NSDictionary  

            //セマフォの値をインクリメント
            queue.async {
                semaphore.signal()
            }
        })

        //本当はこのように上のcompletionHandlerが終わるのを待ってreturn exif したい
        semaphore.wait()
        //completionHandlerの中でexifに代入されるのを待たないとここでいつもnilが返されてしまう。
        //なのでexifに中身が入るように、completionHandlerが終わるのを待ちたい。
        return exif
    }

試したこと

参考にしたのは、
・ディスパッチセマフォの使い方
・非同期の、Voidが戻り値のブロックから値を返せるか?
(→回答は、非同期のやり方に従え、無理に逆らわない方がいい…的な感じ?)
・Wait for async task to finish completion block before returning in app delegate
(ここで、ディスパッチセマフォで待つといいよ的な回答を見て、そうやって待てばいいんか!ってこのやり方を試しています)

補足情報(言語/FW/ツール等のバージョンなど)

Swift3.0.2、Xcode8(version8.2.1)、iOS10.2で試しました。
使ってるOSはMacOS10.12.3です。

非同期のやり方に従え…変なやり方すんな...的な感じなら教えてください…。
自分の知識不足のせいで変なことやろうとしてる気もするので…。

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

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

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

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

checkベストアンサー

+1

asset.requestContentEditingInput
の処理はgetExifの実行がreturnし、その関数を呼び出した処理が全て終わってから始まるので、セマフォでブロックしてしまってはデッドロック状態になると思います。

requestContentEditingInputの結果をgetExifの呼び出し側に情報として返したいのであれば、

private func getExif(asset : PHAsset)->NSDictionary?


の定義を

private func getExif(asset :PHAsset, exifCompletionHandler: @escaping (dictionary: NSDictionary?) -> Void) -> Void


に変更し、

exif = self.meta!["{Exif}"] as? NSDictionary
exifCompletionHandler(exif)


としたらどうでしょうか。

getExif()はすぐさまreturnするので、それ以降の処理をgetExif()の引数exifCompletionHandlerにクロージャとして渡すことになります。

ソースコードが書かれた順番で処理されるように、プログラムの見通しをよくしたい場合は、
RxSwiftやPromiseKitなどのフレームワークを使えばある程度可能ですが、
学習コストが要求されるので、まずはcompletionHandlerなどの基本的なやり方をマスターしてはどうでしょうか。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2017/02/11 14:18 編集

    なるほど!
    返信ではMarkdown記法が使えない(多分?)ようなので見づらくてすみませんが、
    private func getExif(asset :PHAsset, exifCompletionHandler: @escaping (_ dictionary: NSDictionary?) -> Void) -> Void{
    let editOptions = PHContentEditingInputRequestOptions()
    editOptions.isNetworkAccessAllowed = true

    var exif:NSDictionary?

    asset.requestContentEditingInput(with: editOptions, completionHandler: { (contentEditingInput, _) -> Void in
    let url = contentEditingInput?.fullSizeImageURL
    let inputImage:CIImage = CoreImage.CIImage(contentsOf: url!)!

    self.meta = inputImage.properties as NSDictionary?

    exif = self.meta!["{Exif}"] as? NSDictionary
    exifCompletionHandler(exif)
    })
    }

    として、
    呼び出し側では例として
    self.getExif(asset: self.photoAssets.first!, exifCompletionHandler: {(exif) -> Void in
    print(exif!["DateTimeDigitized"] as! String)
    })
    などとすれば良いのですね!(試したらちゃんと実際動いてくれました)

    >「asset.requestContentEditingInput
    の処理はgetExifの実行がreturnし、その関数を呼び出した処理が全て終わってから始まる」…「デッドロック状態になる」
    のところとか全然わからなかったので知れて嬉しいです、回答ありがとうございました!

    キャンセル

同じタグがついた質問を見る

  • Swift

    7644questions

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

  • Xcode

    4316questions

    Xcodeはソフトウェア開発のための、Appleの統合開発環境です。Mac OSXに付随するかたちで配布されています。

  • iOS

    4166questions

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

  • MacOS(OSX)

    2030questions

    MacOSとは、Appleの開発していたGUI(グラフィカルユーザーインターフェース)を採用したオペレーションシステム(OS)です。Macintoshと共に、市場に出てGUIの普及に大きく貢献しました。