热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

码农交流日常——我只能说:MDZZ(手动微笑)

本文属吐槽文,我只想说:对神论还蜜汁自信的怎么破?背景介绍上半年参与了一个水的不行的项目,总的架构如下:用C++作为实现语言在Windows下,用户界面以EXE的形式出现,其他模块

本文属吐槽文,我只想说:对神论还蜜汁自信的怎么破?

------------------------------背景介绍---------------------------------------

上半年参与了一个水的不行的项目,总的架构如下:

技术分享

用C++作为实现语言

在Windows下,用户界面以EXE的形式出现,其他模块以DLL的形式通过LoadLibrary加载

数据总控模块提供一套API给“自定义功能模块”使用,实现用户界面和功能模块的交互,自然,所有的自定义功能模块也都按照约定的接口实现,以便和数据总控模块的统一交互

分工好了以后,每个人负责各自的模块进行开发,我负责其中的若干个自定义功能模块。

很简单是吧?但是跟我写的标题有毛关系?对,的确暂时看不出来有毛关系,但是,我为了把我标题所指的事说清楚,这些是必须提到的背景材料,而且,我还得先引用一下:

https://msdn.microsoft.com/en-us/library/windows/desktop/ms681914(v=vs.85).aspx

Types of Dynamic Linking

There are two methods for calling a function in a DLL:

    • In load-time dynamic linking, a module makes explicit calls to exported DLL functions as if they were local functions. This requires you to link the module with the import library for the DLL that contains the functions. An import library supplies the system with the information needed to load the DLL and locate the exported DLL functions when the application is loaded.
    • In run-time dynamic linking, a module uses the LoadLibrary or LoadLibraryEx function to load the DLL at run time. After the DLL is loaded, the module calls theGetProcAddress function to get the addresses of the exported DLL functions. The module calls the exported DLL functions using the function pointers returned byGetProcAddress. This eliminates the need for an import library.

简单说,上面一大堆意思就是,你要加载DLL有两种办法:

1. 编译阶段:通过给链接器提供你想加载的DLL对应的import library(这玩意儿是.LIB文件),这样在运行的时候系统会帮你自动加载你想要的DLL,你就不用写代码手动加载

2. 运行阶段:通过代码里调用LoadLibrary进行手动加载

任何一个上过幼儿园的码农都应该知道,C++的“编译阶段”和“运行阶段”是他妈的两个八杆子打不到一起的不同阶段吧?

------------------------------MDZZ---------------------------------------

最近各种测试,然后,在前几天,组里负责“数据总控模块”的人按照需求变更对该模块进行了变更(简称变更X),增加了新的接口(对于我负责的模块来说根本没影响,因为我的模块用不着新增的接口,所以只需要更新一下头文件,重新编译一下就可以,我连代码都不用改,所以我更新了一下头文件编译过了之后就直接git push了)

变更之后,又进行测试

今天组里负责测试的某人给我发个消息

TA:变更X之后,你的模块崩溃了

我:不会吧,上次提交测试不是好好的吗,变更X又跟我负责的模块没有直接关系,不就是增加了一个API么,我又没用到,我的模块连代码和输入数据都不用改啊,怎么可能是我的模块引起的崩溃?

【备注:然后我调试了一下,看了下出错:std::bad_alloc,再看了下调用堆栈,显示崩溃的地方在“数据总控模块”内部,因为我的模块没有做任何变更,包括输入的数据也没有任何变更,所以我当时估摸着问题应该跟“数据总控模块”的变更有关】

TA:反正崩了,其他人负责的自定义功能模块也有同样的崩溃情况

我:你看这调用堆栈,明明是在总控模块里面崩的,这太明显了吧,那我debug我自己的模块能解决什么问题,肯定要让总控模块那人去debug总控模块啊

【备注:我发了个截图给TA看调用堆栈】

技术分享

TA:但是他们其他的功能模块的崩溃问题都解决了,你也解决一下

我:他们怎么解决的?问题在什么地方?

TA:你把总控模块的DLL更新一下啊

【备注:我这里就纳闷了,我TM更不更新总控模块的DLL跟我的模块有没有BUG有鸡毛关系?难道总控模块有法轮大法?】

我:我当然更新了的啊,我刚才测试之前就更新了的啊,你自己看我git pull截图

技术分享

【备注:好了,准备好亮瞎眼

技术分享

【备注:我这里就看晕了,什么叫我的模块里有没有更新?我突然感觉到事态严峻(我要跟TA解释不清楚了):TA认为数据总控模块是“嵌入”到我的模块里才能运行的,然而实际上,这个项目里所有的模块都是相对独立的,并不存在谁嵌入谁的概念,只不过是数据在模块之间传递而已,所以,我就干脆顺着TA的问题问下去,要怎么更新?然后请看】

技术分享

【备注:前面已经提到过,编译的时候如果提供了import library(是一个LIB文件)的话,也可以的,然而,我们这个项目都是用LoadLibrary,根本没有用import library】

【备注:我很无奈,所以干脆就顺着TA的信仰继续问下去:】

技术分享

【备注:TA给我回个截图,我估计意思就是:你他妈更新这个DLL啊,嵌入到你的项目编译啊(然而,前面已经说过,我他妈早就更新了,而且,更不更新这玩意儿跟这个BUG没有鸡毛关系啊,大神!)】

技术分享

技术分享

【备注:我无奈,继续顺着TA的神论说下去,然后TA继续神论,我顿时无语,你TM确定我编译的时候要加载别的模块的DLL?你是说我编译的时候?你是说我编译的时候?你是说我编译的时候?你是说我编译的时候?】

