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

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

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

Swift Playgroundは、Swiftをインタラクティブに習得できるiPad向けのアプリケーション。コーディングの知識は一切必要なく、Swift Playgrounds上でプログラミングしたコードによりドローン・ロボットを自在に動かすことが可能です。

for

for文は、様々なプログラミング言語で使われている制御構造です。for文に定義している条件から外れるまで、for文内の命令文を繰り返し実行します。

同期

複数のディレクトリに存在するファイルを更新した場合に、すべてのファイルにも更新が行われる事、又は、同じ記憶領域に同時にアクセスして内容の整合性が失われてしまう事をを防ぐ制御などを同期と呼びます。

Xcode

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

Swift

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

Q&A

解決済

1回答

1577閲覧

Swift for文の中でUIの描画をしたい

tis

総合スコア20

Swift Playground

Swift Playgroundは、Swiftをインタラクティブに習得できるiPad向けのアプリケーション。コーディングの知識は一切必要なく、Swift Playgrounds上でプログラミングしたコードによりドローン・ロボットを自在に動かすことが可能です。

for

for文は、様々なプログラミング言語で使われている制御構造です。for文に定義している条件から外れるまで、for文内の命令文を繰り返し実行します。

同期

複数のディレクトリに存在するファイルを更新した場合に、すべてのファイルにも更新が行われる事、又は、同じ記憶領域に同時にアクセスして内容の整合性が失われてしまう事をを防ぐ制御などを同期と呼びます。

Xcode

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

Swift

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

0グッド

0クリップ

投稿2021/03/31 16:47

前提・実現したいこと

Swiftの質問です。
ボタンを押すとFor文の処理が走り、For文で処理している最中にLabelの文字を逐一変えたいです。
しかしうまくいかずFor文の処理が完全に終わるまでLabelは何も変わりません。

ストップウォッチなどのように数字が目にも止まらない速さで変わっていくような描画やアプリのローディング(進捗の%がわかる)のような描画をしたいです。

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

なし

該当のソースコード

下のソースコードではUITextViewから取得した長い英語の文の単語数を数えるプログラムになっております。

もしお試しになる場合は下のリンクなどからサンプルの英文をコピーしてみてください。
英語文字列長文サンプル

Swift

1 func countWords(){ 2 str = textView.text! 3 let arr:[String] = str.components(separatedBy: " ") 4 5 for i in 0..<arr.count{ 6 let word = arr[i] 7 var count = 0 8 for num in arr { 9 if num == word { 10 count = count + 1 11 } 12 } 13 print(i) 14 print("(word)が(count)個ありました") 15 16 //ここでUIを変更しているのに変わらない 17 totalCountLabel.text = "計(i)個" 18 } 19}

試したこと

クロージャー,コンプレーションハンドラーなどの書き方を試してみましたが、同じ動作で、なかなかうまく行きませんでした。

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

Swift5

ご回答していただけたら本当ありがたいです。
よろしくお願いいたします!

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

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

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

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

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

tomato879241

2021/03/31 23:51

For-loopの中でusleep()を使ってください。
tis

2021/04/01 08:57

hoshi-takanoriさん for文の中の処理をDispatchQueue.main.asyncAfterで0.5秒遅らせましたが、できませんでした。 どのタイミングで遅延処理を行えばUIの更新を行ってくれるのでしょうか? ありがとうございます。
tis

2021/04/01 09:01

tomato879241さん usleep(500000)、、、 (0.5秒遅らせる)をFor-loopの中で行う処理の前に書きましたが処理が遅くなっただけでUIの更新は行 われませんでした。 これも遅延処理だと思いますが、遅延させただけでUIの更新が行われるようになるのですか? ありがとうございます。
guest

回答1

0

ベストアンサー

Timer を使って書くとこんな感じでしょうか。

swift

1class ViewController: UIViewController { 2 3 @IBOutlet weak var totalCountLabel: UILabel! 4 5 var timer: Timer? 6 7 override func viewDidLoad() { 8 super.viewDidLoad() 9 // Do any additional setup after loading the view. 10 } 11 12 @IBAction func countWords() { 13 if timer != nil { 14 return 15 } 16 17 let str = """ 18 Alice was beginning to get very tired of sitting by her sister on the bank, and of having nothing to do. Once or twice she had peeped into the book her sister was reading, but it had no pictures or conversations in it, "and what is the use of a book," thought Alice, "without pictures or conversations?" 19 """ 20 let arr: [String] = str.components(separatedBy: " ") 21 22 var i = 0 23 timer = Timer.scheduledTimer(withTimeInterval: 0.1, repeats: true) { timer in 24 let word = arr[i] 25 let count = arr.filter { $0 == word }.count 26 print(i) 27 print("(word)が(count)個ありました") 28 self.totalCountLabel.text = "計(i)個" 29 30 i += 1 31 if i == arr.count { 32 timer.invalidate() 33 self.timer = nil 34 } 35 } 36 } 37}

Thread を使う方法も書いておきます。

swift

1 @IBAction func countWords() { 2 let str = """ 3 Alice was beginning to get very tired of sitting by her sister on the bank, and of having nothing to do. Once or twice she had peeped into the book her sister was reading, but it had no pictures or conversations in it, "and what is the use of a book," thought Alice, "without pictures or conversations?" 4 """ 5 let arr: [String] = str.components(separatedBy: " ") 6 7 Thread.detachNewThread { 8 // バックグラウンドスレッドで処理を行う。 9 for i in 0..<arr.count { 10 let word = arr[i] 11 let count = arr.filter { $0 == word }.count 12 print(i) 13 print("(word)が(count)個ありました") 14 15 // UI の更新はメインスレッドで行う必要がある。 16 DispatchQueue.main.async { 17 self.totalCountLabel.text = "計(i)個" 18 } 19 20 Thread.sleep(forTimeInterval: 0.1) 21 } 22 } 23 }

投稿2021/04/01 09:32

編集2021/04/01 17:51
hoshi-takanori

総合スコア7895

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

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

tis

2021/04/01 11:12

試してみたところ高速でUIの描画をすることができました。 for文と比較してもほぼ処理スピードに違いも出ませんでした。 このやり方は他のプログラミング言語でも使われるようなやり方なのでしょうか。Timerを使うやり方以外にはないのでしょうか?それも知りたいと思っております。 処理を待つ時のLoading画面などはいろんなアプリで使われており、そういったものは基本的に全てこのTimerのような方法で実装されているのでしょうか? ご回答ありがとうございます!
hoshi-takanori

2021/04/01 17:56

OS や言語や処理の内容によっていろいろなやり方があり、すべてを紹介するのは難しいですね。 とりあえず Thread を使う方法を追加しておきましたが、データを更新する場合は競合に気をつける必要があります。
tis

2021/04/02 06:45

Threadを使う方法もありがとうございます。こうやってサブスレッドでやるんですね。 こちらも試してみました。 UIの高速描画は行われますがタイマーを使った時よりも処理が2倍くらい遅かったです。 他の機能を実装するときサブスレッドも使った実装も検討していきたいと思います。 ありがとうございます!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問