c/c++ 静态库的链接原理和使用

静态库的制作和使用本质

1 静态库文件.a文件本质是处理其中的每一个.o文件

2 ar -cr 的本质就是将指定的.o文件放到.a文件中


EG :

main.cpp

#include <stdio.h>
int main () {
    printf ("main\n");
    return 0;
}

test.cpp

#include <stdio.h>
class test{
    public:
        test() {
            printf ("test ctor\n");
        }
};
static test s_test;

lib.cpp

#include <stdio.h>
class lib {
    public:
        lib() {
            printf ("lib ctor\n");
        }
};
static lib s_lib;

接下来在命令行中,进行操作

1 : 生成所有必要的.o文件<如果为了方便,可以写在makefile中>, 为了看的比较直观

g++ -o test.o test.cpp

g++ -o lib.o lib.cpp

g++ -o main main.cpp

2 : 生成可执行文件

g++ -o main test.o lib.o main.o

输出内容 :

test ctor

lib ctor

main

原理解释 :

静态变量 static test s_test, static lib s_lib他们的内存申请在链接的时候都会早于main的执行,虽然main中并没有引用任何test和lib的接口或者数据:

为了方便测试,我们写一个makefile吧

cc = g++
target = main
objs = main.o test.o
srcs = main.cpp test.cpp
lib = libfn.a
libso = libfn.o lib.o
all : cleano cleana $(lib) $(target)
$( lib ) : $( libso )
    ar -cr $@ $^
libfn.o : libfn.cpp
    cc -o libfn.o -c libfn.cpp
lib.o : lib.cpp
    cc -o lib.o -c lib.cpp
$( target ) : $( objs )
     $( cc ) -o $@ $( objs ) -lfn -L.
main.o : main.cpp   
     $( cc ) -o main.o -c main.cpp
test.o : test.cpp   
     $( cc ) -o test.o -c test.cpp
    
.PHONY : clean
clean : cleano cleana   
cleano :
    rm -rf *.o
cleana :
    rm -rf *.a

接下来将lib.o以.a的形式表示,那么用法就会有变化,为了方便表示,添加脚本libfn.cpp

int sum(int a, int b) {
    return a + b;
}

生成.a文件,将libfn.o和lib.o 打入到.a中

ar -cr libfn.a libfn.o lib.o

g++ -o main main.o test.o -lfn -L.

修改main.cpp文件

#include <stdio.h>
int main () {
    extern int sum (int a, int b);
    sum(2, 3);
    printf ("main\n");
    return 0;
}
运行 :

./main

test ctor

main

5

可以看出并没有返回任何和lib.o相关的信息,那么结论是lib.o虽然被打入了.a中<执行 ar -t libfn.a 查看内部组成>. 但并没有被前面的.o文件链接引用, 那么在系统认为这个lib.o文件并没有用,那么就给他直接忽略了。


接下来让我们更改一下我们的生成main可以执行文件的执行顺序

g++ -o main test.o -lfn main.o  -L.

运行./main

g++ -o main test.o -lfn main.o -L.
main.o: In function `main':
main.cpp:(.text+0x13): undefined reference to `sum(int, int)'
collect2: error: ld returned 1 exit status
makefile:19: recipe for target 'main' failed
make: *** [main] Error 1

会提示sum没有找到. 问题的关键就是在在链接libfn.a的时候,前面的.o文件没有人去用到自己的任何接口或者变量,那么直接讲libfn.a内部的.o文件进行了忽略。


参考文章 : https://blog.csdn.net/jiange_zh/article/details/52195604

代码路径 : https://gitee.com/DomTech/Cmake/tree/master/cmake_tutorial/m5library/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值