预备知识

用gcc编译程序要用到一些选项要知道
    -c,只编译,不连接成为可执行文件,编译器只是由输入的.c等源代码文件生成.o为后缀的目标文件,通常用于编译不包含主程序的子程序文件。 
    -o output_filename,确定输出文件的名称为output_filename,同时这个名称不能和源文件同名。如果不给出这个选项,gcc就给出预设的可执行文件a.out。 
    -Idirname,将dirname所指出的目录加入到程序头文件目录列表中,是在预编译过程中使用的参数。
    -lname,在连接时,装载名字为“libname.a”的函数库,该函数库位于系统预设的目录或者由-L选项确定的目录下。例如,-lm(缩写)表示连接名为“libm.a”的数学函数库。 
    -Ldirname,将dirname所指出的目录加入到程序函数档案库文件的目录列表中,是在连接过程中使用的参数。在预设状态下,连接程序ld在系统的预设路径中(如/usr/lib)寻找所需要的档案库文件,这个选项告诉连接程序,首先到-L指定的目录中去寻找,然后到系统预设路径中寻找,如果函数库存放在多个目录下,就需要依次使用这个选项,给出相应的存放目录。
...
1.静态库
先整个静态库练下手
书上用的就是hello world的例子,我也没啥抄袭的意思,我就用Hey! Girl!
首先咱要写个头文件
test.h

void out(char* arg);

然后就是.c了
test.c

#include <stdio.h>
#include "test.h"
void out(char* arg)
{
    printf("%s\n",arg);
}
再来调用他
main.c
#include "test.h"
main()
{
    out("Hey!Girl!");
}

好了,编译先
# gcc -c test.c main.c
这样就生成了test.o和main.o
然后再连接
# gcc -o main main.o test.o
连接生成了一个可执行文件main
运行一下
# ./main
Hey!Girl!

这个.o只是目标文件
我们要用ar命令把他放到静态库中,后面有ar命令的说明
# ar crv libtest.a test.o
-a test.o
好了放到静态库libtest.a中了,你可以直接用静态库来连接main.o了
# gcc -o main main.o libtest.a
也可以
# gcc -o main main.o -L. -ltest
别忘了.表示当前目录下 
我们可以用nm命令来查看静态库

静态库.sa(前面用的后缀是.a,书上这么写的都一样用了,反正LINUX不看后缀)、共享库.so和ms的动态连接库DLL、静态连接库LIB很象嘛
可能是跟我一样没啥抄袭的意思~~

2.共享库
跟WINDOWS类似
也是得到函数的指针然后直接用
在windows里用loadlibrary和freelibrary,在linux里用dlopen和dlclose
用这些要#include <dlfcn.h>
windows用GetProcAddress得到指针linux用dlsym

这里就要用到一些gcc特殊选项了
-D_REENTRANT    使得预处理器符号 _REENTRANT 被定义,这个符号激活一些宏特性。
-fPIC        选项产生位置独立的代码。由于库是在运行的时候被调入,因此这个选项是必需的,因为在编译的时候,装入内存的地址还不知道。如果不使用这个选项,库文件可能不会正确运行。
-shared      选项告诉编译器产生共享库代码。
-Wl,-soname    -Wl 告诉编译器将后面的参数传递到连接器。而 -soname 指定了共享库的 soname。

直接用上面的test.c test.h来编译成so
# gcc -c test.c -o test.o
# gcc -shared -fPIC test.o -o test.so
-fPIC选项,使生成的代码是位置无关的
好了共享库出来了
然后调用他把main.c改改
/*main.c*/
#include "test.h"
#include <stdio.h>
#include <dlfcn.h>
main()
{
void *handle = NULL;
void (*fpOut)(char*) = NULL;
handle = dlopen("test.so",RTLD_LAZY);
if(handle == NULL)
   printf("dlopen == NULL\n");
else
   {
      fpOut = dlsym(handle,"out");
      if(fpOut == NULL)
        printf(fpOut == NULL);
      else
        fpOut("Hey!Girl!");
      dlclose(handle);
   }
}
# gcc -o main main.c -ldl test.so(-ldl是连接dl库)
你需要把LD_LIBRARY_PATH设置到你的.so的路径下
# export LD_LIBRARY_PATH=./

函数dlopen需要在文件系统中查找目标文件并为之创建句柄。有四种方法指定目标文件的位置:
绝对路径
在环境变量LD_LIBRARY_PATH指定的目录中
在/etc/ld.so.cache中指定的库列表中
在/usr/lib或者/lib中

