详解Linux下静态库/动态库的生成和使用(含代码示例和操作流程)&&动态库和静态库的区别

关于gcc的使用方法可以参考下方链接博客:
Linux下详解gcc编译过程(含代码示例)&& gcc使用教程

一、库的概念

  • 库是一种组件技术。
  • 库里封装了数据和函数,提供给用户程序调用。
  • 库只执行到第三阶段编译,没有链接。
  • 库的使用可以使程序模块化,提高程序的编译速度,实现代码复用。

Windows的库

Windows系统本身提供并使用了大量的库;
包括:静态库(.lib)和动态链接库(.dll)

Linux的库

· Linux系统通常把库文件存放在/usr/lib或/lib目录下,
· Linux库文件名组成:前缀lib + 库名 + 后缀(3部分组成)
————————————————————————————
动态库:以.so作为后缀
静态库:通常以.a.la作为后缀。


二、动态库和静态库的区别

1. 载入顺序不同

静态库的代码在编译时就拷贝到应用程序中,因此当多个应用程序同时引用一个静态库函数时,内存中将会调用函数的多个副本。其优点是节省编译时间。
动态库是在程序开始运行后且调用库函数时才被载入,被调函数在内存中只有一个副本,并且动态库可以在程序运行期间释放动态库所占用的内存。

2. 大小与共享的差异

静态链接库就是在程序编译的时候就被加载进来,这样的可执行文件会比较大一些,还不能共享 ;动态链接库是在程序执行的时候加载,可共享 。

3. 库函数调用的差异

静态链接是指把要调用的函数或者过程链接到可执行文件中,成为可执行文件的一部分。
动态链接所调用的函数代码并没有被拷贝到应用程序的可执行文件中去,而是仅仅在其中加入了所调用函数的描述信息(往往是一些重定位信息)。仅当应用程序被装入内存开始运行时,在linux的管理下,才在应用程序与相应的.so之间建立链接关系。当要执行所调用动态库中的函数时,根据链接产生的重定位信息,才转去执行动态库中相应的函数代码。


三、静态库的创建和使用

1. 创建静态库的步骤:

  1. 在一个头文件中声明静态库所导出的函数
  2. 在一个源文件中实现静态库所导出的函数
  3. 编译源文件,生成可执行代码(没有连接)
  4. 将可执行代码所在目标文件加入到某个静态库中,并将静态库拷贝到系统默认的存放库目录下(/usr/lib或/lib)。

(一)ar命令的使用方法

  • 作用:创建和修改静态库
  • 格式:ar [选项] 静态库名.a 目标文件1.o 目标文件2.o …

ar命令可用选项:

选项说明
d从库中删除成员
p在终端上打印库中指定的成员
r在库中加入新成员文件,如果要加入的成员文件存在,则替换之;默认情况下,新的成员文件增加在库的结尾处
t显示库的成员文件清单
a在库的一个已经存在的成员后面增加一个新的成员文件
c创建一个库
i在库的一个已经存在的成员前面增加一个新的成员文件
s无论ar命令是否修改了库内容,都强制重新生成库符号表
x从库中提取一个成员文件。如果不指定要提取的成员文件则提取库中所有的文件

(二)生成静态库的实例

(1)头文件mylib.h —— 声明静态库所导出的函数

#ifndef  _mylib_H_     //如果没有定义此标识符,编码以下程序
#define  _mylib_H_    1 
  void welcome();
  void outstring (const char * str);
#endif

(2) 对应于头文件的源文件mylib.c —— 实现静态库所导出的函数

#include “mylib.h”
#include <stdio.h>
void welcome() {
     printf (“welcome to libmylib\n”);
}
void outstring(const char * str) {
     if (str !=NULL)
         printf (%s”,str);
}

(3)编译mylib.c生成目标文件:
# gcc –o mylib.o –c mylib.c
注: -c是只编译而不生成可执行文件,-o指定文件名

更多关于gcc的使用方法可以参考下方链接博客:
Linux下详解gcc编译过程(含代码示例)&& gcc使用教程

(4)将目标文件加入到静态库中,静态库为libmylib.a
#ar rcs libmylib.a mylib.o

