Linux编译多个不同目录下的文件以及静态库、动态库的使用

简介: 先看两篇博文,作为基础知识。如果对C/C++编译链接过程都了解的话,可以跳过不看。 http://www.firedragonpzy.com.cn/index.php/archives/2556 http://www.

先看两篇博文,作为基础知识。如果对C/C++编译链接过程都了解的话,可以跳过不看。

http://www.firedragonpzy.com.cn/index.php/archives/2556

http://www.cppblog.com/shifan3/archive/2007/01/05/17325.html

 

 

一、  编译不同目录下的多个文件

 

各个文件的布局如下:

       

head.h文件的代码:

 

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
 
 
  1. #ifndef  HEAD_H  
  2. #define  HEAD_H  
  3.   
  4. int add(int a, int b);  
  5.   
  6.   
  7. #endif  /*HEAD_H*/  

 

head.cpp文件的代码:

 

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
 
 
  1. #include    "head.h"  
  2.   
  3. int add(int a, int b)  
  4. {  
  5.     return a + b;  
  6. }  

 

main.cpp文件的代码(head.h头文件还没包含

 

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
 
 
  1. #include  <iostream>  
  2. using namespace std;  
  3.   
  4. int main(int argc, char *argv[])  
  5. {  
  6.     cout<<add(3, 5)<<endl;  
  7.     return 0;  
  8. }  

 

1)  以相对路径的方式直接包含头文件

为了能够使用add函数,必须包含add所在的头文件。 最简单的方法是直接在main.cpp文件中,用相对路径包含head.h文件.即 #include”function/head.h”。完整代码为

 

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
 
 
  1. #include  <iostream>  
  2. #include    "function/head.h"  
  3. using namespace std;  
  4.   
  5. int main(int argc, char *argv[])  
  6. {  
  7.     cout<<add(3, 5)<<endl;  
  8.     return 0;  
  9. }  

 

 

此时,编译命令为 :$g++ main.cpp function/head.cpp -o main

        这种用相对路径包含头文件的方式有很多弊端。当function目录改成其它名字,或者head.h文件放到其它目录了,这时都要对main.cpp文件进行修改,如果head.h头文件被很多其它文件包含的话,这个工作量就大多了。

 

2)  用编译选项 –I(大写i)

        其实,可以想一下,为什么iostream文件不在当前目录下,就可以直接使用呢?这是因为,编译器会在一些默认的目录下(/usr/include,/usr/inlucde/c++/4.4.3等目录)搜索头文件。所以,iostream头文件不用添加。但我们不能每写一个头文件就放到那里。

        知道了原理,现在来用一下一个编译选项 –I(include的缩写)用来告诉编译器,还可以去哪里找头文件。

        使用这个编译命令,$g++ main.cpp function/head.cpp -Ifunction -o main

        此时main.cpp文件写成

 

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
 
 
  1. #include  <iostream>  
  2. #include  <head.h>  
  3. using namespace std;  
  4.   
  5. int main(int argc, char *argv[])  
  6. {  
  7.     cout<<add(3, 5)<<endl;  
  8.     return 0;  
  9. }  

 

        可以看到head.h文件是用<>而不是””。想一下C语言书中,两者的区别。这说明,用-I选项,相当于说明了一条标准路径。

 

3)  使用.o文件

        此时,对于head.cpp文件,在编译命令中,还是要用到路径function/head.cpp。现在的想法是去掉这个。这时可以先根据head.cpp文件生成一个.o文件,然后就可以了去掉那个路径了。

 

        先cd 到function目录。

        输入命令:$g++ -c head.cpp -o head.o

        生成一个head.o目标文件,

 

        此时把生成的head.o文件复制到function的父目录,就是main.cpp所在的目录。

        然后回到function的父目录,输入命令$g++ main.cpp head.o -Ifunction -o main

 

        此时,直接使用head.o即可,无需head.cpp了。但头文件head.h还是要的。因为编译的时候要用到。链接的时候不用头文件。这个可以拆分成两条命令

 

        $g++ -c main.cpp -Ifunction -o main.o

        $g++ main.o head.o -o main

 

        第一条是编译命令,后一条是链接命令。

 

