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

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

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

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

Swift

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

Q&A

解決済

3回答

1079閲覧

通信に失敗した時にアラートを表示する方法

Haruto513

総合スコア52

Xcode

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

Swift

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

0グッド

0クリップ

投稿2019/07/08 05:26

編集2019/07/11 08:46

swift初心者です。
http通信を行ってデータをダウンロードする処理を書いているのですが、ダウンロードに失敗した時、シンプルなアラートを表示したいと考えていますが、なかなかうまくいきません。

試したこと①

※alert(msg: )はアラート用に作った関数です。

swift

1 2func httpRequest(urlst: String) { 3 var req = URLRequest(url: URL(string: urlst)!) 4 req.httpMethod = "GET" 5 let task = URLSession.shared.dataTask(with: req, completionHandler: { data, response, error in 6 if error != nil { 7 self.alert(msg: "接続失敗") 8 return 9 } 10 }) 11} 12

これではアラートは出ませんでした。

試したこと②

swift

1override func viewWillAppear(_ animated: Bool) { 2 alertSignal = { (signal: Int) -> Void in 3 switch signal { 4 case: 0: 5 self.alert(msg: "接続失敗") 6  default: 7 break 8 } 9 } 10} 11 12var alertSignal: ((Int) -> Void)? 13 14func httpRequest(urlst: String) { 15 var req = URLRequest(url: URL(string: urlst)!) 16 req.httpMethod = "GET" 17 let task = URLSession.shared.dataTask(with: req, completionHandler: { data, response, error in 18 if error != nil { 19 self.alertSignal?(0) 20 return 21 } 22 }) 23} 24

これでも特に何も起こりませんでした。

試したこと③

swift

