其他分享
首页 > 其他分享> > ascs 简明开发教程(二十二):可靠的UDP传输

ascs 简明开发教程(二十二):可靠的UDP传输

作者:互联网

QQ交流群:198941541

可靠UDP基于KCP(https://github.com/skywind3000/kcp),用户需要熟悉KCP的窗口大小,重传模式,流控等,ascs还是原来的的原则,不封装KCP,用户通过ikcpcb* get_kcpcb() 得到ikcpcb做任意想要的配置,当然应该在数据收发之前(构造函数里面和reset虚函数里面,后者会在socket被重用之后马上被调用)。

reliable_socket_base继承自原来的 UDP socket_base,所以如果你不创建KCP(create_kcpcb),可靠UDP会退化成普通的UDP,参考udp_test。

使用可靠UDP需要用户提供一个回调函数,签名为:int output(const char* buf, int len, ikcpcb * kcp, void* user),不支持仿函数(因为KCP是纯C的),由于ascs是header only的,我很难去提供这么一个函数而不造成重复定义,如果定义成inline或者static,那么编译得到的所有.o文件里面都将有一份实现,即使不造成重复定义也不好,如果大家有更好的办法,欢迎群里面讨论。用户在这个回调函数里面直接调用reliable_socket_base的output函数即可,无需考虑多线程安全的问题,参考udp_test。这个回调函数通过KCP函数ikcp_setoutput送给ikcpcb(如果没有这一步,编译无法检测但程序会core dump),ikcpcb通过reliable_socket_base::get_kcpcb() 得到,当然这之前你必须已经调用过reliable_socket_base::create_kcpcb 创建了ikcpcb了。

可靠UDP,需要用宏ASCS_RELIABLE_UDP_NSND_QUE或者set_max_nsnd_que设置一个发送队列大小(消息条数)上限,这是因为KCP的缓存大小是无限的,所以我们必需要在必要时限速,当KCP的发送队列消息数量大于这个值之后,ascs将暂停向其输送(ikcp_send)新的消息,此时消息会停在ascs的发送缓存里面,而ascs的发送缓存也是有限制,这样最终将造成用户调用send_msg失败。

KCP也没有解决的问题——连接重置:

在决定用KCP之前,我也想过自己开发一个可靠UDP,但就是这个连接重置难倒了我,后来在群里朋友的推荐下,采用了KCP。不管谁来实现可靠UDP,总是会有缓存来解决消息丢失,消息重复,消息乱序等问题,所以缓存是很复杂的,那如果通信双方一方重启了,接收方可能会永远接收不到某个消息,那么这个消息之后的消息,即便是收到了,也不能派发出来(否则消息乱序),那么对于这种无法恢复的问题(或者设定时间内无法解决的问题),我们需要重置连接,意思是通信双方从这一刻开始重新来过,过往的所有错误既往不咎,可惜这种自动连接重置很难实现,只能在业务层上来做。我推荐的做法是,再建一条TCP连接,专门用于通知这种连接重置,一方出问题恢复之后,通过TCP告诉对方,并协商一个新的conv(与之前的不一样,创建ikcpcb时用的,参考reliable_socket_base::create_kcpcb),然后双方开始用新的conv创建新的ikcpcb,这样可靠UDP又可以工作了。

注意你不能随时调用reliable_socket_base::create_kcpcb,必须先关闭socket(force_shutdown等),对于single_reliable_socket_service,如果你想重用它,等到after_close被调用之后,你就可以调用reliable_socket_base::create_kcpcb创建新的ikcpcb了(用新的conv),然后调用start(或者service_pump带i_service的start_service)再次启用它,其实最简单的办法还是创建一个新的single_reliable_socket_service;对于multi_reliable_socket_service,你调用其add_socket得到socket之后(虽然叫add,其实可能的话也是会重用的,并不一定是add,如果是重用,object_pool会先调用socket的reset,再返回socket,然后add_socket再返回给你),就可以调用reliable_socket_base::create_kcpcb创建新的ikcpcb了(用新的conv)。

上一篇 ascs 简明开发教程(21)

标签:UDP,socket,reliable,ikcpcb,教程,ascs,base,KCP
来源: https://blog.csdn.net/yang79tao/article/details/120101124