二、  静态库

        虽然上面说到的先生成.o目标文件,但如果function目录下有多个.cpp文件。那么就要为每一个.cpp文件都生成一个.o文件,这个工作量是会比较大。此时可以用静态库。静态库是把多个目标文件打包成一个文件。Anarchive(or static library) is simply a collection of object filesstored as a single file(摘自《Advanced Linux Programming》)。

 

        下面介绍静态库

        此时,文件布局为:

         

 

        head.h文件代码

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
 
 
  1. #ifndef  HEAD_H  
  2. #define  HEAD_H  
  3.   
  4. int add(int a, int b);  
  5. int sub(int a, int b);  
  6.   
  7.   
  8. #endif  /*HEAD_H*/  

 

 

        add.cpp文件代码

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
 
 
  1. #include    "head.h"  
  2.   
  3. int add(int a, int b)  
  4. {  
  5.     return a + b;  
  6. }  

 

 

        sub.cpp文件代码

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
 
 
  1. #include    "head.h"  
  2.   
  3. int sub(int a, int b)  
  4. {  
  5.     return a - b;  
  6. }  


 

        首先,生成.o目标文件。

        cd进入function目录,输入命令$g++ -c sub.cpp add.cpp  这会分别为两个源文件目标文件,即生成sub.o和add.o文件。

        然后,打包生成静态库,即.a文件。

        输入命令$ar -cr libhead.a add.o sub.o

 

        注意:

        1)  命令中,.a文件要放到 .o文件的前面

        2)  .a文件的格式。要以lib作为前缀, .a作为后缀。

        选项 c是代表create,创建.a文件。

        r是代表replace,如果之前有创建过.a文件,现在为了提高性能而更改了add函数里面的代码,此时,就可以用r选项来代替之前.a文件里面的add.o

        可以用命令$ar -t libhead.a 查看libhead.a文件里面包含了哪些目标文件。其执行结果自然为add.o  sub.o

 

       现在回过头来关注main.cpp文件。

        此时的main.cpp的代码为.

 

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
 
 
  1. #include  <iostream>  
  2. #include  <head.h>  
  3. using namespace std;  
  4.   
  5. int main(int argc, char *argv[])  
  6. {  
  7.     cout<<add(3, 5)<<endl;  
  8.     cout<<sub(10, 6)<<endl;  
  9.     return 0;  
  10. }  

 

        回到main.cpp文件所在的目录。

        输入命令:$g++ main.cpp -Ifunction -Lfunction -lhead -o main 生成可执行程序

        

        现在要解释一下使用静态库要用到的-L和-l(小写的L)选项。

        -L表示要使用的静态库的目录。这和前面所讲的-I(大写i)差不多,就是用来告诉编译器去哪里找静态库。因为可能-L所指明的目录下有很多静态库,所以除了要告诉去哪里找之外,还要告诉编译器,找哪一个静态库。此时,就要用到-l(小写L)了。它用来说明链接的时候要用到哪个静态库。

        注意:

        1. 注意是使用-lhead,而不是-llibhead

        命令中是使用-lhead,这是因为编译器会自动在库中添加lib前缀和.a后缀。

        2. 要把-l放到命令的尽可能后的位置,必须放到源文件的后面。

        如果使用命令中的顺序,将出现下面错误。

        

 

        还记得前面所链接的两篇博文的内容吧。当编译器在命令中碰到main.cpp文件后,会得到该文件的未解决符号表。然后,会在-l所指明的静态库中查找里面的每一个目标文件,把需要的部分抽取出来(编译器很聪明,不会全部统统接收)。但编译器只会对命令中的静态库查找一次,之后不再查找。如前面的错误命令那样,编译器先解析到-lhead。此时,未解决符号表都为空。编译器不会从libhead.a库中提取任何东西。当遇到main.cpp参数后,会得到未解决符号表。但此时已经错过libhead.a库了。编译器不会再次查找libhead.a库了。所以就出现错误了。(下面的动态库并不会出现这个问题。)

 

