当前位置: 首页 > news >正文

【gcc】 gcc编译工具生成动态库(共享库)和静态库

目录

一、GCC 简介

GCC编译流程

GCC编译选项

二、GCC生成动态库和静态库

三、生成动态库和静态库实例

四、静态库和动态库的使用和配置

静态库的使用

共享库的使用

不到你指定链接的so错误


一、GCC 简介

《gcc命令》http://man.linuxde.net/gcc

gcc作为编译工具,用在Linux操作系统中,可以编译C、C++、Object-C、JAVA等语言。编译过程中可以带编译选项,选择编译过程。

GCC编译流程

1)预处理 Pre-Processing
2)编译 Compiling
3)汇编 Assembling
4)链接 Linking

GCC编译选项

1、一般选项

1) -c :只编译,不链接,生成目标文件“.o”。
2) -S :只编译,不汇编,生成汇编代码“.S”。
3) -E :只进行预编译/预处理,不做其他处理。

4) -o file:输出文件名为file
5) -g :在可执行程序中包含标准调试信息。
6) -v :打印出编译器内部编译各过程的命令行信息和编译器的版本。
7) -I dir :在头文件的搜索路径列表中添加dir目录
8) -L dir :在库文件的搜索路径列表中添加dir目录
9) -static :连接静态库(静态库也可以用动态库链接方式链接)
10) -llibrary :连接名为library的库文件(显示指定需要链接的动态库文件)

-shared 该选项指定生成动态连接库l -fPIC:表示编译为位置独立的代码,不用此选项的话编译后的代码是位置相关的所以动态载入时是通过代码拷贝的方式来满足不同进程的需要,而不能达到真 正代码段共享的目的。l -ltest:编译器查找动态连接库时有隐含的命名规则,即在给出的名字前面加上lib,后面加上.so来确定库的名称l LD_LIBRARY_PATH:这个环境变量指示动态连接器可以装载动态库的路径。l 当然如果有root权限的话,可以修改/etc/ld.so.conf文件,然后调用 /sbin/ldconfig来达到同样的目的,不过如果没有root权限,那么只能采用输出LD_LIBRARY_PATH的方法了。   

2、gcc告警和出错选项


1) -ansi :支持符合ANSI标准的C程序
2) -pedantic :允许发出ANSI C标准所列出的全部警告信息
3) -pedantic-error :允许发出ANSI C标准所列出的全部错误信息
4) -w :关闭所有警告
5) -Wall :允许发出gcc提供的所有有用的报警信息
6) -werror :把所有的告警信息转化为错误信息,并在告警发生时终止编译过程3、gcc优化选项

1)-On:n的取值范围不一致,比较典型的范围为0变化到2或者3。
虽然优化选项可以加速代码的运行速度,但对于调试而言将是一个很大的挑战。因为代码在经过优化之后,原先在源程序中声明和使用的变量很可能不再使用,控制流也可能会突然跳转到意外的地方,循环语句也可能因为循环展开而变得到处都有。

二、GCC生成动态库和静态库

(1)静态库     静态是指每个用到该库的应用程序都拥有一份自己的库拷贝;(2) 共享库     一个共享库有可能被多个所有应用程序共享。

1)动态库生成
1.单个源文件/目标直接生成动态库
a. 
gcc -fPIC -shared xxx.c -o libxxx.so
b.
gcc -fPIC -shared xxx.o -o libxxx.so2.多个源文件/目标生成动态库
a.
gcc -fPIC -shared xxx1.c xxx2.c xxx3.c -o libxxx.so 
b.
gcc -fPIC -shared xxx1.o xxx2.o xxx3.o -o libxxx.so 2)静态库生成

简单地说,静态库是一个目标文件的简单集合。因此,首先有目标文件。

第一步:gcc -c myfunc.c myproc.c将得到myfunc.o和myproc.o。

第二步:由ar(archive,归档的意思)把多个目标文件集合起来。$ar -r libmyjob.a myfunc.o myproc.o    

通常,静态库的命名方式应遵守libXXXXX.a格式。应用程序在使用静态库的时候,通常只需要把命名中的XXXXX部分传递给gcc即可。


