2

我有一个可解码的类:

struct AuthenticationResponse : Decodable {
var status: String
var error: Error
var access_token: String? = ""
var expires_in: Double? = 0
var token_type: String? = ""
var scope: String? = ""
var refresh_token: String? = " 
}

struct Error : Decodable {
var desc: String
var code: String
}

在错误类中,我有:

为了解码到这个类,我有:

 URLSession.shared.dataTask(with: request) { (data:Data?, response:URLResponse?, error:Error?) in
            if let jsonData = data{
                let decoder = JSONDecoder()
                print("hey")
                print("response: \(String(data:jsonData, encoding:.utf8))")
                completion(try! decoder.decode(AuthenticationResponse.self, from: jsonData))
            }
    }.resume()

由于我收到的一些回复是(成功回复):

{
“status”: “SUCCESS”  “error”: null,  "access_token":
"MWVmOWQxMDYwMjQyNDQ4NzQyNTdkZjQ3NmI4YmVjMGZjZGM5N2IyZmNkOTA1 N2M0NDUzODEwYjM5ZWQyNGNkZg",
"expires_in": 3600, "token_type": "bearer", "scope": null, 
"refresh_token":
"ZGEwOGZiOWZhMzhhYjBmMzAyOGRmZTA5NjJhMjY2MTk3YzMyMmE1ZDlkNWI2N mJjYmIxMjNkMjE1NWFhNWY0Mg"
}

然后一个失败的响应只包含一个错误对象,其中包含 desc 和代码。

我想要实现的是一个适用于两种场景的可解码类(当响应成功和失败时)但是我不知道如何实现这一点。我知道我可以制作 2 个单独的可解码类,但这会使事情变得更加混乱,因为我必须确定响应是否是错误并填充以返回不同的类。

有谁知道我应该如何实现这个>

4

1 回答 1

1

我会给它一个try,但首先我们需要理清我认为有点粗制滥造的问题。由于Error是 a 的名称(著名且广泛使用)protocol,因此应该重命名它,并且由于您希望能够将其保留为空,AuthenticationResponse因此它显然必须是可选的(提出了为什么它在其中的问题Response,但是我将把它放在一边)。这给我们留下了以下内容:

struct AuthError : Decodable {
    var desc: String
    var code: String
}

struct AuthenticationResponse : Decodable {
    var status: String
    var error: AuthError?
    var access_token: String? = ""
    var expires_in: Double? = 0
    var token_type: String? = ""
    var scope: String? = ""
    var refresh_token: String? = ""
}

然后我们需要一些相关案例的示例数据,我使用了:

let okData = """
    {
    "status": "SUCCESS",
    "error": null,
    "access_token":
    "MWVmOWQxMDYwMjQyNDQ4NzQyNTdkZjQ3NmI4YmVjMGZjZGM5N2IyZmNkOTA1N2M0NDUzODEwYjM5ZWQyNGNkZg",
    "expires_in": 3600,
    "token_type": "bearer",
    "scope": null,
    "refresh_token":
    "ZGEwOGZiOWZhMzhhYjBmMzAyOGRmZTA5NjJhMjY2MTk3YzMyMmE1ZDlkNWI2NmJjYmIxMjNkMjE1NWFhNWY0Mg"
    }
    """.data(using: .utf8)!

let errData = """
    {
        "desc": "username or password incorrect",
        "code": "404"
    }
    """.data(using: .utf8)!

现在我们可以定义一个enum允许所有情况的返回类型:

enum AuthResult {
    case ok(response: AuthenticationResponse)
    case authError(error: AuthError)
    case parseError(description: String)
    case fatal
}

这最终允许我们parse为接收到的身份验证数据编写函数:

func parse(_ jsonData:Data) -> AuthResult {
   let decoder = JSONDecoder()
    do {
        let authRes = try decoder.decode(AuthenticationResponse.self, from: jsonData)
        return .ok(response: authRes)
    } catch {
        do {
            let errRes = try decoder.decode(AuthError.self, from: jsonData)
            return .authError(error: errRes)
        } catch let errDecode {
            return .parseError(description: errDecode.localizedDescription)
        }
    }
}

游乐场中的所有这些都将允许使用

switch parse(okData) {
case let .ok(response):
    print(response)
case let .authError(error):
    print(error)
case let .parseError(description):
    print("You threw some garbage at me and I was only able to \(description)")
default:
    print("don't know what to do here")
}

与您在大多数其他语言中造成的混乱相比,这仍然很优雅,但是调用仍然没有解决,只是定义AuthenticationResponse为函数的(常规)返回类型并通过ing some 来parse提供其余部分没有意义(符合)和一些合适的有效载荷。throwenumError

(主要)来自 Java,我仍然避免将异常用作“某种程度的”常规控制流(如在“常规”登录失败中),但鉴于 Swift 对异常采取更合理的方法,这可能需要重新考虑。

无论如何,这为您提供了一个功能来解析您的服务回复的任何一种情况,以及一种以“统一”方式处理它们的体面方式。由于您可能无法修改处理您的服务的行为,request这可能是唯一可行的选择。但是,如果您能够修改服务,您应该争取一个“统一”的回复,该回复可以通过一次调用来解析JSONDecoder.decode。您仍然需要解释选项(正如您在上面的示例中应该做的那样,因为使用它们仍然很痛苦,即使考虑到 Swift 出色的编译器支持迫使您“做正确的事情”),但它会使您的解析不易出错。

于 2017-12-20T10:13:11.833 回答