Swift part10 [無断転載禁止]©2ch.net

■ このスレッドは過去ログ倉庫に格納されています
2017/02/20(月) 10:00:13.40ID:ChbPWtRt0
WWDC2014で発表されたAppleの新言語Swiftについて語りましょう

関連スレ

プログラミング言語Swift Part4
http://potato.2ch.net/test/read.cgi/mac/1484763495/

[SDK]iPhoneアプリ開発初心者質問箱48[touch][iPad]
http://potato.2ch.net/test/read.cgi/mac/1484217623/

Xcode part14
http://potato.2ch.net/test/read.cgi/mac/1476190499/

Swiftアンチスレ part1
http://echo.2ch.net/test/read.cgi/tech/1458491343/

前スレ
Swift part9
http://echo.2ch.net/test/read.cgi/tech/1476758084/
VIPQ2_EXTDAT: default:vvvvv:1000:512:----: EXT was configured
2デフォルトの名無しさん (ワッチョイ 7ba1-vagz)
垢版 |
2017/02/20(月) 11:06:31.94ID:Gb72M66o0
< `∀´>ニダー
2017/02/20(月) 13:14:50.34ID:mI2RJMjC0
さて、また荒らしと文字列の扱いについて談笑しようじゃないか
2017/02/20(月) 13:18:27.57ID:ij1Njg09d
極上の言語のスレはここかな?
5デフォルトの名無しさん (ワッチョイ 8bc9-ykbm)
垢版 |
2017/02/20(月) 16:05:25.21ID:IDSTiL890
WebAPIを叩く最小限のコードを書いています。
Terminal.Appで実行したところ、動くには動くのですが、最後のsleep文をコメントアウトすると、結果が表示されません。
sleep文はダサいので、うまい具合にbackgroundで実行中のThreadを待ち合わせる方法は無いでしょうか?なお、Swift3.0.2です。

import Foundation
var dic: Any = ["": ""]

func printJSON(_ data: Data) {
do {
let json = try JSONSerialization.jsonObject(with: data, options: .allowFragments)
dic = json
print(json)
} catch {
print("parse error!")
}

}
let url = URL(string: "http://date.jsontest.com/";)!
let task = URLSession.shared.dataTask(with: url) { data, response, error in
if let jsonData = data {
printJSON(jsonData)
DispatchQueue.main.async(execute: { print("dic = ¥(dic)") })
//mainスレッドを捕まえて実行
}
}
task.resume()
print("OK")
sleep(1) // 1秒待つ
2017/02/20(月) 17:57:22.66ID:E/pOlJmJ0
Dispatch Group使ってwaitするか
DispatchWorkItem使ってwaitするか
かな
2017/02/20(月) 18:12:47.50ID:mI2RJMjC0
NSConditionでの実装例
ttp://swift.sandbox.bluemix.net/#/repl/58aab29626c3ba5cbe1d44ee

文字列でなんやかんや話してたエロい人たちがより良い例を出してくれると信じてる
2017/02/20(月) 20:20:31.52ID:jVgNOv8dd
>>5
Sleep()がカッコ悪いのでsyncで呼ぶ
2017/02/20(月) 21:23:20.98ID:E/pOlJmJ0
let task = URLSession.shared.dataTask(with: url) { data, response, error in

動きがなんかおかしいと思ったら
これcompletionHandler設定できてないような
2017/02/20(月) 21:45:53.42ID:E/pOlJmJ0
最後にdispatchMain()を追加して明示的にexit()すればいいみたい
http://stackoverflow.com/questions/31944011/how-to-prevent-a-command-line-tool-from-exiting-before-asynchronous-operation-co

#! /usr/bin/env swift
import Foundation

var dic: Any = ["": ""]

func printJSON(_ data: Data) {
do {
let json = try JSONSerialization.jsonObject(with: data, options: .allowFragments)
dic = json
print(json)
} catch {
print("parse error!")
}

}

let url = URL(string: "http://date.jsontest.com/";)!
let task = URLSession.shared.dataTask(with: url, completionHandler: { data, response, error in
if let jsonData = data {
printJSON(jsonData)
DispatchQueue.main.async(execute: {print("dic = ¥(dic)"); exit(EXIT_SUCCESS)})
}
})

task.resume()
print("OK")
dispatchMain()
2017/02/20(月) 21:51:31.66ID:E/pOlJmJ0
関係ないけどこの外部ステートへの依存の仕方はちょっと気持ち悪く感じる
var dic: Any = ["": ""]
12デフォルトの名無しさん (ワッチョイ 8b5b-fFgi)
垢版 |
2017/02/20(月) 21:53:42.29ID:4Xv+kGUd0
>>9
メソッド引数の最後のclosureは()から出して記述できるんでは?
ただ、
DispatchQueue.main.async(execute: { print("dic = ¥(dic)") })

の部分がPlaygroundでは実行されるのに、terminal.appでは実行されない?
13デフォルトの名無しさん (ワッチョイ 8b5b-fFgi)
垢版 |
2017/02/20(月) 22:01:34.57ID:4Xv+kGUd0
>>10
素晴らしい!
dispatchMain()
このグローバル関数が何をしてるのか?ようわからんけど。
2017/02/20(月) 22:05:11.29ID:E/pOlJmJ0
>>12
あらまほんとだ
ラベル関係なく使えるのね
15デフォルトの名無しさん (ワッチョイ 8b5b-fFgi)
垢版 |
2017/02/20(月) 22:12:48.75ID:4Xv+kGUd0
Executes blocks submitted to the main queue.

って事は、
DispatchQueue.main.async(execute: { print("dic = ¥(dic)") })
を実行しているみたいだ。dispatchMain()は。
exit(EXIT_SUCCESS)が無いと、dispatchMain()は永遠に実行待ちするみたい。

しょうが無いので、^Z + kill %1した。
2017/02/20(月) 22:14:47.48ID:SiY39E3I0
while task.state == .running {
RunLoop.current.run(mode: .commonModes, before: .distantFuture)
}
2017/02/20(月) 23:26:54.19ID:SiY39E3I0
extension URLSessionTask {
func wait() {
while state == .running {
RunLoop.current.run(mode: .commonModes, before: .distantFuture)
}
}
}

let task = ...dataTask(...) { ... }
task.resume()
task.wait()

これならdownloadTaskとか他のtaskでも、上のextensionで1つで全部対応できていいと思う
task.wait()だけで済むから、DispatchGroupとかDispatchSemaphoreみたいにenter/leave/signalとかが各所に散らばる面倒臭さもない

DispatchSemaphore使ってsyncDataTaskみたいなのextensionに書く例stackoverflowにあったけど
これでもいいけど他のtask使いたくなったとき、そのtaskのsyncバージョンをまた別に書かないといけないのが面倒臭い
http://stackoverflow.com/a/34308158

dispatchMainはexitの置き場所で困りそう
2つのtask待ち合わせるならどこにexit置くのと考えると問題を先送りしてるだけな気がする
2017/02/20(月) 23:34:47.43ID:6K9wp/bS0
>>4
そうそう
プログラミング言語史上類を見ない最強にエレガントでモダンな至高言語Swiftのスレだよ
Enjoy!
2017/02/21(火) 07:58:47.31ID:eTJT09tJ0
dispatchMain
> Applications ... must not call dispatchMain()

RunLoop
> You should never try to call the methods of an RunLoop object running in a different thread

なんでリファレンスで危ないから使うなって言われてるものを優先して挙げるのか
DispatchWorkItemのサンプルはよ、一番これが「モダン」だと思う
20デフォルトの名無しさん (ワッチョイ 8bc9-ykbm)
垢版 |
2017/02/21(火) 08:14:40.71ID:3Vda5W860
>>19
次の3つのGlobal関数内では使ってはイケナイって書いてある。
今回はOK!

UIApplicationMain(_:_:_:_:) (iOS), NSApplicationMain(_:_:) (macOS), or CFRunLoopRun()
21デフォルトの名無しさん (ワッチョイ 8bc9-ykbm)
垢版 |
2017/02/21(火) 08:17:33.97ID:3Vda5W860
>>20
追伸です。
19のコメントは間違ってました。
アプリ内で使うのは良く無いそうです。Terminal.Appで実行する時だけにした方が良いです。
2017/02/21(火) 08:36:49.36ID:EuRy1Wt8d
>>20
その顔文字みたいなのは何なの?...
間違いじゃないなら可読性絶望的すぎじゃね?
2017/02/21(火) 08:42:18.14ID:4QHQyE1ya
それが顔文字に見えるなら、お前そうとう病んでるよ
24デフォルトの名無しさん (ワッチョイ 8bc9-ykbm)
垢版 |
2017/02/21(火) 08:47:28.88ID:3Vda5W860
>>22
セレクタでっせ。
メソッド名とラベルから構成される。ラベルが省略されたセレクタもあって、underscoreで省略可を示す訳だ。
C++で言う、シグネチャだわな。
25デフォルトの名無しさん (ワッチョイ 8bc9-ykbm)
垢版 |
2017/02/21(火) 08:49:53.12ID:3Vda5W860
セレクタを記述できないと、Notificationを扱う事ができないので、mustな!
26デフォルトの名無しさん (ワッチョイ 8bc9-ykbm)
垢版 |
2017/02/21(火) 08:53:52.12ID:3Vda5W860
Overload resolver(メソッド多重定義解決)はセレクタ情報を手掛かりに、どのメソッドを呼び出すのか?解決する訳だ。
参照:C++ FAQ
2017/02/21(火) 08:59:24.03ID:arZsGKioF
Objective-Cのときってもっとわかりやすい表記だったような
2017/02/21(火) 09:23:38.61ID:eTJT09tJ0
ObjCの表記を覚えてない => ObjC使ってない
Swiftの表記を自然に読み取れない => Swift使ってない

うーん、このどうしようもない感
2017/02/21(火) 09:35:17.94ID:HF5TYBzbM
swiftってiosアプリ書く以外であえて選んでるって人いるの?
2017/02/21(火) 09:40:38.00ID:wDwSQYHdd
macアプリ…
2017/02/21(火) 12:37:07.59ID:01HI8sK+d
>>29
IBM.........
2017/02/21(火) 12:40:37.23ID:HroqGx6Ep
>>29
SUZUKI.........
2017/02/21(火) 13:29:54.58ID:h4Mc9UUPM
>>19
current threadのrunloopインスタンスのmethodを呼び出しているのだからまったく問題ないだろ…
2017/02/21(火) 16:29:44.15ID:m+/zu2/z0
1日経って思ったけど、 >>17 じゃtaskを待ってはいるけど
completionHandlerを待ってはいないから >>5 への回答には全くなってなかったね
忘れてくれ
completionHandlerに入った時点でtask.stateは.completedだった
2017/02/21(火) 16:47:25.83ID:IX256A9wp
>>33
他に適したAPIがある上、while state がsleep 並にダサい
無限ループで状態監視とか日曜プログラマでも回避するコードだ
2017/02/21(火) 18:04:35.19ID:aFqTV30T0
特に無限ループがダサいとは思わないけどな
少し低いレイヤーを意識したコードではあると思うけど
2017/02/21(火) 18:08:20.55ID:aFqTV30T0
DispatchWorkItem版
if #availableはguard文だとコンパイル通らなかった‥

if #available(OSX 10.10, *) {
let queue = DispatchQueue(label: "queue", attributes: .concurrent)
let printTask = DispatchWorkItem { print("dic = ¥(dic)") }
let url = URL(string: "http://date.jsontest.com/";)!
let task = URLSession.shared.dataTask(with: url) { data, response, error in
if let jsonData = data {
printJSON(jsonData)
queue.async(execute: printTask)
}
}

task.resume()
printTask.wait()
exit(EXIT_SUCCESS)

} else {
print("require OSX10.10 or newer"); exit(EXIT_FAILURE);
}
2017/02/21(火) 18:37:29.74ID:m+/zu2/z0
スクリプトモードでメインスレッドでメインキューを待つ方法と、URLSessionで同期的に実行する方法で問題をごっちゃにしてたな
>>5 の質問の本意がどっちにあるのかわからないけど、「スクリプトモードでメインスレッドでメインキューを待つ方法」ならRunLoopを回すのが正しい回答なはず


単純化すれば
DispatchQueue.main.async {
sleep(1)
print("Hello")
}
print("Done")
これでスクリプトモードでどうやって "Hello" を表示させるか
できれば"Hello"→"Done"の順序で

この場合DispatchSemaphore等ではwaitした時点でメインキューに入れた非同期タスクに永久にたどり着けないのでwaitで固まるので誤り
待ってるのも非同期タスクも両方メインだから
let sema = DispatchSemaphore(value: 0)
DispatchQueue.main.async {
sleep(1)
print("Hello")
sema.signal()
}
sema.wait()
print("Done")
2017/02/21(火) 19:10:04.95ID:IX256A9wp
>>36
ttp://swift.sandbox.bluemix.net/#/repl/58ac0f8c861c326c636916bf
一般常識として無限ループの状態監視するやつは死ねと思う
SIGNALが使えないほどの低レイヤーでそれが必要だとしてもsleepは入れる

ループ中にsleep入れないなら死ぬし、sleep入れるなら本末転倒だよねー
2017/02/21(火) 19:13:38.94ID:lTdRkQq/M
ビジーループと無限ループをごっちゃにして死ねとかアホか
2017/02/21(火) 19:17:25.53ID:IX256A9wp
いや、だからsleep入れろって言ってるだろw
2017/02/21(火) 20:44:27.44ID:5UhBXSO50
RunLoopのSwift実装読んだけど酷過ぎじゃね
returnAfterSourceHandledを強制trueにして
ブロッキングキューの利点ぶち壊すとか意味分からん

別実装で書いてみた
http://swift.sandbox.bluemix.net/#/repl/58ac26055d046936d91eba1c
2017/02/21(火) 21:21:46.46ID:5UhBXSO50
CFRunLoopRunInMode(,,false)がポンコツでなければ
キューが空の間はセマフォのwait等と同様に
待機状態になるはずでCPUは消費しない
2017/02/21(火) 21:33:17.76ID:jelvKUPR0
PR出してみたら?コードがあるなら前スレのAffineTransformよりはマシなレビューが期待できそう
2017/02/21(火) 21:58:36.49ID:5UhBXSO50
出すとしたら
https://github.com/apple/swift-corelibs-foundation/blob/master/Foundation/NSRunLoop.swift#L149
> CFRunLoopRunInMode(modeArg, ti, true)

ここの第3引数も指定出来て,
CFRunLoopRunInModeの戻り値をenumに入れ替えて返す
runのオーバーロード追加を要求するくらいか

とはいえコンソール系でのテスト実行以外では
UIApplicationMain等に任せる所だし放置になるのでは
2017/02/21(火) 22:21:50.16ID:jelvKUPR0
ならいらねーなw
2017/02/22(水) 08:12:05.22ID:l4ZVTbpn0
お疲れ様でした
2017/02/22(水) 12:33:22.98ID:pE0UWB3Fd
>>5
質問と直接関係ないけど

この文脈でのsleep()はc言語の、つまりPOSIX APIのsleep()が呼び出されると思うんだよな

無自覚にインラインCになっちゃうって、便利なのか?、危険なのか?
49デフォルトの名無しさん (ワッチョイ 8bc9-ykbm)
垢版 |
2017/02/22(水) 12:36:17.73ID:XTvCwCCI0
>>48
Thread Type Method sleep(forTimeInterval:)
こんなのもあるね。秒単位ね。
■ このスレッドは過去ログ倉庫に格納されています