依赖管理


在Unix的设计哲学中,do one thing 被广大软件设计开发人员奉为圭臬,很多底层的基础代码只需要做成库,就可一劳永逸重复使用。但由于软件的升级,很多采用了包发布的方式,虽然方便了开发者免受“晨后综合症”的困扰,却也带来了依赖地狱这个问题。本文试图阐述库开发过程中的问题以及应对事项。


菱形依赖

菱形依赖(diamond denpendency)是指当主模块所依赖的两个库引用了同一库的不同版本的情况。如图所示:

diamond dependency.png

A模块使用了B和C,但B依赖D的version-2,C依赖D的version-1. 这种情况下,很多依赖管理工具就会直接(或提示你)将D的版本统一到version-2链接到最终的可执行文件中。但实际上,这是存在风险的。

分散式编译:lib.a的头文件陷阱

为了简化表达,我们将模块B省略,直接让A依赖D的version-2,并在最终的仲裁机制中选择了D的version-2. C原本依赖D的version-1,现在改为依赖version-2. 如图。

castrate-B.png

这样会有什么问题呢?

C/C++编译步骤

我们知道,C/C++的编译需要4步:

compiling process.png

第一步预处理则是针对头文件中的#define 或者 #include等。假设D中某个常量,比如SHOW_ME_THE_MONEY 从100升级到1000. 那么libc.a中所使用的就是100,但A中被编译进去的则是1000. 原理其实很简单,正是因为编译发生在两个时间,仅仅通过发布头文件和.a文件的方式,导致时间和空间的耦合。
如果不相信,我们稍后见代码

链接的困惑

更难以接受的是某些功能不兼容的修改。比如下图这种情况:

link_compatible.png

本来X期望A库中的freturn 0的实现,但依赖仲裁将其升级到return 1
干说了这么多,你可能不相信,我们上代码:

/*base.h version-1*/
#ifndef BASE_H
#define BASE_H
const int SHOW_ME_THE_MONEY = 100;
#define I_AM_BLIND "But not d"
struct Base{
    int id;
    Base(int id):id(id){}
    Base(const Base & other):id(other.id){}
    void foo();
};

#endif
/*base.cpp version-1*/
#include <iostream>
#include "base.h"

void Base::foo(){
    std::cout << "old base, id:" << id << std::endl;
}

这里的base相当与我们最底层的库,它在一次升级中增加了我的MONEY和类中的一个flag。

/*base.h, upgrade to version-2, more money, add a flag member*/
#ifndef BASE_H
#define BASE_H
const int SHOW_ME_THE_MONEY = 1000;
#define I_AM_BLIND "But not deaf"
struct Base{
    int id;
    bool flag;
    Base(int id, bool flag):id(id),flag(flag){}
    Base(const Base & other):id(other.id),flag(other.flag){}
    void foo();
};

#endif

/***new version: base.cpp****/
#include <iostream>
#include "base.h"

void Base::foo(){
     std::cout << "i'm new foo,  id:" << id << " flag: " << flag << std::endl;
     flag = false;
}

我们的另外一个依赖库x正在依赖version-1的base:

/*libx.h*/
#ifndef LIBX_H
#define LIBX_H
#include "base.h"
void call_libx();
#endif
----------------cpp below------------
/*libx.cpp : remeber , base version-1 in use..*/
#include <iostream>
#include "base.h"
void call_libx(){
    std::cout << "my money in lib: " 
        << SHOW_ME_THE_MONEY 
        << " DH: " << I_AM_BLIND 
        << std::endl;
   Base b(110);
   b.foo();
}

接下来是我们的主模块,他同时依赖x和base:

#include <iostream>
#include "base.h"
#include "libx.h"

int main(int argc, char* argv[]){
    std::cout << "my money in app:"
        << SHOW_ME_THE_MONEY
        << " DH:" <<  I_AM_BLIND
        << std::endl;
    call_libx();
    return 0;
}

我们将新旧base.h,base.cpp都编译成libbase.a,然后将libx也编译成.a,供主模块编译链接,运行后的结果和前文所说一致:

result.png

常量在不同的模块有不同的值,flag字段则出现随机值。(这很危险哦)

如何应对

一般来讲,如果你只负责上图中A模块或者APP的代码,并无库的权限,也许只能向库作者提交ISSUE单了。不过假设我们负责整个架构的代码,如何应对呢?

源码依赖

对于分散式编译的问题,只要在我们的代码中废弃.a这种方式,就能将编译时刻统一到当前。

后向兼容和末端依赖

对于库开发者,除非历史包袱特别的重,应后向兼容所有版本号小于自己的版本。类似Python,最新的2.7可以运行2.5或者2.6的代码.
而对应使用者,则应永远依赖最新的发布。在依赖管理工具上,提供依赖最新的这种抽象依赖手段。

更新

git ci --amend -m "thanks to 微笑的鱼Lilian"

谢谢微笑的鱼Lilian, 提出了一个为什么libx中使用了Base(int)的构造函数仍然可以链接通过的问题。
原因是第一个版本的base.h的构造函数是inline在头文件中的,当编译libx.cpp的时候,#include将其展开在了libx.cpp中导致的。
这让我警醒到,原来不同版本的inline函数,和常量、宏一样,容易发生分散式编译的问题。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,874评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,102评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,676评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,911评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,937评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,935评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,860评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,660评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,113评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,363评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,506评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,238评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,861评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,486评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,674评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,513评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,426评论 2 352

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,649评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,050评论 25 707
  • 依赖管理是Gradle的一个亮点。在最好的情况下,你只需要在构建文件中添加一行代码,Gradle就可以从远程仓库下...
    sollian阅读 5,127评论 0 3
  • 過去的早已煙消雲散, 現在的他依然與眾不同。 但又如何, 那人已是故人。 假若再見, 不會淚流滿面。 人已不是當時...
    十幾桃花阅读 149评论 0 0