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

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

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

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

Xcode

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

Swift

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

Q&A

解決済

1回答

1438閲覧

[swift/iOS]アニメーションGIFを切り替えたい

entaro12345

総合スコア75

iOS

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

Xcode

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

Swift

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

0グッド

0クリップ

投稿2020/05/14 04:04

編集2020/05/15 02:45

前提・実現したいこと

iOSアプリを作成してます。
一定間隔でAPI通信をしてその結果によってアニメーションGIFを切り替えたり、
画面遷移をさせたりしたいです。
実装してみたのですが、アニメーションGIFの切り替え部分で
メモリー関連のエラーが発生してしまいうまく動作しません。

お力を貸してください。

<追記>
TsukubaDepotさまにご連絡いただいた過去の問を参考にして、GIFファイルを圧縮して試してみましたが、
NGでしたので、別のライブラリを使用することも検討していこうと思います。

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

Message from debugger: Terminated due to memory issue

該当のソースコード

swift

1import UIKit 2import SwiftGifOrigin 3 4class ViewController: UIViewController { 5 6 var timer: Timer! 7 var dispatch = DispatchGroup() 8 var alert = UIAlertController() 9 10 var imgGif = UIImageView(image: UIImage.gif(name: "confirm_0")) 11 12 override func viewDidLoad() { 13 super.viewDidLoad() 14 15 // 画面サイズを取得 16 let iWidthScreen = self.view.frame.size.width 17 let iHeightScreen = self.view.frame.size.height 18 19 // アニメーションGIF 20 imgGif.image = imgConfRadioWave 21 imgGif.frame = CGRect(x: 15, y: 15, width: 150, height: 150) 22 self.view.addSubview(imgGif) 23 24 // 30秒ごとにAPI通信 25 timer = Timer.scheduledTimer(timeInterval: 30, target: self, selector: #selector(regularExec), userInfo: nil, repeats: true) 26 } 27 28 override func viewDidAppear(_ animated: Bool) { 29 dispatch = DispatchGroup() 30 dispatch.enter() 31 } 32 33 // API処理(30秒ごと) 34 @objc func regularExec() { 35 36 let sParam = "count=1" 37 let sData = sParam.data(using: String.Encoding.utf8) 38 39 let urlString = "https://XXX.XX.XX" 40 41 var sRes = "" 42 var iCnt = 0 43 if let url = URL(string: urlString) { 44 45 let req = NSMutableURLRequest(url: url) 46 req.httpMethod = "POST" 47 req.httpBody = sData 48 let task = URLSession.shared.dataTask(with: req as URLRequest) { data, resp, err in 49 50 do { 51 let json:NSArray = try! JSONSerialization.jsonObject(with: data!, options: JSONSerialization.ReadingOptions.mutableContainers) as! NSArray 52 let aData = json[0] as! Dictionary<String, AnyObject> 53 54 sRes = aData["result"] as! String 55 iCnt = Int(aData["count"] as! String)! 56 self.dispatch.leave() 57 } 58 } 59 task.resume() 60 61 // task終了後処理 62 self.dispatch.notify(queue: .main) { 63 if (sRes == "0") { 64 // 回数に応じてGIFアニメを切り替え 65 switch (iCnt) { 66 case 1: 67 self.imgConfRadioWave = UIImage.gif(name: "confirm_1") 68 self.imgViewConfRadioWave.image = self.imgConfRadioWave 69 break; 70 case 2: 71 self.imgConfRadioWave = UIImage.gif(name: "confirm_2") 72 self.imgViewConfRadioWave.image = self.imgConfRadioWave 73 break; 74 case 3: 75 self.imgConfRadioWave = UIImage.gif(name: "confirm_3") 76 self.imgViewConfRadioWave.image = self.imgConfRadioWave 77 break; 78 default: 79 break; 80 } 81 self.dispatch = DispatchGroup() 82 self.dispatch.enter() 83 } else { 84 self.timer.invalidate() 85 if (sRes == "1") { 86 // 画面遷移 87 let nextView = NextViewController() 88 self.present(nextView, animated: true, completion: nil) 89 } 90 } 91 } 92 } 93 } 94} 95※エラー処理等は割愛しています

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

MacOS HighSieera
Xcode 10.1
Swift 3.1

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

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

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

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

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

TsukubaDepot

2020/05/14 06:41

アニメーション gif の表示にはライブラリか何か入れてらっしゃるのでしょうか。 その辺りの情報もご質問に追記していただければと思います。
entaro12345

2020/05/14 06:43

ご連絡ありがとうございます。 importが一つ抜けてしまってましたので、追記します。 ちなみに、「SwiftGifOrigin」を使用しています。
TsukubaDepot

2020/05/14 06:51

そうすると、次の過去問は参考になりますでしょうか。 https://teratail.com/questions/200957 ちなみに、今回の問題とは直接関係ありませんが、ご提示いただいたコードだと、サーバのレスポンスが30秒以上かかった場合にはデッドロックになりそうな気がします。
entaro12345

2020/05/14 08:45

ありがとうございます。 GIFファイルを圧縮して試してみます。 仰る通り、確かに30秒以上かかった場合の処理も入れるようにします。 アドバイスありがとうございます。
TsukubaDepot

2020/05/15 03:18

https://github.com/swiftgif/SwiftGif/issues/6 上記の報告の最後、2018年にmemory leak issue として報告されていますが、それ以降対策が取られていないように見えます。 1. When I switching view controller, Gif doesn't release memory. How can I deal with this issue? How can I manage OR solve this issue? 2. When I try to load some other GIF with same viewcontroller on same imageview (On different button action), It's going to very horrible memory leak. It's going to 650MB. Please Help. 上記の1., 2. ともに今回のパターンと同一のケースと考えられますが、いかがでしょうか。 1. のview controllerを切り替えるときにインスタンスを解放しない,というのは // 画面遷移 let nextView = NextViewController() self.present(nextView, animated: true, completion: nil) ただし、この場合はpresentでモーダル表示しているので、メモリを解放しなくても当然かもしれません。 2. は同じクラスに別のインスタンスを代入するときに古いインスタンスが解放されない) self.imgConfRadioWave = UIImage.gif(name: "confirm_1") self.imgViewConfRadioWave.image = self.imgConfRadioWave
entaro12345

2020/05/15 04:11

ありがとうございます。 仰る通りで、同一のケースにような気がします。 同じViewに対してGIFを代入するとメモリリークが発生してしまうのですね・・。 別のライブラリを検討します。
TsukubaDepot

2020/05/15 04:26

ライブラリに直接手を入れるのも手かもしれませんが、いま別のライブラリをお探しのようなので、それが使えるのなら、保守されている別のライブラリを使うのがいいかもしれませんね。 別のライブラリを使うことで解決しそうであれば、こちらは自己解決ということで(メモリリークがあった、というのも有益な情報ですので)お願いできますでしょうか。
entaro12345

2020/05/15 05:35

かしこまりました。 解決しましたら、自己解決として投稿します。
entaro12345

2020/05/15 07:33

SwiftyGifを使用して、動作することができました。 色々と情報ありがとうございました。
guest

回答1

0

自己解決

最初、SwiftGifOriginを使用して実装していましたが、GIF切り替え時に、
「メモリリーク」のエラーが発生してしまい、GIFファイルの圧縮等も試みましたが、NGであったため、
使用するライブラリを変更してSwiftyGifを使用したところメモリリークなどの問題が発生せずに、
動作することができました。

swift

1import UIKit 2import SwiftyGif 3 4class ViewController: UIViewController { 5 6 var timer: Timer! 7 var dispatch = DispatchGroup() 8 var alert = UIAlertController() 9 10 var imgConf = UIImage(gifName: "confirm_0") 11 var imgViewConf = UIImageView(gifImage: UIImage(gifName: "confirm_0"), manager: SwiftyGifManager(memoryLimit:20)) 12 13 override func viewDidLoad() { 14 super.viewDidLoad() 15 16 // 画面サイズを取得 17 let iWidthScreen = self.view.frame.size.width 18 let iHeightScreen = self.view.frame.size.height 19 20 // アニメーションGIF 21 imgViewConf.image = imgConf 22 imgViewConf.frame = CGRect(x: 15, y: 15, width: 150, height: 150) 23 self.view.addSubview(imgViewConf) 24 25 // 30秒ごとにAPI通信 26 timer = Timer.scheduledTimer(timeInterval: 30, target: self, selector: #selector(regularExec), userInfo: nil, repeats: true) 27 } 28 29 override func viewDidAppear(_ animated: Bool) { 30 dispatch = DispatchGroup() 31 dispatch.enter() 32 } 33 34 // API処理(30秒ごと) 35 @objc func regularExec() { 36 37 let sParam = "count=1" 38 let sData = sParam.data(using: String.Encoding.utf8) 39 40 let urlString = "https://XXX.XX.XX" 41 42 var sRes = "" 43 var iCnt = 0 44 if let url = URL(string: urlString) { 45 46 let req = NSMutableURLRequest(url: url) 47 req.httpMethod = "POST" 48 req.httpBody = sData 49 let task = URLSession.shared.dataTask(with: req as URLRequest) { data, resp, err in 50 51 do { 52 let json:NSArray = try! JSONSerialization.jsonObject(with: data!, options: JSONSerialization.ReadingOptions.mutableContainers) as! NSArray 53 let aData = json[0] as! Dictionary<String, AnyObject> 54 55 sRes = aData["result"] as! String 56 iCnt = Int(aData["count"] as! String)! 57 self.dispatch.leave() 58 } 59 } 60 task.resume() 61 62 // task終了後処理 63 self.dispatch.notify(queue: .main) { 64 if (sRes == "0") { 65 // 回数に応じてGIFアニメを切り替え 66 switch (iCnt) { 67 case 1: 68 self.imgConf = UIImage(gifName: "confirm_1") 69 self.imgViewConf.setGifImage(self.imgConf) 70 break; 71 case 2: 72 self.imgConf = UIImage(gifName: "confirm_2") 73 self.imgViewConf.setGifImage(self.imgConf) 74 break; 75 case 3: 76 self.imgConf = UIImage(gifName: "confirm_3") 77 self.imgViewConf.setGifImage(self.imgConf) 78 break; 79 default: 80 break; 81 } 82 self.dispatch = DispatchGroup() 83 self.dispatch.enter() 84 } else { 85 self.timer.invalidate() 86 if (sRes == "1") { 87 // 画面遷移 88 let nextView = NextViewController() 89 self.present(nextView, animated: true, completion: nil) 90 } 91 } 92 } 93 } 94 } 95} 96※エラー処理等は割愛しています

投稿2020/05/15 07:32

entaro12345

総合スコア75

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問