现在便可以执行了
其实他还提供了错误处理函数dlerror
我找了一个例子,贴了部分,不完全,仅供学习
const char* hError;

int main(int argc,char* argv[])
{
   slib=dlopen("xxx.so",RTLD_LAZY);
   hError=dlerror();
   if (hError)
   {
      printf("dlopen Error!\n");
      return 1;
   }
   func=dlsym(slib,"func");
   hError=dlerror();
   if (hError)
   {
      printf("dlsym Error!\n");
      return 1;
   }
   func("How do you do?\n");
   dlclose(slib);
   hError=dlerror();
   if (hError)
   {
      printf("dlclose Error!\n");
      return 1;
   }
   return 0;
}

void * dlopen(const char *pathname, int mode);  
mode是打开方式:

RTLD_LAZY:打开动态库后只重定位库中数据地址引用而不重定位而函数引用,  
函数引用在该函数要被激活时才定位,的确LAZY呵呵,但省开销;)  
RTLD_NOW: 与上者相比,动态库一被打开就重定位所有函数的引用。  
RTLD_GLOBAL:打开动态库里的全局符号可以被其它所有库重定位。  
...还有,我就不列举了,其实我也不知道

基本上可以编程了现在
以后全靠书了

ps:
下面是ar命令的格式:

  ar [-]{dmpqrtx}[abcfilNoPsSuvV] [membername] [count] archive files...

  例如我们可以用ar rv libtest.a hello.o hello1.o来生成一个库,库名字是test,链接时可以用-ltest链接。该库中存放了两个模块hello.o和hello1.o。选项前可以有‘-#39;字符,也可以没有。下面我们来看看命令的操作选项和任选项。现在我们把{dmpqrtx}部分称为操作选项,而[abcfilNoPsSuvV]部分称为任选项。

  {dmpqrtx}中的操作选项在命令中只能并且必须使用其中一个,它们的含义如下:

d:从库中删除模块。按模块原来的文件名指定要删除的模块。如果使用了任选项v则列出被删除的每个模块。 
m:该操作是在一个库中移动成员。当库中如果有若干模块有相同的符号定义(如函数定义),则成员的位置顺序很重要。如果没有指定任选项,任何指定的成员将移到库的最后。也可以使用#39;a','b',或'I'任选项移动到指定的位置。 
p:显示库中指定的成员到标准输出。如果指定任选项v,则在输出成员的内容前,将显示成员的名字。如果没有指定成员的名字,所有库中的文件将显示出来。 
q:快速追加。增加新模块到库的结尾处。并不检查是否需要替换。#39;a','b',或'I'任选项对此操作没有影响,模块总是追加的库的结尾处。如果使用了任选项v则列出每个模块。 这时,库的符号表没有更新,可以用'ar s'或ranlib来更新库的符号表索引。 
r:在库中插入模块(替换)。当插入的模块名已经在库中存在,则替换同名的模块。如果若干模块中有一个模块在库中不存在,ar显示一个错误消息,并不替换其他同名模块。默认的情况下,新的成员增加在库的结尾处,可以使用其他任选项来改变增加的位置。 
t:显示库的模块表清单。一般只显示模块名。 
x:从库中提取一个成员。如果不指定要提取的模块,则提取库中所有的模块。 
  下面在看看可与操作选项结合使用的任选项:

a:在库的一个已经存在的成员后面增加一个新的文件。如果使用任选项a,则应该为命令行中membername参数指定一个已经存在的成员名。 
b:在库的一个已经存在的成员前面增加一个新的文件。如果使用任选项b,则应该为命令行中membername参数指定一个已经存在的成员名。 
c:创建一个库。不管库是否存在,都将创建。 
f:在库中截短指定的名字。缺省情况下,文件名的长度是不受限制的,可以使用此参数将文件名截短,以保证与其它系统的兼容。 
i:在库的一个已经存在的成员前面增加一个新的文件。如果使用任选项i,则应该为命令行中membername参数指定一个已经存在的成员名(类似任选项b)。 
l:暂未使用 
N:与count参数一起使用,在库中有多个相同的文件名时指定提取或输出的个数。 
o:当提取成员时,保留成员的原始数据。如果不指定该任选项,则提取出的模块的时间将标为提取出的时间。 
P:进行文件名匹配时使用全路径名。ar在创建库时不能使用全路径名(这样的库文件不符合POSIX标准),但是有些工具可以。 
s:写入一个目标文件索引到库中,或者更新一个存在的目标文件索引。甚至对于没有任何变化的库也作该动作。对一个库做ar s等同于对该库做ranlib。 
S:不创建目标文件索引,这在创建较大的库时能加快时间。 
u:一般说来,命令ar r...插入所有列出的文件到库中,如果你只想插入列出文件中那些比库中同名文件新的文件,就可以使用该任选项。该任选项只用于r操作选项。 
v:该选项用来显示执行操作选项的附加信息。 
V:显示ar的版本。 
2.nm基本用法命令

  nm用来列出目标文件的符号清单。下面是nm命令的格式:

  nm [-a|--debug-syms] [-g|--extern-only] [-B][-C|--demangle] [-D|--dynamic] [-s|--print-armap][-o|--print-file-name] [-n|--numeric-sort][-p|--no-sort] [-r|--reverse-sort] [--size-sort][-u|--undefined-only] [-l|--line-numbers] [--help][--version] [-t radix|--radix=radix][-P|--portability] [-f format|--format=format][--target=bfdname] [objfile...]

  如果没有为nm命令指出目标文件,则nm假定目标文件是a.out。下面列出该命令的任选项,大部分支持"-"开头的短格式和"—"开头的长格式。

-A、-o或--print-file-name:在找到的各个符号的名字前加上文件名,而不是在此文件的所有符号前只出现文件名一次。 
例如nm libtest.a的输出如下:

CPThread.o:
00000068 T Main__8CPThreadPv
00000038 T Start__8CPThread
00000014 T _._8CPThread
00000000 T __8CPThread
00000000 ? __FRAME_BEGIN__
.......................................

则nm -A 的输出如下:

libtest.a:CPThread.o:00000068 T Main__8CPThreadPv
libtest.a:CPThread.o:00000038 T Start__8CPThread
libtest.a:CPThread.o:00000014 T _._8CPThread
libtest.a:CPThread.o:00000000 T __8CPThread
libtest.a:CPThread.o:00000000 ? __FRAME_BEGIN__
..................................................................

-a或--debug-syms:显示调试符号。 
-B:等同于--format=bsd,用来兼容MIPS的nm。 
-C或--demangle:将低级符号名解码(demangle)成用户级名字。这样可以使得C++函数名具有可读性。 
-D或--dynamic:显示动态符号。该任选项仅对于动态目标(例如特定类型的共享库)有意义。 
-f format:使用format格式输出。format可以选取bsd、sysv或posix,该选项在GNU的nm中有用。默认为bsd。 
-g或--extern-only:仅显示外部符号。 
-n、-v或--numeric-sort:按符号对应地址的顺序排序,而非按符号名的字符顺序。 
-p或--no-sort:按目标文件中遇到的符号顺序显示,不排序。 
-P或--portability:使用POSIX.2标准输出格式代替默认的输出格式。等同于使用任选项-f posix。 
-s或--print-armap:当列出库中成员的符号时,包含索引。索引的内容包含:哪些模块包含哪些名字的映射。 
-r或--reverse-sort:反转排序的顺序(例如,升序变为降序)。 
--size-sort:按大小排列符号顺序。该大小是按照一个符号的值与它下一个符号的值进行计算的。 
-t radix或--radix=radix:使用radix进制显示符号值。radix只能为"d"表示十进制、"o"表示八进制或"x"表示十六进制。 
--target=bfdname:指定一个目标代码的格式,而非使用系统的默认格式。 
-u或--undefined-only:仅显示没有定义的符号(那些外部符号)。 
-l或--line-numbers:对每个符号,使用调试信息来试图找到文件名和行号。对于已定义的符号,查找符号地址的行号。对于未定义符号,查找指向符号重定位入口的行号。如果可以找到行号信息,显示在符号信息之后。 
-V或--version:显示nm的版本号。 
--help:显示nm的任选项。   

 
http://www.cnblogs.com/SunWentao/archive/2008/11/03/1325756.html