三、  动态库

        使用命令$g++ -c -fPIC add.cpp sub.cpp生成位置无关的目标文件。

        使用命令$g++ -shared -fPIC add.o sub.o -o libhead.so 生成.so动态链接库。

 

        利用动态库生成可执行文件 $g++ -Ifunction -Lfunction -lhead main.cpp -o main

 

 

        尝试运行. $./main 得到下面错误

        

 

        这说明加载的时候没有找到libhead.so动态库。这是因为,Linux查找动态库的地方依次是

  1. 环境变量LD_LIBRARY_PATH指定的路径
  2. 缓存文件/etc/ld.so.cache指定的路径
  3. 默认的共享库目录,先是/lib,然后是/usr/lib

        运行./main时,明显这三个地方都没有找到。因为我们没有把libhead.so文件放到那里。

 

        其实,我们可以在生成可执行文件的时候指定要链接的动态库是在哪个地方的。

        $g++ -Ifunction ./libhead.so main.cpp -o main

 

        这个命令告诉可执行文件,在当前目录下查找libhead.so动态库。注意这个命令不再使用-L 和 -l了。

        另外,还可以使用选项-Wl,-rpath,XXXX.其中XXXX表示路径。

        

四、  打造自己的库目录和头文件目录

        三个要解决的东西:指定编译时的头文件路径、指定链接时的动态库路径、指定运行时Linux加载动态库的查找路径

 

 

1.指定运行时Linux加载动态库的查找路径

        利用前面所说的Linux程序运行时查找动态库的顺序,让Linux在运行程序的时候,去自己指定的路径搜索动态库。

 

        可以修改环境变量LD_LIBRARY_PATH或者修改/etc/ld.so.cache文件。这里选择修改/etc/ld.so.cache文件。

        1)  创建目录/mylib/so。这个目录可以用来存放自己以后写的所有库文件。由于是在系统目录下创建一个目录,所以需要root权限

        2)  创建并编辑一个mylib.conf文件。输入命令$sudo vim /etc/ld.so.conf.d/mylib.conf

 

        在mylib.conf文件中输入 /mylib/so 

        保存,退出。

        3)  重建/etc/ld.so.cache文件。输入命令$sudo ldconfig

 

 

        输入下面命令,生成main文件。注意,其链接的时候是用function目录下的libhead动态库。

        $g++  -Ifunction -Lfunction -lhead main.cpp 

 

        直接运行./main。并没有错误。可以运行。说明,Linux已经会在运行程序时自动在/mylib/so目录下找动态链接库了。

 

2. 指定编译时的头文件路径

        先弄清编译器搜索头文件的顺序。

        1.先搜索当前目录(使用include””时)

        2.然后搜索-I指定的目录

        3.再搜索环境变量CPLUS_INCLUDE_PATH、 C_INCLUDE_PATH。两者分别是g++、gcc使用的。

        4.最后搜索默认目录 /usr/include  和 /usr/local/include等

 

        知道这些就简单了。输入下面命令。编辑该文件。

        $sudo vim /etc/bash.bashrc  这个文件是作用了所有Linux用户的,如果不想影响其他用户,那么就编辑~/.bashrc文件。

        打开文件后,去到文件的最后一行。输入下面的语句。

        

        修改环境变量。然后保存并推出。

        输入命令$bash 或者直接打开一个新的命令行窗口,使得配置信息生效。原理可以参考:http://blog.csdn.net/luotuo44/article/details/8917764

 

        此时,可以看到 已经可以不用-I选项 下面编译命令能通过了。

        $g++ -Lfunction -lhead mian.cpp -o main

 

3.指定链接时的动态库路径

         需要注意的是,链接时使用动态库和运行时使用动态库是不同的。

        同样先搞清搜索顺序:

        1. 编译命令中-L指定的目录

        2. 环境变量LIBRARY_PATH所指定的目录

        3. 默认目录。/lib、/usr/lib等。

 

        接下来和指定头文件路径一样。输入命令$sudo vim /etc/bash.bashrc   在文件的最后一行输入

        

        保存,退出。

 

        同样,输入bash,使得配置信息生效。

 

        这是终极目标了。其中-lhead是不能省的。因为,编译器要知道,你要链接到哪一个动态库。当然,如果想像C运行库那样,链接时默认添加的动态库,那么应该也是可以通过设置,把libhead.so库作为默认库。但并不是所有的程序都会使用这个库。要是设置为默认添加的,反而不好。

 

