gpt4 book ai didi

ios - Swift Combine-多线程上的Publishers.CombineLatest

转载 作者:行者123 更新时间:2023-12-01 15:49:12 25 4
gpt4 key购买 nike

当将Publishers.CombineLatest与运行主Main以外的线程的Publisher一起使用时,Publishers.CombineLatest的.sink不会总是被调用。

并非每次都出现此问题,这就是为什么我创建了单元测试,并连续尝试100次的原因。通常它们在4-5次迭代后失败。

import XCTest
import Combine

class CombineLatestTests: XCTestCase {

override func setUp() {
continueAfterFailure = false
}

func testCombineLatest_receiveOn() {
for x in 0...1000 {
print("---------- RUN \(x)")
let queue1 = DispatchQueue.global(qos: .userInitiated)
let queue2 = DispatchQueue.global(qos: .background)

let subj1 = PassthroughSubject<Int, Never>()
let subj2 = PassthroughSubject<Int, Never>()

let publ1 = subj1.receive(on: queue1).map { value -> Int in
print("-- Observer 1: \(value), Thread: \(Thread.current)")
return value
}
let publ2 = subj2.receive(on: queue2).map { value -> Int in
print("-- Observer 2: \(value), Thread: \(Thread.current)")
return value
}

let exp = expectation(description: "expect values")
exp.assertForOverFulfill = false
let canc = Publishers.CombineLatest(publ1, publ2)
.sink { value1, value2 in
print("-- recieved \(value1):\(value2) on \(Thread.current)")
if value1 == 10, value2 == 20 {
exp.fulfill()
}
}

subj1.send(5)
subj2.send(20)
subj1.send(10)

wait(for: [exp], timeout: 10)
canc.cancel()
}
}

func testCombineLatest_currentValue_receiveOn() {
for x in 0...100 {
print("---------- RUN \(x)")
let queue1 = DispatchQueue.global(qos: .userInitiated)
let queue2 = DispatchQueue.global(qos: .background)

let subj1 = CurrentValueSubject<Int, Never>(0)
let subj2 = CurrentValueSubject<Int, Never>(0)

let publ1 = subj1.receive(on: queue1).map { value -> Int in
print("-- Observer 1: \(value), Thread: \(Thread.current)")
return value
}
let publ2 = subj2.receive(on: queue2).map { value -> Int in
print("-- Observer 2: \(value), Thread: \(Thread.current)")
return value
}

let exp = expectation(description: "expect values")
exp.assertForOverFulfill = false
let canc = Publishers.CombineLatest(publ1,
publ2)
.sink { value1, value2 in
print("-- recieved \(value1):\(value2) on \(Thread.current)")
if value1 == 10, value2 == 20 {
exp.fulfill()
}
}

subj1.send(10)
subj2.send(20)

wait(for: [exp], timeout: 3)
canc.cancel()
}
}

func testCombineLatest_subscribeOn() {
for x in 0...100 {
print("---------- RUN \(x)")
let queue1 = DispatchQueue.global(qos: .userInitiated)
let queue2 = DispatchQueue.global(qos: .background)

let subj1 = PassthroughSubject<Int, Never>()
let subj2 = PassthroughSubject<Int, Never>()

let publ1 = subj1.map { value -> Int in
print("-- Observer 1: \(value), Thread: \(Thread.current)")
return value
}
let publ2 = subj2.map { value -> Int in
print("-- Observer 2: \(value), Thread: \(Thread.current)")
return value
}

let exp = expectation(description: "expect values")
exp.assertForOverFulfill = false
let canc = Publishers.CombineLatest(publ1, publ2)
.sink { value1, value2 in
print("-- recieved \(value1):\(value2) on \(Thread.current)")
if value1 == 10, value2 == 20 {
exp.fulfill()
}
}

queue1.async {
subj1.send(5)
subj1.send(10)
}

queue2.async {
subj2.send(20)
}

wait(for: [exp], timeout: 5)
canc.cancel()
}
}

}


