今日現在、以下の設定でやっています。
🧑🏻💻 お天気情報を取得する
無料で公開されている WEB API を使います。
シンプルに取得する。
import SwiftUI
struct TestWeatherView: View {
private let url = URL(string: "https://wttr.in/?format=3")!
var body: some View {
Text(try! String(contentsOf: url))
}
}
#Preview {
TestWeatherView()
}
data:image/s3,"s3://crabby-images/f61e7/f61e7a571d5a0640742bfb4689b51e2e8368f3cf" alt=""
ちょっと窮屈なので砕いていきます。
struct TestWeatherView: View {
@State private var text = ""
private let url = URL(string: "https://wttr.in/?format=3")!
var body: some View {
Text(text)
.onAppear {
text = try! String(contentsOf: url)
}
}
}
OK です。
次は、Web クライアントを汎用性のある URLSession
に変えます。
struct TestWeatherView: View {
@State private var text = ""
private let url = URL(string: "https://wttr.in/?format=3")!
var body: some View {
Text(text)
.task {
let (data, _) = try! await URLSession.shared.data(from: url)
text = String(data: data, encoding: .utf8)!
}
}
}
ここで、警告がでます。
data:image/s3,"s3://crabby-images/4ffd2/4ffd2cb999f1d5443907844fa069afc29a9e6a7f" alt=""
Passing argument of non-sendable type '(any URLSessionTaskDelegate)?' outside of main actor-isolated context may introduce data races
これは、なんですか。
🤔 Passing argument of non-sendable type '(any URLSessionTaskDelegate)?' outside of main actor-isolated context may introduce data races
URLSession
のドキュメントを見ておきます。
func data(from url: URL) async throws -> (Data, URLResponse)
👉 data(from:) | Apple Developer Documentation data:image/s3,"s3://crabby-images/4cdf8/4cdf851b141e5f97e51ab8e7ae7abb4bdb5fd2eb" alt="hatena-bookmark"
Task
内の以下の部分
let (data, _) = try! await URLSession.shared.data(from: url)
左辺はメインスレッド、
--- 境界 ---
右辺はバックグラウンドスレッド
👉 【Swift】concurrency をマスターするための一つのきっかけ data:image/s3,"s3://crabby-images/bceb8/bceb809756494e578630020aa72bfd58bf6ad6aa" alt="hatena-bookmark"
ということで、
いわゆる「アクター境界」を越えているので、
data()
の返り値は Sendable
でなければなりません。
リファレンスやコードを追いかけてみると、
Data
, URLResponse
共に Sendable
に準拠しています。
data:image/s3,"s3://crabby-images/fd413/fd4132c092a8d2741b8c7b47785082bfe7c588c1" alt=""
👉 Data | Apple Developer Documentation data:image/s3,"s3://crabby-images/f4b3d/f4b3db62cca571aa6323d3b0e3849503c200b349" alt="hatena-bookmark"
👉 URLResponse | Apple Developer Documentation data:image/s3,"s3://crabby-images/5fc9d/5fc9d819be5bb2846558839c7993c7be4f31efab" alt="hatena-bookmark"
あ、Tuple
かな ?
などと思いましたがなんか違う。
👉 Pitch: User-defined tuple conformances - Evolution / Pitches - Swift Forums data:image/s3,"s3://crabby-images/04b0b/04b0bd33e96b45d54da9a9344d07cfbb6ac1855c" alt="hatena-bookmark"
🧑🏻💻 Actor を使う
Actor
は Sendable
に準拠しています。
なので、これでバックグラウンドの通信部分をラップします。
actor WeatherA {
// OK
static func getData(url: URL) async -> (Data, URLResponse) {
try! await URLSession.shared.data(from: url)
}
// OK
nonisolated func getData(url: URL) async -> (Data, URLResponse) {
try! await URLSession.shared.data(from: url)
}
}
これで、無警告でOKとなりました。
🧑🏻💻 extension 化してメインスレッドを避ける
こんな方法でもいけます。
extension URLSession {
func dataEx(url: URL) async -> (Data, URLResponse) {
try! await data(from: url)
}
}
🧑🏻💻 まとめ
分かれば、なるほど感あるけど、分からなければ全く謎で時間だけ食うので、
そんな誰かと自分用のメモとして。
class でも警告なしでいけるようです。
「Actor
はデータの非同期操作のために作られてる」感じがする。
外野から使ってみた雰囲気だけだけれども。
🧑🏻💻 参考