16

在下面的代码中,当按下按钮时,应该在控制台中打印“测试”,但事实并非如此。该事件不通过发布者发送。知道 Xcode 11 Beta 5 中的 PassthroughSubject 发生了什么吗?(在 Xcode 11 Beta 4 中运行良好)

var body: some View {  

    let publisher = PassthroughSubject<String, Never>()

    publisher.sink { (str) in  
        print(str)  
    }  
    return Button("OK") {  
        publisher.send("Test")  
    }  
}

PS我知道按下按钮时还有其他方法可以打印字符串,我只想展示一个简单的发送接收示例

4

2 回答 2

48

.sink()返回一个AnyCancellable对象。你永远不应该忽视它。永远不要这样做

// never do this!
publisher.sink { ... }
// never do this!
let _ = publisher.sink { ... }

如果将其分配给变量,请确保它不是短暂的。一旦可取消对象被释放,订阅也将被取消。

// if cancellable is deallocated, the subscription will get cancelled
let cancellable = publisher.sink { ... }

既然您要求sink在视图中使用,我将发布一种方法。但是,在视图中,您可能应该.onReceive()改用。它更简单。

使用水槽:

在视图中使用它时,您需要使用一个@State变量,以确保它在生成视图主体后仍然存在。

DispatchQueue.main.async是必需的,以避免在视图更新时修改状态。如果你不这样做,你会得到一个运行时错误。

struct ContentView: View {
    @State var cancellable: AnyCancellable? = nil

    var body: some View {
        let publisher = PassthroughSubject<String, Never>()

        DispatchQueue.main.async {
            self.cancellable = publisher.sink { (str) in
                print(str)
            }
        }

        return Button("OK") {
            publisher.send("Test")
        }
    }
}

使用.onReceive()

struct ContentView: View {

    var body: some View {
        let publisher = PassthroughSubject<String, Never>()

        return Button("OK") {
            publisher.send("Test")
        }
        .onReceive(publisher) { str in
            print(str)
        }
    }
}
于 2019-07-30T13:29:54.777 回答
13

订阅接收器时缺少 .store。您可以使用 .onReceive,但您的代码没有接收到值,因为您需要添加 .store(in: &subscription)

var body: some View {  
    var subscription = Set<AnyCancellable>()
    let publisher = PassthroughSubject<String, Never>()

    publisher.sink { (str) in  
        print(str)  
    }.store(in: &subscription)

    return Button("OK") {  
        publisher.send("Test")  
    }  
}
于 2019-11-09T09:37:52.367 回答