【C】20.链接过程介绍

链接器

每个C语言源文件被编译后都生产目标文件,这些文件最终通过链接器生成最终的可执行程序。

链接器的主要作用是把各个模块之间相互引用的部分处理好,是得各个模块之间能够正确的衔接。

两种链接方式

  • 静态链接
  • 动态链接

静态链接

由链接器在链接时将库的内容直接加入到可执行程序中

Linux 下静态库的创建和使用

  • 编译静态源码
gcc -c lib.c -o lib.o
  • 生成静态库文件
ar -p lib.a lib.o
// ar 表示打包
  • 使用静态库编译
gcc main.c lib.a 

示例:

// main.c
#include <stdio.h>
extern char* name();
extern int add(int a, int b);

int main()
{
    printf("Name: %s\n", name());
    printf("Result: %d\n", add(2, 3));

    return 0;
}

// dlib.c
char* name()
{
    return "Dynamic Lib";
}


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

编译过程

[email protected]:/mnt/hgfs/100ask$ ar -q slib.a slib.o
ar: creating slib.a
[email protected]:/mnt/hgfs/100ask$ gcc 20-1.c slib.a
[email protected]:/mnt/hgfs/100ask$ ./a.out
Name: Static Lib
Result: 5

此时把 slib.c 文件删除也不影响最中的结果

动态连接

  • 可执行程序在运行时才动态加载库进行链接
  • 库的内如不会进入可执行程序当中

  • 编译动态库源码
gcc -shared dlib.c -o dlib.so
  • 使用动态库链接
gcc main.c -ldl 

关键系统调用

  • dlopen: 打开动态库文件
  • dlsym: 查找动态库的函数并返回调用地址
  • dlclose: 关闭动态库文件

#include <stdio.h>
#include <dlfcn.h>

int main()
{
    void* pdlib = dlopen("./dlib.so", RTLD_LAZY);

    char* (*pname)();
    int (*padd)(int, int);

    if( pdlib != NULL )
    {
        pname = dlsym(pdlib, "name");
        padd = dlsym(pdlib, "add");
 
        if( (pname != NULL) && (padd != NULL) )
        {
            printf("Name: %s\n", pname());
            printf("Result: %d\n", padd(2, 3));
        }

        dlclose(pdlib);
    }
    else
    {
        printf("Cannot open lib ...\n");
    }

    return 0;
}

// dlib.c
char* name()
{
    return "Dynamic Lib";
}


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

在本环境下要添加 -fPIC

[email protected]:~$ gcc -fPIC -shared dlib.c -o dlib.so

删除 dlib.so ,运行

Cannot open lib ...

说明动态库是在程序启动后才加载的

实际的意义:

动态链接 在应用程序更新,只需要更新需要部分动态库就行。静态连接适用于小程序里面的可以用的

小结

  • 链接是目标文件最终链接成可执行程序
  • 根据链接过程方式不同,链接过程可以分静态链接和动态链接
发布了84 篇原创文章 · 获赞 0 · 访问量 729

猜你喜欢

转载自blog.csdn.net/zhabin0607/article/details/103723719