探索一种C++二进制模块的热更新机制

    现如今,提供稳定可靠且能满足人民群众日益增长的物质文化需要的服务是互联网服务商的基本责任,所以服务端软件一定要够壮够强够灵活。服务程序一旦跑起来那就最好7X24小时地永远别挂,而且多变的、不停增长的用户需求也得尽快满足。可问题是,永远也别指望程序员写出没有bug的程序,任何架构师也没有水晶球可以预测将来的花花世界,无论
当时看起来多么完美的代码,将来也会因为种种原因要被修改(或者被丢弃?)。既然如此,那么我们或许应该想办法给程序加上点进化能力,让它能永不停歇地任劳任怨地工作,而同时还不断地反省自己、纠正自己并茁壮成长。

    本文是写给C++程序员的,如果你的工具是Lisp、Erlang、Ruby这样的动态语言,那么因为它们牛B的高级动态特性,你压根就用不着象我们这样,在二进制层干刀口舔血的勾当。

    简单来说,热更新就是程序边运行边更新。有人一定觉得我在故意(象专家那样)装B,把简单的问题搞得异常复杂,因为动态链接库本身就是可以动态加载和卸载的,只要在新的动态库build和部署好后通知程序重新加载一下不就搞定了?

    在这里,我要语重心长地告诉你们:第一,我没有装B,因为不想遭雷劈;第二,这种简单的方案在少数情况下是行得通的,但在大多数情况下却不行,因为实际的程序是代码与数据结构的正确结合,代码几乎总是要操作相应的数据结构。举个例子,A库的create函数创建了数据对象data,foo函数能正确地操作data,然后我们用B库热更新了A库,这样现在的create和foo函数都是B库实现的,而且新的create产生的数据对象与data(二进制布局)格式不同,新的foo也只能正确地操作新的数据对象;假设此后应用程序又需要用(B库的)foo操作由A库创建的data(我们无法避免,因为数据的生存周期是和应用逻辑息息相关的),这个时候严重的错误是不是就极可能发生?所以一个模块被热替换掉后,由它创建的所有数据对象也要跟着进行格式转换,转换为与新版本兼容的(二进制布局)格式。可是这又带来了新问题:如何才能找到所有由旧模块创建的数据对象?我们就象蹩脚数学家一样,把一个肮脏的问题转化成了另外一个肮脏的问题。

    换一种思路,如果在编程时愿意遵循一定的规范(规范是一种约束,但合理的约束却常常能提高总体的自由),而这规范使我们能避开找到所有旧版本数据对象这样的棘手问题,那么就能实现安全的热更新。

    本文建议的规范是采用类似COM、XPCOM这样的组件对象模型:程序由一个个的组件对象组成,每种对象提供了若干功能,外部只能通过对象的接口来使用相应功能。接口通常都用C++抽象基类来构建,从ABI(Application Binary Interface)的角度来看,C++抽象基类最关键的是规定了子类的虚函数表的布局。也可以用其它方法来构建接口机制,但是要保证与C++的虚函数表模型(g++, vc++等主流编译器在这方面的实现都是一样的)在ABI层上兼容。模块是物理上的对象容器,可以包含一个或多个组件对象。模块最常见的形式就是动态链接库(so或dll),本文探索的动态更新机制便是以模块为最小单位。

    从ABI层来看,通过组件对象接口来调用相关功能实际就是调用该对象虚表中对应项所指向的虚函数实现,正因为调用虚函数需要一个查表才能找到真正函数地址的中间操作,所以才使得我们能够hook住组件对象的调用,从而有机会把老版本的对象转换为新版本。那么如何才能
  • 2
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 8
    评论
### 回答1: Qt是一种跨平台的应用程序开发框架,不仅提供了GUI程序开发所需的控件等基本功能,还提供了网络通信等重要的功能模块。其中,UDP(User Datagram Protocol)是一种无连接的传输协议,在网络通信中应用广泛。 Qt提供了丰富的网络编程接口以及UDP相关的类和函数。使用Qt进行UDP通信可以方便快捷地实现网络数据传输功能。在进行UDP通信时,需要将要发送的数据进行编码和解码,这时候就需要使用二进制文件进行数据的传输。 二进制文件是一种二进制形式来存储数据的文件,相比于文本文件更加高效,并且可以直接进行存取操作。在网络通信中,二进制文件能够减少数据传输的大小,提高数据传输的速度。 Qt提供了QByteArray类用于存储和操作二进制数据,在UDP通信中可以利用QByteArray对二进制数据进行编码和解码,便于在网络传输过程中进行数据的处理和存储。 总之,使用Qt进行UDP通信时,二进制文件可以提高数据传输的效率和速度,而QByteArray类可以方便地进行二进制数据的存储和操作。 ### 回答2: Qt是一种跨平台的GUI开发框架,支持C++编程语言,在网络编程中也有广泛应用。UDP是一种无连接的传输协议,可以快速传输数据包,但是数据可靠性较差。 在Qt中使用UDP协议进行网络编程,可以通过QUdpSocket类实现。此类可以用于接收和发送UDP数据报。发送数据时,可以使用writeDatagram()函数将二进制文件转换为QByteArray类型并发送出去。接收数据时,使用readyRead信号和pendingDatagramSize()函数获取数据报的大小,再使用readDatagram()函数读取数据报并转换为二进制文件。 在传输大量二进制文件时,可以使用分包技术,将一个文件分成多个小数据包进行传输,接收端再将多个小数据包合并成完整的文件。另外,在传输时可以对数据进行压缩和加密处理,提高文件传输的安全性和效率。 总之,Qt UDP协议可以用于快速传输二进制文件,具有高效性和跨平台特性,为网络编程提供了良好的支持。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

soloist

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值