zoukankan      html  css  js  c++  java
  • 学在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的任选项。     
  • 相关阅读:
    二十一.组合模式
    二十四.桥接模式
    二十六.职责链模式
    二十五.命令模式
    将小写转化成大写
    备份JOB SCHEDULE ENTRY的简单方法
    如何确定哪一个作业锁定QDLS下的一个目标
    WRKACTJOB命令一些有用功能介绍
    如何使用CA/400批处理的方式传输数据
    用前缀给字段命名
  • 原文地址:https://www.cnblogs.com/SunWentao/p/1325756.html
Copyright © 2011-2022 走看看