[英]Swift Type Inference with Generic method
我正在开发一个 SDK,并开发了一个非常简洁的 Combine 管道方法,该方法接受用于解码 json 的通用参数。 实际上,它是 JSON -> Decodable
的可重用组合管道。 效果很好。 这是管道的样子:
func records<Record: Decodable>(forRequest request:RestRequest ) -> AnyPublisher<[Record], Never> {
return NetworkService.publisher(for: request)
.tryMap({ (response) -> Data in
response.asData()
})
.decode(type: Wrapper<Record>.self, decoder: JSONDecoder())
.map({ (record) -> [Record] in
record.records
})
.catch({ _ in
Just([Record]())
})
.eraseToAnyPublisher()
}
用法:
contactsCancellable = NetworkService.records(forRequest: request)
.receive(on: RunLoop.main)
.assign(to: \.contacts, on: self)
我的理解是 Swift+Combine 是从assign(to:, on:)
调用推断泛型参数类型。
但是想要一个非组合版本的权力,我真的很努力想弄清楚如何帮助 Swift 推断类型。 我尝试构建一个像这样的直接模拟:
func fetchRecords<Record: Decodable>(forRequest request: RestRequest,
_ completionBlock: @escaping (Result<[Record], RestClientError>) -> Void) {
RestClient.shared.send(request: request) { result in
switch result {
case .success(let response):
do {
let decoder = JSONDecoder()
let wrapper = try decoder.decode(Wrapper<Record>.self, from: response.asData())
completionBlock(.success(wrapper.records))
} catch {
completionBlock(.success([Record]()))
}
case .failure(let err):
completionBlock(.failure(err))
}
}
}
然而,这会编译,像这样执行该方法:
NetworkService.fetchRecords(forRequest: request) { records in
print(records)
}
导致一个非常神秘的错误无法推断通用参数“记录”
在此非组合版本中,如何指定通用 Record 'type' - 任何符合 Decodable 的内容?
Ps:这是包装结构:
struct Wrapper<R: Decodable>: Decodable {
var totalSize: Int
var done: Bool
var records: [R]
}
您可以在闭包参数列表中指定泛型类型:
NetworkService.fetchRecords(forRequest: request) { (result: Result<[ConcreteRecordType], RestClientError>) {
switch result {
case .success(let records):
// "records" is of type [ConcreteRecordType]
//...
case .failure(let error):
//...
}
}
但是在闭包中提供完整的Result
类型可能很麻烦,所以我建议您通过接受它作为参数来填写泛型类型信息。 (就像Decoder
功能一样。)
func fetchRecords<Record: Decodable>(ofType type: Record.Type, forRequest request: RestRequest, _ completionBlock: @escaping (Result<[Record], RestClientError>) -> Void) {
//... same code...
}
然后,你会这样称呼它:
NetworkService.fetchRecords(ofType: ConcreteRecordType.self, forRequest: request) { result in
// No need to specify closure argument type :)
switch result {
case .success(let records):
// "records" is of type [ConcreteRecordType]
//...
case .failure(let error):
//...
}
}
瞧! 提供给fetchRecords
的显式类型级联到闭包参数类型。 无需在闭包参数列表中提供类型。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.