Вызов веб-службы с помощью httpMethod (POST) в swift 4.1
Позвольте мне сначала описать контроллер представления (дизайн страницы). Я добавил одну кнопку и одну метку. И функция, созданная мной для кнопки. Вот файл viewController.swift:
import UIKit
class ViewController: UIViewController {
@IBOutlet weak var btnHitWebService: UIButton!
@IBOutlet weak var lblResult: UILabel!
override func viewDidLoad() {
super.viewDidLoad()
}
@IBAction func btnHitWebServiceClick(_ sender: Any) {
if (getTokenWithPostMethod() == true)
{
lblResult.text = "yes, we get token: " + g_token
}
else
{
lblResult.text = "there is an error, please try again later"
}
}
func getTokenWithPostMethod() -> Bool{
var funcResult = false
var request = URLRequest(url: URL(string: "https://apiABCDEFG.com/token")!)
request.httpMethod = "POST"
let postString = "grant_type=password&username=ABCDEF&password=ABCDEF"
request.httpBody = postString.data(using: .utf8)
request.addValue("application/x-www-form-urlencoded", forHTTPHeaderField: "Content-Type")
request.addValue("application/json", forHTTPHeaderField: "Accept")
let task = URLSession.shared.dataTask(with: request)
{
data, response, error in
guard let data = data, error == nil else
{
// check for fundamental networking error
print("error=\(String(describing: error))")
return
}
if let httpStatus = response as? HTTPURLResponse, httpStatus.statusCode != 200
{
// check for http errors
print("statusCode should be 200, but is \(httpStatus.statusCode)")
print("response = \(String(describing: response))")
}
do
{
if let jsonObject = try JSONSerialization.jsonObject(with: data, options: []) as? [String:AnyObject]
{
g_token = (jsonObject["access_token"] as? String)!
print(g_token)
g_token_type = (jsonObject["token_type"] as? String)!
print("\n")
print(g_token_type)
funcResult = true ;
}
}
catch let error as NSError
{
print(error)
}
}
task.resume()
return funcResult
}
}
Вот моя проблема: когда я нажимаю кнопку, это дает мне токен на экране вывода в xcode (print), но метод getTokenWithPostMethod() превращает меня в ложь. Почему это возвращает меня ложь? В процессе отладки: он переходит от 'let task = ' к 'task.resume()'. Спасибо! (кстати, я не могу поделиться правильной ссылкой, именем пользователя и паролем. Я надеюсь, что эти 3 не нужны, чтобы мы могли решить эту проблему)
2 ответа
Вы делаете вызов API. Требуется время, чтобы вернуть значение. Здесь обратный оператор не будет ждать ответа. Вы должны использовать либо замыкание, либо делегирование. Вы можете попробовать данный код:
func getTokenWithPostMethod(handler: @escaping (_ funcResult: Bool) -> Void){
var request = URLRequest(url: URL(string: "https://apiABCDEFG.com/token")!)
request.httpMethod = "POST"
let postString = "grant_type=password&username=ABCDEF&password=ABCDEF"
request.httpBody = postString.data(using: .utf8)
request.addValue("application/x-www-form-urlencoded", forHTTPHeaderField: "Content-Type")
request.addValue("application/json", forHTTPHeaderField: "Accept")
let task = URLSession.shared.dataTask(with: request)
{
data, response, error in
var funcResult = false
if error == nil, let data = data {
do
{
if let jsonObject = try JSONSerialization.jsonObject(with: data, options: []) as? [String:AnyObject]
{
g_token = (jsonObject["access_token"] as? String)!
print(g_token)
g_token_type = (jsonObject["token_type"] as? String)!
print("\n")
print(g_token_type)
funcResult = true
}
}
catch let error as NSError
{
print(error)
}
}
// edited
DispatchQueue.main.async {
handler(funcResult)
}
}
task.resume()
}
Это вернется после получения ответа. Замените вызов этого метода на:
@IBAction func btnHitWebServiceClick(_ sender: Any) {
getTokenWithPostMethod { (flag) in
if flag {
lblResult.text = "yes, we get token: " + g_token
} else {
lblResult.text = "there is an error, please try again later"
}
}
}
Для более подробной информации, пожалуйста, посетите
Он возвращает false, потому что вы установили var funcResult = false при запуске, поэтому, прежде чем URLSession запросит сервер и вернет только некоторые данные в ответ, ваша функция возвращает значение funcResult.
Вы можете добавить приведенный ниже код в тело задачи URLSession, сразу после попытки catch, вместо добавления его в btnHitWebServiceClick:
if (getTokenWithPostMethod() == true)
{
lblResult.text = "yes, we get token: " + g_token
}
else
{
lblResult.text = "there is an error, please try again later"
}