當我們安裝了共享庫並運行了 ldconfig
之後,通常我們會在 lib/
裏看到三個 .so
文件。這裏的 so 是 shared object 之意。
libfoo.so.1.0.0
這是普通的(regular)共享庫文件。可能有多個版本的該文件並存。
libfoo.so -> libfoo.so.1.0.0
這個符號鏈接一般用在程序的鏈接過程(link)。你向 linker 指定的參數 -lfoo
便是在尋找並鏈接 libfoo.so
這個文件。通常情況下這總是個指向具體動態庫的符號鏈接,因爲有可能你需要在不同版本的庫間切換。
libfoo.so.1 -> libfoo.so.1.0.0
這個符號鏈接一般用在程序的運行時。每個 .so
都會有一個 SONAME。應用程序根據 SONAME 來尋找並加載共享庫。libfoo.so.1
就是一個 SONAME。你可以通過 objdump -p libfoo.so.1.0.0
或 readelf -d libfoo.so.1.0.0
查看它。對於共享庫的編寫者來說,SONAME 相同的庫應互相兼容。
有些包管理器(如 debian)會使用不同的包名來表示不同版本的共享庫,因此你可以獨立地安裝、卸載、使用特定的共享庫版本。
ldconfig
可以根據 libfoo.so.1.0.0
這樣的共享庫文件生成相應的符號鏈接文件。
你可能會遇到下述情況,這會導致崩潰:
- app 使用了共享庫
liba.so.2
和libb.so.1
libb.so.1
使用了共享庫liba.so.1
可以看到,程序同時加載了不同版本的 liba.so
。