ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

动态库和静态库

2022-04-23 11:03:04  阅读:189  来源: 互联网

标签:文件 gcc %. 静态 编译 output 动态 我们


 

文章目录

 

时间

Access:文件最近被访问的时间,
Modify: 最近一次修改文件内容的时间
Change:最近一次修改文件属性的时间

Change time
在这里插入图片描述
修改文件内容的时候,有可能也会修改文件的属性
比如: 可能会更改文件的大小属性
在这里插入图片描述
我们会发现操作下来, 文件时间貌似没有变化:
在较新的Linux内核中,Accsee中不会被立即刷新更新,而是有一定的时间间隔进行刷新

makefile更新的机制

makefile and gcc 会根据时间问题,来判定源文件和可执行文件次序谁更新,为就以谁为主
如gcc test.c -o test
如果test的modify时间比test.c的modify时间新,说明就已经更新了,反之,如果test.c的时间比test的时间新,说明makefile需要更新

在这里插入图片描述
在这里插入图片描述
对于这种伪目标的,他总是被执行,因为他不关心时间
在这里插入图片描述
touch会更新文件的时间,我们重新make也可以
在这里插入图片描述

静态库和动态库

查看可执行程序依赖库

ldd 【可执行文件】
在这里插入图片描述

第二个是c标准库,我们查看了一下,发现是软链接,
我们发现这个文件也是有inode
在这里插入图片描述

linux当中,一般库分为两种: 动态库和静态库(库也是一个文件)
在Linux中,如果是库文件:库文件一般是以.so作为后缀的,如果是静态库,一般是以.a为后缀的

  1. 库文件的命名:
    ::libXXX.so- or libYYY.a-
  2. 库的真正名字
    去掉包含lib前缀,去掉包含a-,so-的后缀
    如/lib64/libc.so.1真实名字就是c
    /lib64/libc-2.17.so真实名字就是c-2.17

在这里插入图片描述
c++标准库

也可以使用file查看连接库的情况
在这里插入图片描述
动态库只有一个,删除的话,其他文件都无法使用了

如何制作库

库本身就是一个二进制文件,
一套完整的库:

  1. 库文件本身
  2. 头文件.h(文本,会说明库中暴露出来的方法的基本使用!)
    我们在C/C++ 中,为什么在写代码的时候,有时候.h里面放上声明,在.c里面放上定义,为什么要这么设计

因为我们要制作库!

  1. 方便使用 2. 私密,比较安全,用库对头文件进行封装保护,

  • 我们在other目录下面写一个add.cpp add.h这两个,一个声明,一个定义
  • 如果我们想要使用的话,要加上这个库的路径
    在这里插入图片描述
  • 编译的时候
    在这里插入图片描述
    添加上特定的路径中的定义,才能使用

如果没有库,我们想要编译要怎么编译呢

 mytest:%.o
      gcc -o $@ $^
 %.o:%.c
      gcc -c $<
 
  • 1
  • 2
  • 3
  • 4

,o是我们汇编之后生成的
所以%.o 依赖所有的%.c
$<就是把上面的所有东西展开,一个一个的进行编译

  1 obj=mytest.o add.o sub.o
  2 
  3 mytest:$(obj)
  4     gcc -o $@ $^
  5 %.o:%.c
  6     gcc -c $<
  7 
  8 %.o:./other/%.c
  9     gcc -c $<
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

先形成3个.o文件,最后在链接形成可执行

如果我们想把自己的代码给别人用

  1. 提供源代码+头文件
  2. 只提供.o文件也可以链接,所以库的思想:就是把所有的.o打包就是库了

打包成一个库

  1. 所有的源代码要先被编译成为.0文件(可重定向文件)
  2. 制作动静态库的本质就是将所有.0打包好,使用ar 或者gcc来打包
  3. 交付: inlcude + .a or .so文件

ar (类比tar)
-rc(replace and creat)

libmymath.a:sub.o add.o
		ar -rc $@ $^  //把所有的.o文件都打包成一个静态库
%.o:%.c
		gcc -c $<   //把所有的.c都展开成.o文件
 
  • 1
  • 2
  • 3
  • 4
  • 5

查看静态库中的内容
在这里插入图片描述

我们可以查看c标准静态库中的内容都是.o
这样我们就把我们自己的源文件都隐藏起来
别人想要用我们的库,就还需要一些头文件,因为二进制根本看不懂
所以我们可以发布一个output

PHONY:output
output:
		mkdir output 
		cp -rf *.h output 
		cp libmymath.a output
 
  • 1
  • 2
  • 3
  • 4
  • 5

在这里插入图片描述
头文件和库文件都在

使用自己的库

#include"add.h"
#include"sub.h"

int main()
{
    int x=10;
    int y=2;
    int y1=add(x,y);
    printf("y1=%d" ,y1);
    return 0;
}
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
$ gcc demo.c -I./output -L./output -lmymath
 
  • 1

这样编译才可以
-I 指明,我们要使用知名头文件搜索路径
-L 指明:我们要指明库文件的搜索路径
-l 指明:我们要链接那一个库

我们之前写的代码,也用了库,为什么就没有指明选项呢??
之前的库,在系统的默认路径下,/lib64,/user/lib,/user/include等
编译器是能够识别这些路径的,

换一句话说我们如果想不添加这些选项,我是不是可以把对应的库和头文件,拷贝到默认的路径之下
没有问题的,这也就是一般软件的安装过程

我们给别人交付的其实就是一个库文件+ 一套头文件

形成动态库

libmymath.so:add.o sub.o
	gcc -shared -o $@ $^ 
%.o:%.c
	gcc -fPIC -c $<

.PHONY:clean
clean:
	rm -f libmymath.so
.PHONY:lib 
lib:
	mkdir lib
	cp *.h lib
	cp libmymath.so lib
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14

在这里插入图片描述
编译使用的方法和静态库一样的
我们在加载的时候,还要进一步高树系统,我们的库在哪里,因为动态库我们编译的时候需要,不编译的时候就不需要,不像静态库是一次性直接拷贝进去,在编译的时候就不需要去考虑库在哪里

编译

$ gcc x.c  -o x -I./lib -L./lib -lmymath
 
  • 1

导入环境变量

export LD_LIBRARY_PATH=/home/xvzewen/githubgovern/Linuxstudy/link/friend/lib
 
  • 1
  • 2

执行
./x
为什么我们之前写所有的代码都没有报错呢??

默认情况下我们都是动态的,因为我们一定有动态库!,因为我们linux很多都是用c语言写的

转自:https://www.dianjilingqu.com/

标签:文件,gcc,%.,静态,编译,output,动态,我们
来源: https://www.cnblogs.com/yuanyuzhou/p/16181680.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有