学在LINUX下编程(转)

学在LINUX下编程

 
预备知识

用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的任选项。     
原文地址:https://www.cnblogs.com/SunWentao/p/1325756.html