技术分享

【备注:我感觉欲哭无泪,准备挣扎着给TA解释一下这个原理,顺便附上一张图怕自己说得不够清楚】

技术分享

技术分享

【备注:我继续顺着TA的神论说下去,然后TA给我来句这个,你们自己感受下神论大法的威力,TA居然认为直接把另一个模块的DLL扔到我的项目文件夹里就算TA信仰里的“嵌入”了!?你他妈不是在逗我?扔到文件夹中就行?扔到文件夹中就行?】

技术分享

【备注:然后我给TA展示了,即便按TA说的去改也没有卵用,结果TA直接来一句:那别人这么弄就解决了,你怎么解释?-我只能说:你家神论,大法好!我等凡人,参悟不了!】

技术分享

【备注:为了确认TA的意思,我重述了一遍:你确定是把那个模块的DLL放到我的模块的工程目录下就行?-TA蜜汁自信:是的】

技术分享

--------------------------------------------------------后记------------------------------------------------------

最后又折腾一会,BUG的原因找到了,头文件不同步造成的(这人也是神,明知道别人怎么解决的,他妈早说啊!我一开始就问了别人怎么解决的,别人怎么解决的,别人怎么解决的,绕那么远搞个鸡毛,呵呵)(某个头文件中间经过了若干次改动)(C++的pure virtual function的在头文件中的顺序必须一致,如果顺序不同也会导致调用出错,就是这个原因,懒得细说了,很简单的问题)

码农交流日常——我只能说:MDZZ(手动微笑)


推荐阅读
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • 本文讨论了如何优化解决hdu 1003 java题目的动态规划方法,通过分析加法规则和最大和的性质,提出了一种优化的思路。具体方法是,当从1加到n为负时,即sum(1,n)sum(n,s),可以继续加法计算。同时,还考虑了两种特殊情况:都是负数的情况和有0的情况。最后,通过使用Scanner类来获取输入数据。 ... [详细]
  • 本文介绍了C#中数据集DataSet对象的使用及相关方法详解,包括DataSet对象的概述、与数据关系对象的互联、Rows集合和Columns集合的组成,以及DataSet对象常用的方法之一——Merge方法的使用。通过本文的阅读,读者可以了解到DataSet对象在C#中的重要性和使用方法。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • CentOS 7部署KVM虚拟化环境之一架构介绍
    本文介绍了CentOS 7部署KVM虚拟化环境的架构,详细解释了虚拟化技术的概念和原理,包括全虚拟化和半虚拟化。同时介绍了虚拟机的概念和虚拟化软件的作用。 ... [详细]
  • 本文介绍了通过ABAP开发往外网发邮件的需求,并提供了配置和代码整理的资料。其中包括了配置SAP邮件服务器的步骤和ABAP写发送邮件代码的过程。通过RZ10配置参数和icm/server_port_1的设定,可以实现向Sap User和外部邮件发送邮件的功能。希望对需要的开发人员有帮助。摘要长度:184字。 ... [详细]
  • 本文介绍了指针的概念以及在函数调用时使用指针作为参数的情况。指针存放的是变量的地址,通过指针可以修改指针所指的变量的值。然而,如果想要修改指针的指向,就需要使用指针的引用。文章还通过一个简单的示例代码解释了指针的引用的使用方法,并思考了在修改指针的指向后,取指针的输出结果。 ... [详细]
  • 在project.properties添加#Projecttarget.targetandroid-19android.library.reference.1..Sliding ... [详细]
  • 猜字母游戏
    猜字母游戏猜字母游戏——设计数据结构猜字母游戏——设计程序结构猜字母游戏——实现字母生成方法猜字母游戏——实现字母检测方法猜字母游戏——实现主方法1猜字母游戏——设计数据结构1.1 ... [详细]
  • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
  • 本文介绍了一种解析GRE报文长度的方法,通过分析GRE报文头中的标志位来计算报文长度。具体实现步骤包括获取GRE报文头指针、提取标志位、计算报文长度等。该方法可以帮助用户准确地获取GRE报文的长度信息。 ... [详细]
  • PDF内容编辑的两种小方法,你知道怎么操作吗?
    本文介绍了两种PDF内容编辑的方法:迅捷PDF编辑器和Adobe Acrobat DC。使用迅捷PDF编辑器,用户可以通过选择需要更改的文字内容并设置字体形式、大小和颜色来编辑PDF文件。而使用Adobe Acrobat DC,则可以通过在软件中点击编辑来编辑PDF文件。PDF文件的编辑可以帮助办公人员进行文件内容的修改和定制。 ... [详细]
  • CentOS 6.5安装VMware Tools及共享文件夹显示问题解决方法
    本文介绍了在CentOS 6.5上安装VMware Tools及解决共享文件夹显示问题的方法。包括清空CD/DVD使用的ISO镜像文件、创建挂载目录、改变光驱设备的读写权限等步骤。最后给出了拷贝解压VMware Tools的操作。 ... [详细]
  • 深入理解CSS中的margin属性及其应用场景
    本文主要介绍了CSS中的margin属性及其应用场景,包括垂直外边距合并、padding的使用时机、行内替换元素与费替换元素的区别、margin的基线、盒子的物理大小、显示大小、逻辑大小等知识点。通过深入理解这些概念,读者可以更好地掌握margin的用法和原理。同时,文中提供了一些相关的文档和规范供读者参考。 ... [详细]
author-avatar
安安1
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有