1.单个源文件/目标直接生成静态库
a.
ar -rc libxxx.a xxx.o(正确方法)
b. ar -rc libxxx.a xxx.c (静态库可以生成;当运行连接了该静态库的可执行程序会报错:could not read symbols:Archive has no index;run ranlib to add one)2.多个源文件/目标生成静态库
a.
ar -rc libxxx.a xxx1.o xxx2.o xxx3.o (正确方法)
b.
ar -rc libxxx.a xxx1.c xxx2.c xxx3.c (静态库可以生成;当运行连接了该静态库的可执行程序会报错:could not read symbols:Archive has no index;run ranlib to add one)四、多个源文件生成一个可执行文件
gcc xxx1.c xxx2.c xxx3.c xxx4.c main.c -o main

三、生成动态库和静态库实例

https://blog.csdn.net/yanlaifan/article/details/52193118

四、静态库和动态库的使用和配置

(1)静态库     静态是指每个用到该库的应用程序都拥有一份自己的库拷贝;应用程序运行的时候,即使将库删除也没有问题,因为应用程序自己已经有了自己的拷贝。(2) 共享库     一个共享库有可能被多个所有应用程序共享。因此,对每个应用程序来说,即使不再使用某个共享库,也不应将其删除。此外,应用程序需要正确的环境变量设置 (LD_LIBRARY_PATH),从而找到共享库所在的位置,否则,应用程序运行时会报告找不到这个库。二、关于使用库的问题     对于静态库来说,gcc 的命令行参数(-I, -L)shell的环境变量(C_INCLUDE_PATH, LIBRARY_PATH)     对于共享库来说,程序在运行时,如果用到了动态库,也需要找到对应的动态库文件;实现的方法:shell的环境变量 (LD_LIBRARY_PATH)

静态库的使用

1) gcc命令行参数(-I, -L)

 默认情况下,gcc会自动搜索下面的路径:对 头文件:

/usr/local/include/
/usr/include/
对库文件:
/usr/local/lib/
/usr/lib/

开发者自定义路径:   例如:如果工程涉及到GDBM(GNU DataBase Management)包,需要libgdbm库,而系统中安装GDBM的路径是:

头文件:/opt/gdbm-1.8.3 /include
库文件:/opt/gdbm-1.8.3/lib/
  那么,gcc的命令参数是:
$gcc … -I/opt/gdbm-1.8.3/include  -L/opt/gdbm-1.8.3/lib –lgdbm

 注意:为保证兼容性,必须坚决杜绝在C/C++源文件的#include语句中或者其他相关语句中使用上述路径。

2) shell环境变量(Environmental Variable)

     除了用命令行参数,还可以用环境变量来指示gcc搜索适当的路径。(1)Bash     对于Bash来说,除了由系统管理员配置的内容以外,每个用户的用户目录($HOME)下,有个.bash_profile文件。可在该文件内,增加下面 的两个语句来设置GDBM头文件路径的环境变量:

C_INCLUDE_PATH=/opt/gdbm-1.8.3/include
export C_INCLUDE_PATH

     类似地,在该文件内用下面的两个语句来设置库文件路径的环境变量:

LIBRARY_PATH=/opt/gdbm-1.8.3/lib
export LIBRARY_PATH

     在.bash_profile中有了上述语句以后,就不用再使用-I和-L来搜索特定包的路径了。但是链接库的时候,还是要用-l选项。$gcc … –lgdbm  

   在Bash下,要检查有什么样的环境变量,可用env命令。$env

共享库的使用

     使用共享库的应用程序,要通过环境变量LD_LIBRARY_PATH找到对应的共享库文件,不要忘了之前已经设置的共享库目录,所以是=xxxx:$LD_LIBRARY_PATH,即赋值新增的路径和原来的路径:

LD_LIBRARY_PATH=/opt/gdbm-1.8.3/lib:$LD_LIBRARY_PATH
export LD_LIBRARY_PATH

转自:http://blog.csdn.net/gnuhpc/article/details/5448149

