编译动态库时,可以指定 SONAME 。程序运行时,加载器(Dynamic Loader)发现动态库有 DT_SONAME 字段,则会转而去链接 DT_SONAME 指定的动态库。
Linux 动态库的版本号一般格式为 lib + libname + .so + 主版本号 + 子版本号 + 发行版本号 。
情况一:如果版本号发生了改变之后,实际库仍然是兼容的,此时更新软链接,将主版本,即可实现库的更新。
情况二:如果版本号发生了改变之后(一般是主版本),库不再兼容,此时变更 SONAME ,即可实现库的多版本兼容(多个库同时存在,由程序根据 SONAME 选择加载)。
例子:
// add.c // V1.0.0 int add(int a, int b) { return a+b; }
// main.c #include<stdio.h> int add(int a, int b); int main() { int a = 1; int b = 2; int c = add(a, b); printf ("c=%d\n", c); return 0; }
对第一种情况:
版本更新前:
gcc add.c -shared -fPIC -Wl,-soname=libadd.so.1 -o libadd.so.1.0.0 ldconfig -nv . ln -s libadd.so.1 libadd.so gcc main.c -L. -ladd export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH ./a.out
观察动态库和 a.out 的符号:
readelf -d libadd.so.1.0.0 Dynamic section at offset 0x2e80 contains 18 entries: Tag Type Name/Value 0x000000000000000e (SONAME) Library soname: [libadd.so.1] readelf -d a.out Dynamic section at offset 0x2db0 contains 28 entries: Tag Type Name/Value 0x0000000000000001 (NEEDED) Shared library: [libadd.so.1]
也即,编译库的时候,生成的库的名字为最完整的名字(比如 libadd.so.1.0.0),库的 SONAME 字段为兼容版本的库名字(比如 libadd.so.1)。生成 a.out 时,链接的名字始终使用 -ladd ,也即需要创建一个 libadd.so 的链接指向兼容版本的库。最终生成 a.out 时,依赖的库的名字是根据 libadd.so 中的 SONAME 字段生成的(SONAME 字段存在时),也即 a.out 依赖的库名字为 libadd.so.1。
注意:libadd.so 只有编译生成 a.out 时需要使用(DEV)。如果只是部署,则只需要 libadd.so.1 和 libadd.so.1.0.0 就行了。
版本更新后(更新后仍然兼容,只更新子版本号):
rm libadd.1.* gcc add.c -shared -fPIC -Wl,-soname=libadd.so.1 -o libadd.so.1.1.0 ldconfig -nv .
版本更新时,先删除当前已有的兼容版本,然后生成新的兼容版本,并利用 ldconfig 更新兼容版本链接。
ldconfig -nv . .: libadd.so.1 -> libadd.so.1.1.0 (changed)
兼容版本更新过程中,需要删除已有版本。
对第二种情况:
// add.c // V1.0.0 int add(int a, int b, int c) { return a+b+c; }
// main.c #include<stdio.h> int add(int a, int b, int c); int main() { int a = 1; int b = 2; int c = add(a, b, b); printf ("c=%d\n", c); return 0; }
如果出现了库的不兼容版本更新,则两个库可以兼容存在。旧的程序链接旧版本的库,新的程序链接新版本的库,彼此互不干扰。
gcc add.c -shared -fPIC -Wl,-soname=libadd.so.2 -o libadd.so.2.0.0 ldconfig -nv . ln -s libadd.so.2 libadd2.so gcc main.c -L. -ladd2 -o a2.out export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH ./a2.out
这里我们把用于编译的库的名字做了区分(libadd.so 和 libadd2.so),这时为了编译的时候进行版本区分。
这样就实现了多个库的兼容。