我正在使用socket.io客户端进行swift。当我声明并连接套接字并在AppDelegate中使用它的命令时,一切正常。但是,只要我将这些东西从AppDelegate移动到另一个类(使socket成为全局变量)并从AppDelegate调用该类的函数,就会出现套接字错误(错误的描述与此主题无关。我有一个关于Socket.io问题的讨论) 我想问的问题是这件事发生了吗?我用过
deinit {
NSLog("CCNOTIFICATIONS RESOURCES GOT DE INITIALIZED")
}
在该文件中,但永远不会被调用。
根据我的知识,AppDelegate是一个委托,当应用程序的状态发生变化时会被调用,即它的启动或转到后台/前台。而且,在AppDelegate中混乱大量代码并不是一个好习惯。 显然这两个文件的范围没有区别,为什么会发生这种情况呢?
如果我理解错了,请纠正我。 如何解决这个错误问题?或者至少我可以尝试解决这个问题的其他方法? 任何帮助,将不胜感激。
更新 这是代码: 当我在AppDelegate中编写它时:
class AppDelegate: UIResponder, UIApplicationDelegate {
var window: UIWindow?
public static var socket = SocketIOClient(socketURL: URL(string: “URL”)!, config: [.log(true), .compress])
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {
AppDelegate.socket.connect()
AppDelegate.socket.on("connect") {data, ack in
print("Connected successfully")
}
AppDelegate.socket.on("book-adventure-provider") {data, ack in
print("BOOKING CAME")
print(data)
}
return true
}
当我把它写在其他文件中时:
CCNotifications:
public class CCNotifications : INotifications
{
public var socket = SocketIOClient(socketURL: URL(string: “URL”)!, config: [.log(true), .compress, .forcePolling(true), .forceNew(true)])
public func sendBookingRequest(adventureId: String, success: @escaping () -> Void, error: @escaping CErrorCallback) {
if (ConnectionStatus.isConnected() == false)
{
let errorObj = CError()
errorObj.message = "No internet connection"
errorObj.code = ClientConstants.connectionErrorCode
error(errorObj)
}
else
{
let jwtToken = KeychainWrapper.standard.string(forKey: "jwtToken")
let data = ["adventure_id": adventureId, "jwt": jwtToken]
socket.emit("book-adventure", data)
socket.on("book-adventure-seeker") { data, ack in
print(data)
}
}
}
}
CuufyClient:
public class CuufyClient : IClient {
public var notifications: INotifications? = nil
public static var Notifications: INotifications? { get { return CuufyClient.client?.notifications}}
//region Actions
public static func Initialize(completion: ((_ error: CError?) -> Void)?) {
CuufyClient.client?.notifications = CCNotifications ()
completion? (nil)
}
//endregion
}
最后将其称为:
CuufyClient.Notifications?.funcWhichIWannaCall.
注意: 两种方式都有效。但是当我在CCNotifications中编写代码并运行它时,有时会开始给出套接字错误:会话ID未知,然后一次又一次地连接自己。
更新2: 日志中的套接字错误之前的一行我发现了这个错误:
2017-12-15 15:45:34.133480+0500 TestTarget[5332:230404] TIC Read Status [1:0x60000017f440]: 1:57
在搜索此错误后,我知道在Xcode 9中,此错误显示TCP连接已关闭。当我在socket.io分支上打开这个问题时,专家说:
这个库在底层以上的相当高的层次上运行 联网。它依赖于URLSession或Starscream来告诉它 当连接丢失时。
任何人都可以帮助我关于URLSession,为什么TCP会被关闭,因为我对iOS很新。
答案 0 :(得分:0)
首先要说的是,保持AppDelegate
瘦的方法是个好主意。
关于deinit
:当对象不再存在时,将调用此函数。由于全局变量将“永久”生命 - 只要它们被分配给不同的对象 - 它们将仅在程序终止时被清除(不是:后台模式等)。因此,您将永远不会看到deinit调用。
你对生命周期的看法是正确的,如果你已经正确地完成了所有事情,应该没有区别 - 我在这里无法判断,因为你没有显示任何代码如何以及何时创建/初始化全局套接字等。
关于更新的问题:不幸的是,我不知道socket.io库。但是,只需简短一看,您的代码在AppDelegate
和其他版本之间略有不同。这可能是一个时间问题 - 只需检查(并修复)一些事情:
connect()
?...on()
之前设置所有emit
- 处理程序;否则服务器可能在所有设置正确之前做出响应...on()
- 处理程序,而不是每sendBookingRequest
- 调用jwtToken
?它是您的应用程序的API令牌,还是某种类型的标记令牌?您何时存储该值,为什么它存储在钥匙串中?在请求/回复期间,它的价值可能会发生变化吗?答案 1 :(得分:0)
您不应该在sendBookingRequest
方法中放置套接字事件侦听器。
您只需要绑定一次套接字事件侦听器。
socket.on("book-adventure-seeker") { data, ack in
print(data)
}
因此,您可以将其置于connect
事件中。
AppDelegate.socket.on("connect") {data, ack in
print("Connected successfully")
}