gcc -E hello.c -o hello.i   对hello.c文件进行预处理,生成了hello.i 文件
gcc -S hello.i -o hello.s    对预处理文件进行编译,生成了汇编文件
gcc -c hello.s -o hello.o  对汇编文件进行编译,生成了目标文件
gcc hello.o -o hello 对目标文件进行链接,生成可执行文件
gcc hello.c -o hello 直接编译链接成可执行目标文件
gcc -c hello.c 或 gcc -c hello.c -o hello.o 编译生成可重定位目标文件

不到你指定链接的so错误

调用动态库的时候有几个问题会经常碰到,有时,明明已经将库的头文件所在目录 通过 “-I” include进来了,库所在文件通过 “-L”参数引导,并指定了“-l”的库名,但通过ldd命令察看时,就是死活找不到你指定链接的so文件,这时你要作的就是通过修改 LD_LIBRARY_PATH或者/etc/ld.so.conf文件来指定动态库的目录。通常这样做就可以解决库无法链接的问题了。

相关文章:

  • 【cJSON】cJSON 实例
  • 【yum/apt-get】yum/apt-get/rpm/dpkg/brew之间的区别
  • 【删除重复记录】SQL删除重复记录
  • 【myISAM和innoDB】mySql的引擎myisam和innodb的区别/mysiam
  • 【文档/键值数据库】文档数据库和键值数据库有什么区别
  • 【JDBC与JPA】JDBC与JPA
  • Apache+Tomcat配置转发
  • 【SQLite】SQLServerCompact/SQLiteToolboxVS的SQLite插件使用
  • 【mysql索引】MySQL索引原理及慢查询优化
  • 【趋势和新技术】趋势和技术
  • 【mysql中文编码】mysql中文乱码中文编码问题
  • 【RPC】为什么需要RPC,而不是简单的HTTP接口
  • 【JAVA】java中implements和extends的使用小结
  • 【postgreSQL】Postgres-XL--基于PostgreSQL的开源分布式实现
  • 【python C结构体】Python Ctypes结构体指针处理(函数参数,函数返回)
  • 4. 路由到控制器 - Laravel从零开始教程
  • 8年软件测试工程师感悟——写给还在迷茫中的朋友
  • android高仿小视频、应用锁、3种存储库、QQ小红点动画、仿支付宝图表等源码...
  • Centos6.8 使用rpm安装mysql5.7
  • CSS盒模型深入
  • javascript 哈希表
  • JavaScript函数式编程(一)
  • js
  • JS数组方法汇总
  • laravel with 查询列表限制条数
  • Laravel5.4 Queues队列学习
  • React-redux的原理以及使用
  • Spark学习笔记之相关记录
  • Spring Cloud(3) - 服务治理: Spring Cloud Eureka
  • Vue官网教程学习过程中值得记录的一些事情
  • 闭包,sync使用细节
  • 从伪并行的 Python 多线程说起
  • 翻译--Thinking in React
  • 警报:线上事故之CountDownLatch的威力
  • 腾讯优测优分享 | Android碎片化问题小结——关于闪光灯的那些事儿
  • 通过获取异步加载JS文件进度实现一个canvas环形loading图
  • 译有关态射的一切
  • 主流的CSS水平和垂直居中技术大全
  • 阿里云ACE认证学习知识点梳理
  • #includecmath
  • (草履虫都可以看懂的)PyQt子窗口向主窗口传递参数,主窗口接收子窗口信号、参数。
  • (分享)一个图片添加水印的小demo的页面,可自定义样式
  • (简单有案例)前端实现主题切换、动态换肤的两种简单方式
  • (四) Graphivz 颜色选择
  • (学习日记)2024.04.10:UCOSIII第三十八节:事件实验
  • (一)80c52学习之旅-起始篇
  • (转)拼包函数及网络封包的异常处理(含代码)
  • .net分布式压力测试工具(Beetle.DT)
  • .NET精简框架的“无法找到资源程序集”异常释疑
  • [.net]官方水晶报表的使用以演示下载
  • [2009][note]构成理想导体超材料的有源THz欺骗表面等离子激元开关——
  • [2016.7.Test1] T1 三进制异或
  • [Android]使用Retrofit进行网络请求
  • [Angular 基础] - 表单:响应式表单
  • [C#]猫叫人醒老鼠跑 C#的委托及事件