Estou usando o padrão MVVM para fazer chamadas de serviço
código: no viewcontroller postGenericCallVM
a chamada e a resposta também estão chegando, mas print("testing viewmodel data in viewcontroller \(String(describing: postVM.postDataVM?.postCount))")
nula, por quê?
onde estou errado?
struct RequestObject {
var params: [String: Any]? = nil
var method: HTTPMethod
var urlPath: String
var isTokenNeeded: Bool
var isLoaderNeed: Bool = false
var isErrorNeed: Bool = false
var vc: UIViewController?
}
class APINetworkManager2: NSObject {
static let sharedInstance = APINetworkManager2()
fileprivate override init() {
super.init()
}
func serviceCall<T: Decodable>(requestObject: RequestObject, completion: @escaping (Result<APIResponse<T>, Error>) -> Void) {
guard let url = URL(string: requestObject.urlPath) else {
return
}
var urlRequest = URLRequest(url: url)
if requestObject.isTokenNeeded {
let token = "Bearer token" // Replace with your actual token
urlRequest.setValue(token, forHTTPHeaderField: "Authorization")
}
urlRequest.httpMethod = requestObject.method.rawValue
if let params = requestObject.params {
let paramsReq: [String: Any] = ["jsonrpc": "2.0", "params": params]
do {
urlRequest.httpBody = try JSONSerialization.data(withJSONObject: paramsReq)
} catch {
completion(.failure(error))
return
}
}
let task = URLSession.shared.dataTask(with: urlRequest) { data, _, error in
if let data, let string = String(data: data, encoding: .utf8) {
print("debug check \(string)")
}
do {
let response = try JSONDecoder().decode(APIResponse<T>.self, from: data!)
DispatchQueue.main.async {
completion(.success(response))
}
} catch {
DispatchQueue.main.async {
completion(.failure(error))
}
}
}
task.resume()
}
}
e este é o meu código VIewmodel aqui print o/p
result view model: model data coming
class PostViewModel {
var postDataVM: PostGenResult?
func postGenericCallVM(){
let param = ["page_no" : "1"] as? [String:Any]
let requestObject = RequestObject(
params: param,
method: .post,
urlPath: "https://hsfjkhskdn",
isTokenNeeded: true,
isLoaderNeed: true,
isErrorNeed: true
)
APINetworkManager2.sharedInstance.serviceCall(requestObject: requestObject) { (result: Result<APIResponse<PostGenResult>, Error>) in
switch result {
case .success(let response):
self.postDataVM = response.result as? PostGenResult
print("result view model: \(self.postDataVM)")
case .failure(let error):
print("Error decoding JSON: \(error)")
}
}
}
}
o/p: obtendo resposta com sucesso no viewcontroller
modelo de visualização de resultado: Opcional (PracticeCode.PostGenResult (pageCount: 1, perPage: 10, postCount: 5, userReminder: [PracticeCode.UserReminder (id: 17, receiverID: 7, type: "Assinatura", mensagem: "Sua assinatura está vencida em 2
e este é o meu código do viewcontroller: aqui
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
postVM.postGenericCallVM()
print("testing viewmodel data in viewcontroller \(postVM.postDataVM?.postCount)")
}
}
a chamada de serviço está ligando e a resposta está chegando, mas testing viewmodel data in viewcontroller
não dá nada
testando dados do viewmodel no viewcontroller: nil
É verdade, porque você está solicitando uma
async
tarefa epostVM.postGenericCallVM()
levará algum tempo para retornar a chamada.Por outro lado, esses códigos serão executados em sincronia. Para mais detalhes, você pode colocar um log aqui.
Então você verá que a ordem de impressão deve ser:
Como lidar? Abordagem 1, adicione a conclusão @escaping na
postGenericCallVM
função.Abordagem 2, use
async await