学在LINUX下编程(各种情况比较详细)的更多相关文章

  1. Linux下编程学习一

    本篇主要记录一些在学习LINUX下编程时,, C和C++语言的一些基础的常识, 一. 函数指针 void MyFun(int x); 函数声明 void (*FunP)(int ); 函数指针声明 下 ...

  2. Linux下函数调用堆栈帧的详细解释【转】

    转自:http://blog.chinaunix.net/uid-30339363-id-5116170.html 原文地址:Linux下函数调用堆栈帧的详细解释 作者:cssjtuer http:/ ...

  3. linux下mysql-5.5.15安装详细步骤

    linux下mysql-5.5.15安装详细步骤 注:该文档中用到的目录路径以及一些实际的值都是作为例子来用,具体的目录路径以各自安装时的环境为准 mysql运行时需要一个启动目录.一个安装目录和一个 ...

  4. Linux下编程获取本地IP地址的常见方法

    转载于:http://blog.csdn.net/k346k346/article/details/48231933 在进行linux网络编程时,经常用到本机IP地址.本文罗列一下常见方法,以备不时之 ...

  5. Linux 下编程

    关于Linux 下的C语言编译命令和编程要点! https://www.cnblogs.com/wfwenchao/p/3985153.html?utm_source=tuicool&utm_ ...

  6. linux下编程epoll实现将GPS定位信息上报到服务器

    操作系统:CentOS 开发板:fl2440 开发模块:A7(GPS/GPRS),RT3070(无线网卡) ********************************************** ...

  7. linux下gcc编译的参数详细说明

    参考网址:1 http://hi.baidu.com/zengzhaonong/item/f1f9383565fa5c302e0f8125 gcc使用方法 汇总 2 http://s99f.blog. ...

  8. Linux下编译安装redis,详细教程

    话不多说,直接开工 准备工作: 本人测试环境:Win10 虚拟机:VM Linux:CentOS5.5 (已搭建好LNMP环境) 软件包:redis-2.6.14.tar.gz (Linux下redi ...

  9. opencv 4.0 + linux下静态编译,展示详细ccmake的参数配置

    #先安装 cmake 3.14 # cmake安装到了 /usr/local/bin #配置PATH export PATH="$PATH:/usr/local/bin" #下载最 ...

随机推荐

  1. 使用lapack图书馆逆矩阵

    阿土,直接在代码: #include <string> #include "lapacke.h" #include "lapack_aux.h" i ...

  2. WPF 文字换行TextWrapping 显示不全用省略号TextTrimming 显示不全弹提示内容ToolTip

    原文:WPF 文字换行TextWrapping 显示不全用省略号TextTrimming 显示不全弹提示内容ToolTip [TextBlock] 换行    TextWrapping="W ...

  3. 【水水水】【洛谷 U4566】赛车比赛

    题目背景 kkk在赛车~ 题目描述 现在有N辆赛车行驶在一条直线跑道(你可以认为跑道无限长)上.它们各自以某种速度匀速前进,如果有两辆车A车和B车,A车在B车的后面,且A车的速度大于B车的速度,那么经 ...

  4. Python 编程规范 —— TODO 注释(结合 PyCharm)

    编程是代码和注释的统一: TODO 自然表示需要做而未做的一些待完成的事项,有助于事后的检索,以及对整体项目做进一步的修改迭代. # TODO(kl@gmail.com): Use a "* ...

  5. MinGW安装和使用(不是mingw-w32)

    MinGW全称Minimalist GNU For Windows,是个精简的Windows平台C/C++.ADA及Fortran编译器,相比Cygwin而言,体积要小很多,使用较为方便.MinGW提 ...

  6. dom4j解析xml获取所有的子节点并放入map中

    dom4j递归解析所有子节点 //解析返回的xml字符串,生成document对象 Document document = DocumentHelper.parseText(resultXml); / ...

  7. Internet protocol optimizer

    A method for optimizing the throughput of TCP/IP applications by aggregating user application data a ...

  8. Entity Framework加载数据的三种方式。

    MSDN文章Loading Related Entities 有 Eagerly Loading Lazy Loading Explicitly Loading 三种方式. 而看到查询中包含Inclu ...

  9. 利用WPF的ListView进行大数据量异步加载

    原文:利用WPF的ListView进行大数据量异步加载 由于之前利用Winform的ListView进行大数据量加载的时候,诟病良多,所以今天试着用WPF的ListView来做了一下,结果没有让我失望 ...

  10. NS2网络模拟(4)-吞吐率图

    1: #NS2_有线部分\ForGnuplot.plot 2: 3: #gnuplot> 4: #set xtics 0, 1, 10 5: set grid 6: set xrange [0: ...