解开可选值

时间:2018-09-13 16:10:31

标签: ios json swift facebook alamofire

我正在尝试使用代码登录到Macbook,但我不断收到此错误:

  

线程1:致命错误:在展开可选值时意外发现nil

// API to log an user in
func login(userType: String, completionHandler: @escaping (NSError?) -> Void) {

    let path = "api/social/convert-token/"
    let url = baseURL!.appendingPathComponent(path)
    let params: [String: Any] = [
        "grant_type": "convert_token",
        "client_id": CLIENT_ID,
        "client_secret": CLIENT_SECRET,
        "backend": "facebook",
        "token": FBSDKAccessToken.current().tokenString,
        "user_type": userType
    ]

    Alamofire.request(url!, method: .post, parameters: params, encoding: URLEncoding(), headers: nil).responseJSON { (response) in

        switch response.result {
        case .success(let value):

            let jsonData = JSON(value)

            self.accessToken = jsonData["access_token"].string!
            self.refreshToken = jsonData["refresh_token"].string!
            self.expired = Date().addingTimeInterval(TimeInterval(jsonData["expire_in"].int!))

            completionHandler(nil)
            break

        case .failure(let error):
            completionHandler(error as NSError?)
            break
        }
    }

}

错误指的是这一行:

self.accessToken = jsonData["access_token"].string!

这是LoginViewController代码:

import UIKit
import FBSDKLoginKit

class LoginViewController: UIViewController {
    @IBOutlet weak var bLogin: UIButton!
    @IBOutlet weak var bLogout: UIButton!

    var fbLoginSuccess = false
    var userType: String = USERTYPE_CUSTOMER

    override func viewDidLoad() {
        super.viewDidLoad()

        if (FBSDKAccessToken.current() != nil) {
            bLogout.isHidden = false
            FBManager.getFBUserData(completionHandler: {

                self.bLogin.setTitle("Continue as \(User.currentUser.email!)", for: .normal)
                // self.bLogin.sizeToFit()
            })
        }
    }

    override func viewDidAppear(_ animated: Bool) {
        if (FBSDKAccessToken.current() != nil && fbLoginSuccess == true) {
            performSegue(withIdentifier: "CustomerView", sender: self)
        }
    }

    @IBAction func facebookLogout(_ sender: AnyObject) {
        APIManager.shared.logout { (error) in
            if error == nil {
                FBManager.shared.logOut()
                User.currentUser.resetInfo()

                self.bLogout.isHidden = true
                self.bLogin.setTitle("Login with Facebook", for: .normal)
            }
        }
    }

    @IBAction func facebookLogin(_ sender: AnyObject) {
        if (FBSDKAccessToken.current() != nil) {
            APIManager.shared.login(userType: userType, completionHandler:  { (error) in
                if error == nil {
                    self.fbLoginSuccess = true
                    self.viewDidAppear(true)
                }
            })
        } else {
            FBManager.shared.logIn(
                withReadPermissions: ["public_profile", "email"],
                from: self,
                handler: { (result, error) in
                    if (error == nil) {
                        FBManager.getFBUserData(completionHandler: {
                            APIManager.shared.login(userType: self.userType, completionHandler:  { (error) in
                                if error == nil {
                                    self.fbLoginSuccess = true
                                    self.viewDidAppear(true)
                                }
                            })
                        })
                    }
            })
        }
    }
}

雨燕3 Xcode 9 iOS 10.2

我阅读了几本书,以找出导致此类错误的原因,但未能成功。

2 个答案:

答案 0 :(得分:2)

首先,在各处都使用强制展开(!)是一个不好的做法。尽量避免使用它,直到您真正需要它和/或知道您在做什么。在这里,您使用SwiftyJSON。它可以帮助您方便地从JSON提取数据。但是,您不应该依赖于始终从后端获取正确的JSON。有很多原因导致它可以返回错误的JSON,并且不需要值。有两种选择:

  1. 您可以使用.stringValue代替.string-它会返回一个空字符串,而不是nil
  2. 您可以通过以下方式进行操作:if let token = jsonData["access_token"].string {...}甚至使用guard语句

这里有一篇很好的文章,了解了力量展开:https://blog.timac.org/2017/0628-swift-banning-force-unwrapping-optionals/

答案 1 :(得分:1)

当您使用!强制展开符号)时,会发生此错误,这意味着您确定数据将在那里;但实际上,数据不存在-nil

尝试使用guard语句。例如:

guard let self.accessToken = jsonData["access_token"].string else {
    // log error message, if desired; then exit the function
    return
}

或者,您可以使用if let语句:

if let self.accessToken = jsonData["access_token"].string {
    // do stuff if 'jsonData["access_token"].string' is valid
}
else {
    // do other stuff if 'jsonData["access_token"].string' is nil
}

现在,为什么不存在数据(nil)-这是另一个问题。也许检查您的JSON函数以确保其正确处理JSON响应。另外,请检查以确保收到有效的回复:

guard let statusCode = (response as? HTTPURLResponse)?.statusCode, statusCode == 200 else {
    // handle a bad response code
    return
}

// handle the good response code stuff after the guard statement

Basics 中的 Optional Binding 下的《 Swift编程语言(Swift 4.2)指南》中了解如何使用if let处理可能的nil值。 / em>部分。

在指南的控制流部分和 {{3中,在 Early Exit 下了解guard语句}} 部分。