ios - 来自 HTTPS URLSession.dataTask 的 0 字节数据
问题描述
我正在尝试从 URL 中检索一些 JSON 数据,但是它每次在数据中返回 0 个字节。我检查了几个不同的 SO 帖子,但似乎没有一个与我遇到的确切问题相匹配,因为我从中获取数据的服务器确实使用 HTTPS 和 TLS 1.2。所以我不认为有什么需要添加到我的 info.plist 中,尽管我可能是错的。
这是我的代码:
var tournaments: [Tournament] = []
/// Base API URL
let baseURL: String = "https://www.burningmeter.com/tournaments.json?page=1"
// ...
func retrieveAPIData() {
// Build the API string
var request: URLRequest = URLRequest(url: URL(string: baseURL)!)
request.cachePolicy = URLRequest.CachePolicy.reloadIgnoringLocalCacheData
request.addValue("application/json; charset=utf-8", forHTTPHeaderField: "Content-Type")
request.httpMethod = "GET"
// Request the data
let session: URLSession = URLSession.shared
let task = session.dataTask(with: request) { (data, response, error) in
// Did we get an error?
guard error == nil else {
print(error!)
return
}
guard let json = data else {
print("No data")
return
}
guard json.count == 0 else {
print("Zero bytes of data")
return
}
let jsonDecoder = JSONDecoder()
let tournaments = try! jsonDecoder.decode(TournamentPage.self, from: json)
// We got our values, let's go
self.tournaments = tournaments.tournaments
}
task.resume()
}
赛事结构:
struct Tournament : Codable {
let id: Int
let name: String
let game_id: Int
let game_iteration_id: Int
let state: Int
let starts_at: String
let creator_id: Int
let stream_url: String
let entrant_count: Int
let prereg_count: Int
let path: String
}
struct TournamentPage : Codable {
let page: Int?
let results_per_page: String
let tournament_count: Int
let tournaments: [Tournament]
}
解决方案
guard
你的陈述有误。你写==
而不是!=
in json.count == 0
。它应该像下面的代码一样:
guard json.count != 0 else {
print("Zero bytes of data")
return
}
使用以下代码(已移除编码器)在操场上进行了测试:
let baseURL: String = "https://www.burningmeter.com/tournaments.json?page=1"
func retrieveAPIData() {
var request: URLRequest = URLRequest(url: URL(string: baseURL)!)
request.cachePolicy = URLRequest.CachePolicy.reloadIgnoringLocalCacheData
request.addValue("application/json; charset=utf-8", forHTTPHeaderField: "Content-Type")
request.httpMethod = "GET"
// Request the data
let session: URLSession = URLSession.shared
let task = session.dataTask(with: request) { (data, response, error) in
print(data!.count)
print(response)
// Did we get an error?
guard error == nil else {
print(error!)
return
}
guard let json = data else {
print("No data")
return
}
guard json.count != 0 else {
print("Zero bytes of data")
return
}
print(String(decoding: json, as: UTF8.self))
}
task.resume()
}
retrieveAPIData()
打印结果:
8031
Optional(<NSHTTPURLResponse: 0x7fbce1650fc0> { URL: https://www.burningmeter.com/tournaments.json?page=1 } { Status Code: 200, Headers {
"Access-Control-Allow-Origin" = (
"*"
);
"Cache-Control" = (
"max-age=0, private, must-revalidate"
);
Connection = (
"keep-alive"
);
"Content-Type" = (
"application/json; charset=utf-8"
);
Date = (
"Thu, 08 Nov 2018 15:40:40 GMT"
);
Etag = (
"W/\"5d5a17c8ff6f705d6bb56c6ed8b6a099\""
);
Server = (
Cowboy
);
"Strict-Transport-Security" = (
"max-age=31536000"
);
"Transfer-Encoding" = (
Identity
);
Via = (
"1.1 vegur"
);
"X-Content-Type-Options" = (
nosniff
);
"X-Frame-Options" = (
SAMEORIGIN
);
"X-Request-Id" = (
"0d7f8ae9-92d2-41d8-8678-6c9b2bdb5e3b"
);
"X-Runtime" = (
"0.044064"
);
"X-Xss-Protection" = (
"1; mode=block"
);
} })
JSON结果:
{
"page": "1",
"results_per_page": "50",
"tournament_count": 33,
"tournaments": [{
"id": 1872,
"name": "Pinup Biweekly - Nov 1",
"game_id": 90,
"game_iteration_id": 133,
"state": 30,
"starts_at": "2018-11-02T00:00:00.000Z",
"creator_id": 960,
"stream_url": null,
"entrant_count": 6,
"prereg_count": 0,
"path": "/t/1872/pinup-biweekly-nov-1"
}, {
"id": 1874,
"name": "IBTY #45",
"game_id": 21,
"game_iteration_id": 38,
"state": 30,
"starts_at": "2018-11-02T22:03:00.000Z",
"creator_id": 185,
"stream_url": null,
"entrant_count": 11,
"prereg_count": 0,
"path": "/t/1874/ibty-45"
}, {
"id": 1875,
"name": "SFV Weekly 11/2/18",
"game_id": 6,
"game_iteration_id": 14,
"state": 30,
"starts_at": "2018-11-03T00:55:00.000Z",
"creator_id": 957,
"stream_url": null,
"entrant_count": 9,
"prereg_count": 0,
"path": "/t/1875/sfv-weekly-11-2-18"
}, {
"id": 1876,
"name": "Weekly SC6 11/2/18",
"game_id": 106,
"game_iteration_id": 149,
"state": 20,
"starts_at": "2018-11-03T01:04:00.000Z",
"creator_id": 957,
"stream_url": null,
"entrant_count": 6,
"prereg_count": 0,
"path": "/t/1876/weekly-sc6-11-2-18"
}, {
"id": 1879,
"name": "UNIST - WS 11/3",
"game_id": 12,
"game_iteration_id": 23,
"state": 30,
"starts_at": "2018-11-03T18:30:00.000Z",
"creator_id": 949,
"stream_url": null,
"entrant_count": 8,
"prereg_count": 0,
"path": "/t/1879/unist-ws-11-3"
}, {
"id": 1881,
"name": "DBFZ - WS 11/3",
"game_id": 86,
"game_iteration_id": 129,
"state": 30,
"starts_at": "2018-11-03T18:30:00.000Z",
"creator_id": 949,
"stream_url": null,
"entrant_count": 8,
"prereg_count": 0,
"path": "/t/1881/dbfz-ws-11-3"
}, {
"id": 1882,
"name": "SFV - WS 11/3",
"game_id": 6,
"game_iteration_id": 14,
// more
}
// More output here
}
接收长度为 8031
推荐阅读
- java - 如何使用 Maven 处理在两个目录之间拆分的 Java 包
- amazon-ec2 - 来自 EKS 节点的 ERR_CONNECTION_RESET
- python - @tf.function 正在减慢训练步骤
- javascript - Ajax 在另一个 Ajax 调用期间调用以接收服务器的任务计算状态并将其作为进度条显示给客户端
- excel - 插入照片循环:路径为空时如何跳到“下一个”?
- ios - 如何在 Sign In With Apple 中验证 realUserStatus 值
- reactjs - 如何解决 Apollo 订阅“WebSocket 握手期间出错:意外响应代码:400”
- elasticsearch - elasticsearch中的“令牌过滤器”和“摄取节点”有什么区别?
- sql - SQL Listagg 语法
- sql - Power BI - 如果文本列不符合条件,是否可以有条件地格式化它?