目录
相关文章
|
2月前
|
Linux 编译器 开发工具
【Linux快速入门(三)】Linux与ROS学习之编译基础(Cmake编译)
【Linux快速入门(三)】Linux与ROS学习之编译基础(Cmake编译)
129 2
|
2月前
|
Linux
在 Linux 系统中,“cd”命令用于切换当前工作目录
在 Linux 系统中,“cd”命令用于切换当前工作目录。本文详细介绍了“cd”命令的基本用法和常见技巧,包括使用“.”、“..”、“~”、绝对路径和相对路径,以及快速切换到上一次工作目录等。此外,还探讨了高级技巧,如使用通配符、结合其他命令、在脚本中使用,以及实际应用案例,帮助读者提高工作效率。
121 3
|
2月前
|
存储 Linux Windows
linux常用目录
/sbin s就是super User的意思,这里存放的是系统管理员使用的系统管理程序。 /home 存放普通用户的主目录,在Linux中每个用户都有一个自己的目录,一版该目录名是以用户的账号命名的。 /root 该目录为系统管理员,也称为超级权限者的用户主目录。 /lib 系统开机所需要最基本的动态连接共享库,其作用类似于windows里的DLL文件。几乎所有的应用程序都需要用到这些共享库。 /etc 所有的系统管理所需要的配置文件和子目录。 /usr 这是一个非常重要的目录,用户的很多应用程序和文件都放在这个目录下,类似于windows下的program fies目录。 /bo
53 2
|
16天前
|
Ubuntu Linux Go
golang编译成Linux可运行文件
本文介绍了如何在 Linux 上编译和运行 Golang 程序,涵盖了本地编译和交叉编译的步骤。通过这些步骤,您可以轻松地将 Golang 程序编译成适合 Linux 平台的可执行文件,并在目标服务器上运行。掌握这些技巧,可以提高开发和部署 Golang 应用的效率。
127 14
|
1月前
|
存储 编译器 Linux
动态链接的魔法:Linux下动态链接库机制探讨
本文将深入探讨Linux系统中的动态链接库机制,这其中包括但不限于全局符号介入、延迟绑定以及地址无关代码等内容。
466 22
|
17天前
|
Linux
linux查看目录下的文件夹命令,find查找某个目录,但是不包括这个目录本身?
通过本文的介绍,您应该对如何在 Linux 系统中查看目录下的文件夹以及使用 `find` 命令查找特定目录内容并排除该目录本身有了清晰的理解。掌握这些命令和技巧,可以大大提高日常文件管理和查找操作的效率。 在实际应用中,灵活使用这些命令和参数,可以帮助您快速定位和管理文件和目录,满足各种复杂的文件系统操作需求。
46 8
|
2月前
|
运维 监控 网络协议
运维工程师日常工作中最常用的20个Linux命令,涵盖文件操作、目录管理、权限设置、系统监控等方面
本文介绍了运维工程师日常工作中最常用的20个Linux命令,涵盖文件操作、目录管理、权限设置、系统监控等方面,旨在帮助读者提高工作效率。从基本的文件查看与编辑,到高级的网络配置与安全管理,这些命令是运维工作中的必备工具。
205 3
|
2月前
|
Linux Python
Linux 中某个目录中的文件数如何查看?这篇教程分分钟教会你!
在 Linux 系统中,了解目录下文件数量是常见的需求。本文介绍了四种方法:使用 `ls` 和 `wc` 组合、`find` 命令、`tree` 命令以及编程实现(如 Python)。每种方法都附有详细说明和示例,适合不同水平的用户学习和使用。掌握这些技巧,可以有效提升系统管理和日常使用的效率。
1083 6
|
2月前
|
分布式计算 Java Hadoop
linux中HADOOP_HOME和JAVA_HOME删除后依然指向旧目录
通过以上步骤,可以有效地解决 `HADOOP_HOME`和 `JAVA_HOME`删除后依然指向旧目录的问题。确保在所有相关的配置文件中正确设置和删除环境变量,并刷新当前会话,使更改生效。通过这些措施,能够确保系统环境变量的正确性和一致性。
34 1
|
2月前
|
Linux Python
Linux 中某个目录中的文件数如何查看?这篇教程分分钟教会你!
在 Linux 系统中,了解目录下的文件数量是常见的需求。本文介绍了多种方法,包括使用 `ls` 和 `wc` 命令组合、`find` 命令、`tree` 命令以及编程方式(如 Python)。无论你是新手还是有经验的用户,都能找到适合自己的方法。掌握这些技巧将提高你在 Linux 系统中的操作效率。
90 4