1override func viewWillAppear(_ animated: Bool) { 2 let notification = NotificationCenter.default 3 notification.addObserver(self, selector:#selector(severError(notification:)), name:Notification.Name(rawValue: "0"), object: nil) 4 5} 6 7@objc func serverError(notification: Notification) { 8 self.alert(msg: "接続失敗") 9} 10 11 12func httpRequest(urlst: String) { 13 var req = URLRequest(url: URL(string: urlst)!) 14 req.httpMethod = "GET" 15 let task = URLSession.shared.dataTask(with: req, completionHandler: { data, response, error in 16 if error != nil { 17 NotificationCenter.default.post(name: Notification.Name(rawValue: "0"), object: nil) 18 return 19 } 20 }) 21}

試したこと④

swift

1class ViewController: UIViewController, URLSessionTaskDelegate { 2 3 func httpRequest(urlst: String) { 4 var req = URLRequest(url: URL(string: urlst)!) 5 req.httpMethod = "GET" 6 let task = URLSession.shared.dataTask(with: req, completionHandler: { data, response, error in 7 //成功した場合のいろいろな処理 8 }) 9} 10 11func urlSession(_ session: URLSession, task: URLSessionTask, didCompleteWithError error: Error?) { 12 DispathQueue.main.async { 13 self.alert(msg: "接続失敗") 14} 15} 16} 17

このように、いろいろ試してみましたが、アラートは表示されませんでした。
解決への糸口をご存じの方がいらっしゃれば、よろしくお願いいたします。

追記

alert関数の中身はこんな感じになっています。

swift

1func alert(msg: String) { 2 let alert = UIAlertController(title: nil, message: msg, preferredStyle: .alert) 3 let cancel: UIAlertAction = UIAlertAction(title: "OK", style: .cancel, handler: nil) 4 alert.addAction(cancel) 5 present(alert:, animated: true, completion: nil) 6} 7

さらに追記

調べたところ、どうやらアラートを呼び出している最中にアラートを呼び出しているのが問題のようでした。
関係ないだろうと質問を省いていました、申し訳ございません。

swift

1 2@IBAction func Button(_ sender: Any) { 3 let alert = UIAlertController(title: nil, message: msg, preferredStyle: .alert) 4 let next: UIAlertAction = UIAlertAction(title: "次へ", style: .default, handler: {(action: UIAlertAction) -> Void in 5  httpRequest(urlst: "http://xxxxxxxxxxxxxxxxxxxxxxxxxxxxxx") 6}) 7 let cancel: UIAlertAction = UIAlertAction(title: "OK", style: .cancel, handler: nil) 8 alert.addAction(cancel) 9 alert.addAction(next) 10 present(alert, animated: true, completion: nil) 11} 12 13func httpRequest(urlst: String) { 14 var req = URLRequest(url: URL(string: urlst)!) 15 req.httpMethod = "GET" 16 let task = URLSession.shared.dataTask(with: req, completionHandler: { data, response, error in 17 DispathQueue.main.async { 18 self.alert(msg: "接続失敗") 19 } 20 }) 21 task.resume() 22} 23 24func alert(msg: String) { 25 let alert = UIAlertController(title: nil, message: msg, preferredStyle: .alert) 26 let cancel: UIAlertAction = UIAlertAction(title: "OK", style: .cancel, handler: nil) 27 alert.addAction(cancel) 28 present(alert, animated: true, completion: nil) 29} 30

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

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

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

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

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

fuzzball

2019/07/08 06:45 編集

「alert」とういうのが何なのか分かりませんが、その呼び出しの代わりに print("接続失敗") として表示されるのであれば、alert(msg:) の呼び出しをメインスレッドで行ってみて下さい。
TakuyaAso

2019/07/08 06:28

可能であれば alert 関数も見たいですね。
Haruto513

2019/07/08 06:33

alert(msg:)はUIAlertControllerを表示テキストだけ変えて使いまわすように作った関数です。 中身は let alert = UIAlertController(title: msg,... ~~ present(alert, animated: true, completion: nil) といった内容です。 メインスレッドで呼び出す、ということは、 https://developer.apple.com/documentation/code_diagnostics/main_thread_checker 上記のページに載っているように DispathQueue.main.async { code } を、completionHandlerの中で行えばいいということでしょうか。
fuzzball

2019/07/08 06:44

そうです。(聞く前に試しましょう)
Haruto513

2019/07/08 07:13

試してみて、やはり表示されないので不思議だと思ってデバッグエリアをよくよく見てみたら、すでにアラートが表示されているといった内容の文章を見つけました。実は、関係ないと思い込んで質問では省きましたが、関数httpRequestは、画面配置したボタン押下時、確認のアラートを表示してOKが押されたときに呼びされるという書き方をしていました。アラートコントローラーはすでに一つ呼び出されていると次のアラートが呼び出せないようになっている?ようでした。 どうやら自分の問題はこっちにあるようです。こちらの方向で、もう一度調べなおしてみます。
fuzzball

2019/07/08 07:22 編集

ちなみにprint文は表示されてたんですよね?表示は一回だけでしたか? あと、present(alert:completion:)の中身も教えて下さい。
Haruto513

2019/07/08 07:35

print文は表示されていました。表示も一回だけです。
fuzzball

2019/07/08 08:14

もう修正中かもしれませんが、とりあえず原因を特定させたいのであれば、sleep()を挟むとか、asyncの代わりにasyncAfterを使うなど、エラー用Alertの表示を遅らせてみてはどうでしょうか。
guest

回答3

0

自己解決

皆様解答等ありがとうございました。
以下のサイトを参考にしたら無事解決したことを報告いたします。

https://qiita.com/t_nagano/items/bc505982151fe73eb22d

投稿2019/07/23 06:30

Haruto513

総合スコア52

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

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

0

ネットワーク通信のスレッドはメインスレッドで実行されなこともありますが、alertはメインスレッドでしか表示されません。

Swift

1 DispatchQueue.main.async { 2 alert(msg: "接続失敗") 3 }

このようなコードでメインスレッドで実行できます。
いかがでしょうか?

投稿2019/07/11 14:51

k2moons

総合スコア184

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

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

0

URLSession.shared.dataTask(with:completionHandler:)の返り値を受け取っただけの状態になっていそうなので、通信処理自体が実行されていないように見えます。
task.resume()を実行してみてください。

投稿2019/07/10 15:23

marty-suzuki

総合スコア25

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.50%

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

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

質問する

関連した質問