参数说明:
/* r —在库中加入新成员文件,如果要加入的成员文件存在,则替换之默认情况下,新的成员文件增加在库的结尾处
/* c 创建一个库
/* s 无论ar命令是否修改了库内容,都强制重新生成库符号表

(5)将静态库拷贝到 linux 的库目录(/usr/lib或/lib)下
#cp libmylib.a /usr/lib/libmylib.a

到这里静态库就生成完了,下面我们尝试用一个程序链接它。

2. 调用静态库的测试程序

简而言之就是:用一个.c源文件在编译时做静态链接

(一)调用静态库的测试程序:test.c

#include “mylib.h”
#include <stdio.h>
int  main()
   {
     printf (“create and use library:\n”);
     welcome();  //静态库中的函数原型
     outstring(“It’s successful\n”);  //静态库中的函数原型
}

(二)编译使用了库函数的程序

#gcc –o test test.c –lmylib

参数说明:
(1)选项l为链接选项说明在链接时就将静态库链接至源程序中
(2)在Linux中约定所有库都以前缀lib开始,静态库以.a结尾,动态库以.so结尾,在编译时,无需带上前缀(lib)和后缀(.a)。


四、动态库的创建和使用

注:动态库没有归档,在编译过程中就要生成符号表

  • 在Linux环境下,只要在编译函数库源程序时加上
    -fPIC -shared选项即可生成动态链接库。

1. 动态库的创建

# gcc –fPIC -o mylib.o -c mylib.c (生成符号表)
# gcc –shared -o libmylibs.so mylib.o(防止重复加载)

· 或者可以直接使用一条命令
# gcc –fPIC –shared –o libmylibs.so mylib.c (可以省略 -c 和 .o后缀文件名)

· 将生成的静态库拷贝至/usr/lib或/lib目录中
# cp libmylibs.so /usr/lib/libmylibs.so

2. 动态库的使用

(1)通过gcc命令调用
# gcc –o test test.c -lmylibs

(2)通过调用系统函数来使用动态链接库
在这里插入图片描述

其中:
dlopen函数的参数flag可取的值有:RTLD_LAZY、 RTLD_NOW、 RTLD_GLOBAL:

  1. RTLD_LAZY:在dlopen()返回前,对于动态库中存在的未定义的变量不执行解析,即不解析这个变量的地址
  2. RTLD_NOW:在dlopen返回前,解析出每个未定义变量的地址,如果解析不出来,dlopen会返回NULL,错误为“Undefined symbol:”
  3. RTLD_GLOBAL:使库中被解析出来的变量在随后的其它链接库中也可以使用,即全局有效

3. 通过调用系统函数来使用动态链接库的实例

创建testso.c文件:

#include <stdioh>
#include <dlfcn.h>
 int main (void)
{
  void *handle;
  char *error;
  void (*welcome)(); //要变成指针函数来用(由符号表引出的指针)
if ((handle = dlopen(/usr/lib/libmylibs.so”, RTLD_LAZY) == NULL)
  {
    printf (“dlopen error\n”);
    exit(1);
  }
welcome = dlsym(handle, “welcome”);
  if ((error = dlerror()) != NULL)
   {
     printf (“dlsym error \n”);
     exit(1);
   }
  welcome ();
  dlclose (handle);
  exit(0);
}

编译: #gcc –o testso testso.c -ldl
其中:-ldl 指明dlopen等函数所在的库类型

  • 27
    点赞
  • 66
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Java动态代理是一种通过在运行时期间生成代理对象来实现对目标对象进行代理的技术。它可以在不修改目标对象的情况下,为目标对象提供额外的功能。 Java动态代理实现的核心是利用了Java的反射机制和动态生成类的技术。在动态代理中,我们需要定义一个代理类和一个实现了InvocationHandler接口的处理器类。 代理类是在运行时动态生成的类,它是目标对象的代理,它实现了与目标对象相同的接口,并且在方法调用时会通过InvocationHandler接口的实现类来处理方法的调用。InvocationHandler接口中只有一个方法invoke(Object proxy, Method method, Object[] args),这个方法就是用来处理方法调用的。 下面是一个简单的Java动态代理的示例代码: ``` import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; public class DynamicProxy implements InvocationHandler { private Object target; public DynamicProxy(Object target) { this.target = target; } public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("before method"); Object result = method.invoke(target, args); System.out.println("after method"); return result; } public static void main(String[] args) { RealObject realObject = new RealObject(); DynamicProxy dynamicProxy = new DynamicProxy(realObject); Interface proxyObject = (Interface) Proxy.newProxyInstance( Interface.class.getClassLoader(), new Class[] { Interface.class }, dynamicProxy); proxyObject.doSomething(); } } interface Interface { void doSomething(); } class RealObject implements Interface { public void doSomething() { System.out.println("RealObject doSomething"); } } ``` 在这个示例中,我们定义了一个DynamicProxy类作为代理处理器,它实现了InvocationHandler接口。在DynamicProxy类中,我们定义了一个Object类型的target属性,它表示目标对象。 在DynamicProxy类的invoke方法中,我们先输出了一句话“before method”,然后通过反射机制调用目标对象的方法,最后输出了一句话“after method”。 在DynamicProxy类的main方法中,我们首先创建了一个RealObject对象作为目标对象,然后创建了一个DynamicProxy对象,并将RealObject对象作为参数传递给DynamicProxy对象的构造方法。接着,我们通过Proxy.newProxyInstance方法动态生成了一个代理对象,并将DynamicProxy对象作为参数传递给它。最后,我们调用代理对象的doSomething方法。 当我们运行这个程序时,它会输出以下内容: ``` before method RealObject doSomething after method ``` 这表明,在代理对象调用doSomething方法时,它会先调用DynamicProxy类的invoke方法,在invoke方法中,我们将先输出一句话“before method”,然后调用目标对象的方法,最后输出一句话“after method”。 Java动态代理的优点是可以在运行时期间动态生成代理对象,不需要预先定义代理类,这样可以大大减少代码量。同时,Java动态代理也具有很好的灵活性,可以对不同的目标对象生成不同的代理对象,实现不同的处理逻辑。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

狱典司

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值