这是第三次测试的日志
Test Case '-[xxxx.CombineLatestTests testCombineLatest_currentValue_receiveOn]' started.
---------- RUN 0
-- Observer 2: 0, Thread: <NSThread: 0x6000004e0f80>{number = 9, name = (null)}
-- Observer 1: 0, Thread: <NSThread: 0x6000004f0000>{number = 7, name = (null)}
-- Observer 2: 20, Thread: <NSThread: 0x6000004f6e00>{number = 6, name = (null)}
-- recieved 0:0 on <NSThread: 0x6000004f0000>{number = 7, name = (null)}
-- Observer 1: 10, Thread: <NSThread: 0x600000439880>{number = 4, name = (null)}
-- recieved 10:20 on <NSThread: 0x600000439880>{number = 4, name = (null)}
---------- RUN 1
-- Observer 2: 0, Thread: <NSThread: 0x6000004f0000>{number = 7, name = (null)}
-- Observer 1: 0, Thread: <NSThread: 0x6000004f6e00>{number = 6, name = (null)}
-- Observer 2: 20, Thread: <NSThread: 0x6000004e0f80>{number = 9, name = (null)}
-- recieved 0:0 on <NSThread: 0x6000004f6e00>{number = 6, name = (null)}
-- Observer 1: 10, Thread: <NSThread: 0x6000004e80c0>{number = 10, name = (null)}
-- recieved 10:20 on <NSThread: 0x6000004e80c0>{number = 10, name = (null)}
---------- RUN 2
-- Observer 2: 0, Thread: <NSThread: 0x6000004f6e00>{number = 6, name = (null)}
-- Observer 1: 0, Thread: <NSThread: 0x6000004e0f80>{number = 9, name = (null)}
-- Observer 2: 20, Thread: <NSThread: 0x6000004e80c0>{number = 10, name = (null)}
-- recieved 0:0 on <NSThread: 0x6000004e0f80>{number = 9, name = (null)}
-- Observer 1: 10, Thread: <NSThread: 0x600000439880>{number = 4, name = (null)}
-- recieved 10:20 on <NSThread: 0x600000439880>{number = 4, name = (null)}
---------- RUN 3
-- Observer 2: 0, Thread: <NSThread: 0x600000439880>{number = 4, name = (null)}
-- Observer 1: 0, Thread: <NSThread: 0x6000004e0f80>{number = 9, name = (null)}
-- Observer 2: 20, Thread: <NSThread: 0x6000004f6e00>{number = 6, name = (null)}
-- recieved 0:0 on <NSThread: 0x6000004e0f80>{number = 9, name = (null)}
-- Observer 1: 10, Thread: <NSThread: 0x6000004e80c0>{number = 10, name = (null)}
-- recieved 10:20 on <NSThread: 0x6000004e80c0>{number = 10, name = (null)}
---------- RUN 4
-- Observer 1: 0, Thread: <NSThread: 0x6000004f6e00>{number = 6, name = (null)}
-- Observer 2: 0, Thread: <NSThread: 0x6000004f0000>{number = 7, name = (null)}
-- recieved 0:0 on <NSThread: 0x6000004f0000>{number = 7, name = (null)}
-- Observer 1: 10, Thread: <NSThread: 0x600000439880>{number = 4, name = (null)}
-- Observer 2: 20, Thread: <NSThread: 0x6000004e80c0>{number = 10, name = (null)}
-- recieved 10:0 on <NSThread: 0x600000439880>{number = 4, name = (null)}
CombineLatestTests.swift:93: error: : Asynchronous wait failed: Exceeded timeout of 3 seconds, with unfulfilled expectations: "expect values".
Test Suite 'CombineLatestTests' failed at 2020-03-04 20:37:24.957.
Executed 3 tests, with 3 failures (0 unexpected) in 18.159 (18.161) seconds

最佳答案

试试这个测试

 func testCombineLatest_receiveOn() {
for x in 0...100 {
print("---------- RUN \(x)")
let q1 = DispatchQueue(label: "q1", qos: .background)
let q2 = DispatchQueue(label: "q2", qos: .utility, attributes: .concurrent)

let subj1 = PassthroughSubject<Int, Never>()
let subj2 = PassthroughSubject<Int, Never>()

let publ1 = subj1
.map { value -> Int in
print("-- Observer 1: \(value), Thread: \(Thread.current)")
return value
}
let publ2 = subj2
.map { value -> Int in
print("-- Observer 2: \(value), Thread: \(Thread.current)")
return value
}

let exp = expectation(description: "expect values")
exp.assertForOverFulfill = false
// you have to use the same serial queue for publishers which you like to combine
let canc = Publishers.CombineLatest(publ1.receive(on: q1), publ2.receive(on: q1))
// this just redirect it to different queue which could be even concurrent
// (it has no effect at all)
.receive(on: q2)
.sink { value1, value2 in
print("-- recieved \(value1): \(value2) on \(Thread.current)")
if value1 == 10, value2 == 20 {
exp.fulfill()
}
}

let q = DispatchQueue.global()
// the values could be updated concurently
q.async {
subj1.send(5)
}
q.async {
subj2.send(20)
}
q.async {
subj1.send(10)
}


wait(for: [exp], timeout: 10)
canc.cancel()
}
}

运行它,检查打印输出并查看上面片段中的注释

打印输出的一部分
---------- RUN 6
-- Observer 1: 5, Thread: <NSThread: 0x6000020aec40>{number = 6, name = (null)}
-- Observer 1: 10, Thread: <NSThread: 0x6000020aec40>{number = 6, name = (null)}
-- Observer 2: 20, Thread: <NSThread: 0x6000020d1380>{number = 4, name = (null)}
-- recieved 10: 20 on <NSThread: 0x6000020aec40>{number = 6, name = (null)}
---------- RUN 7
-- Observer 1: 5, Thread: <NSThread: 0x6000020aec40>{number = 6, name = (null)}
-- Observer 2: 20, Thread: <NSThread: 0x6000020d1380>{number = 4, name = (null)}
-- Observer 1: 10, Thread: <NSThread: 0x6000020aec00>{number = 7, name = (null)}
-- recieved 5: 20 on <NSThread: 0x6000020aec00>{number = 7, name = (null)}
-- recieved 10: 20 on <NSThread: 0x6000020aec40>{number = 6, name = (null)}

您可以在其中看到并发发送值的效果

关于ios - Swift Combine-多线程上的Publishers.CombineLatest,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60527631/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com