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

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

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

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

Q&A

解決済

1回答

627閲覧

Thread 1: Fatal error: Unexpectedly found nil while implicitly unwrapping an Optional valueエラーの解消

vangogh---

総合スコア4

Swift

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

0グッド

0クリップ

投稿2022/01/19 12:03

編集2022/01/21 12:29

quizCountCorrectが3になったらアラームを止める動作を実装したいのですが、Thread 1: Fatal error: Unexpectedly found nil while implicitly unwrapping an Optional valueというエラーが出ます。numberOfLoopsを0にすることで止めることができるのかなと思ったのですが、、、

import UIKit import AVFoundation class QuizViewController: UIViewController { @IBOutlet var quizNumberLabel:UILabel! @IBOutlet var quizTextView:UITextView! @IBOutlet var answerButton1:UIButton! @IBOutlet var answerButton2:UIButton! @IBOutlet var answerButton3:UIButton! @IBOutlet var answerButton4:UIButton! var csvArray: [String] = [] var quizArray: [String] = [] var quizCount = 0 var quizCountCorrect = 0 let alarm = Alarm() override func viewDidLoad() { super.viewDidLoad() csvArray = loadCSV(fileName: "quiz") quizArray = csvArray[quizCount].components(separatedBy: ",") quizNumberLabel.text = "第\(quizCount + 1)問" quizTextView.text = quizArray[0] answerButton1.setTitle(quizArray[2], for: .normal) answerButton2.setTitle(quizArray[3], for: .normal) answerButton3.setTitle(quizArray[4], for: .normal) answerButton4.setTitle(quizArray[5], for: .normal) // Do any additional setup after loading the view. } @IBAction func btnAction(sender: UIButton) { if sender.tag == Int(quizArray[1]) { print("正解") quizCountCorrect += 1 } else { print("不正解") quizCountCorrect = 0 } nextQuiz() } func nextQuiz() { quizCount += 1 quizArray = csvArray[quizCount].components(separatedBy: ",") quizNumberLabel.text = "第\(quizCount + 1)問" quizTextView.text = quizArray[0] answerButton1.setTitle(quizArray[2], for: .normal) answerButton2.setTitle(quizArray[3], for: .normal) answerButton3.setTitle(quizArray[4], for: .normal) answerButton4.setTitle(quizArray[5], for: .normal) if quizCountCorrect == 3 { alarm.stopTimer() alarm.audioPlayer.numberOfLoops = 0 performSegue(withIdentifier: "GoodMorning", sender: nil) } } func loadCSV(fileName: String) -> [String] { let csvBundle = Bundle.main.path(forResource: fileName, ofType: "csv")! do { let csvData = try String(contentsOfFile: csvBundle, encoding: String.Encoding.utf8) let lineChange = csvData.replacingOccurrences(of: "\r", with: "\n") csvArray = lineChange.components(separatedBy: "\n") csvArray.removeLast() } catch { print("エラー") } return csvArray }

ちなみにAlarmファイルの一部を貼っておきます。

import AVFoundation import UIKit class Alarm{ var selectedWakeUpTime:Date? var audioPlayer: AVAudioPlayer! var sleepTimer: Timer? var seconds = 0 func runTimer(){ seconds = calculateInterval(userAwakeTime: selectedWakeUpTime!) guard sleepTimer == nil else { return } if sleepTimer == nil{ sleepTimer = Timer.scheduledTimer(timeInterval: 1, target: self, selector: #selector(updateTimer), userInfo: nil, repeats: true) } } @objc private func updateTimer(){ if seconds != 0{ seconds -= 1 print(seconds) }else{ sleepTimer?.invalidate() sleepTimer = nil let soundFilePath = Bundle.main.path(forResource: "Clock-Alarm05-1", ofType: "mp3")! let sound:URL = URL(fileURLWithPath: soundFilePath) print(sound) do { audioPlayer = try AVAudioPlayer(contentsOf: sound, fileTypeHint:nil) try AVAudioSession.sharedInstance().setCategory(.playback, mode: .default, options: []) try AVAudioSession.sharedInstance().setActive(true) } catch { print("Cannot load file") } audioPlayer.numberOfLoops = -1 audioPlayer.play() print(audioPlayer.isPlaying) } } func stopTimer(){ if sleepTimer != nil { sleepTimer!.invalidate() sleepTimer = nil } } private func calculateInterval(userAwakeTime:Date)-> Int{ var interval = Int(userAwakeTime.timeIntervalSinceNow) if interval < 0{ interval = 86400 - (0 - interval) } let calendar = Calendar.current let seconds = calendar.component(.second, from: userAwakeTime) return interval - seconds } }

回答いただけると嬉しいです。質問には何でも答えます。よろしくお願いします。

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

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

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

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

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

hoshi-takanori

2022/01/19 21:30

そのエラーはどこで起こってますか? また、断片的なコードを貼られても、変数がどこでどう (どんな型で) 宣言されてるか分からないと答えようがありません。
vangogh---

2022/01/20 08:40

ありがとうございます。 alarm.audioPlayer.numberOfLoops = 0 このコードにエラーが出ています。 コードを全て載せさせていただきました。よろしくお願いします。
guest

回答1

0

ベストアンサー

QuizViewController の alarm.audioPlayer.numberOfLoops = 0 でそのエラーが出るってことは、alarm.audioPlayer が nil だと思われます。

Alarm クラスの audioPlayer プロパティがどうなってるかを抜き出すと、

swift

1class Alarm{ 2 3 // audioPlayer が ! 付きで宣言されてるので、初期値は nil 4 var audioPlayer: AVAudioPlayer! 5 6 // タイマーが動作している間、毎秒呼ばれるメソッド 7 @objc private func updateTimer(){ 8 9 // 残り時間が 0 秒でない場合、秒数を減らす 10 if seconds != 0{ 11 seconds -= 1 12 13 // 残り時間が 0 秒になったら、タイマーを停止して音を鳴らす 14 }else{ 15 do { 16 // ここで初めて audioPlayer に値がセットされる 17 audioPlayer = try AVAudioPlayer(contentsOf: sound, fileTypeHint:nil) 18 19 // 以下略

となってますが、このクラスの使われ方は、

  1. 画面を開くと同時に、Alarm オブジェクトが生成される
  2. クイズ開始時に runTimer が呼ばれ、タイマーが起動される (呼んでるところが見当たりませんが…)
  3. タイマー起動後、毎秒 updateTimer が呼ばれて、seconds がどんどん減っていく

4a. 決められた時間が経つ前に全問解答した場合、タイマーと音を止める
4b. 決められた時間が経つと、updateTimer の else で音を鳴らす

という感じだと思います。ここで、1 〜 3 では audioPlayer は生成されておらず、ずっと nil のままです。そして、4b に来ればそこで audioPlayer を生成して音を鳴らすのでいいのですが、4a に来た場合には audioPlayer は nil のままなのにその numberOfLoops プロパティに値をセットしようとしてエラーになります。


解決策としては、audioPlayer を ! から ? にして alarm.audioPlayer?.numberOfLoops = 0 とするか、Alarm クラスの初期化の際に audioPlayer を生成するか、でしょうね。(鳴らす直前に audioPlayer を生成すると音声ファイルの読み込みに時間がかかってしまう可能性があるので、たぶん後者の方がいいのかな…。)

そして、根本的な問題点としては、! は基本的に使うべきではないってことですね。例外は @IBOutlet くらいでしょう。また、QuizViewController から音を止めてますが、音を止めるのは stopTimer メソッドの中でやるべきかと。

ついでに言うと、updateTimer メソッドでは sleepTimer?.invalidate() でタイマーを止めてますが、stopTimer メソッドでは sleepTimer!.invalidate() と ! が使われてるのも気になります。if sleepTimer != nil でチェックしてるとはいえ、! を使うこと自体が間違いの元だと考えるべきです。if を使うなら if let すべきだし、この場合は ? で充分でしょう。

投稿2022/01/21 03:21

編集2022/01/21 03:29
hoshi-takanori

総合スコア7893

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.50%

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

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

質問する

関連した質問