展开可选值

问题描述 投票:1回答:2

我试图使用代码登录macbook但我一直收到此错误:

线程1:致命错误:在解开Optional值时意外发现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)
                                }
                            })
                        })
                    }
            })
        }
    }
}

Swift 3 Xcode 9 iOS 10.2

我阅读了几篇文章,找出造成这类错误的原因,但没有成功。

ios json swift facebook alamofire
2个回答
2
投票

首先,在任何地方使用force unwrap(!)是一种不好的做法。尽量避免它,直到你真的需要它和/或知道你在做什么。在这里你使用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
投票

当你有一个!(强制解包符号)时会发生这个错误,这意味着你确定数据会在那里;但事实上,数据不存在 - 它是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

了解如何使用if let在基础知识部分的nil下的Swift编程语言(Swift 4.2)指南中处理可能的Optional Binding值。

在本指南的“控制流”部分以及参考文献的guard部分中了解Early Exit下的Statements语句。

© www.soinside.com 2019 - 2024. All rights reserved.