来自 威尼斯国际官方网站 2019-09-16 09:40 的文章
当前位置: 威尼斯国际官方网站 > 威尼斯国际官方网站 > 正文

linux动态库编写翻译和选用详细剖判

引言

linux动态库编写翻译和利用详细深入分析,linux动态编写翻译解析

引言

     重点陈述linux上选取gcc编写翻译动态库的一对操作.并且对其深切的案例解析.
末尾介绍一下动态库插件能力, 让代码向后包容.关于linux上利用gcc基础编写翻译,

预编写翻译,编写翻译,生成机械码最终链接输出可实施文件流程参照上边.

  gcc编写翻译流程 

而本文重视是解析动态库相关的知识点. 首先看供给动用的测量试验素材

 heoo.h **

#ifndef _H_HEOO
#define _H_HEOO

/*
 * 测试接口,得到key内容
 *      : 返回key的字符串
 */
extern const char* getkey(void);

/*
 * 测试接口,得到value内容
 * arg      : 传入的参数
 *          : 返回得到的结果
 */
extern void* getvalue(void* arg);

#endif // !_H_HEOO

 heoo-getkey.c 

#include "heoo.h"

/*
 * 测试接口,得到key内容
 *      : 返回key的字符串
 */
const char*
getkey(void) {
     return "heoo-getkey.c getkey";
}

 heoo-getvalue.c 

#include "heoo.h"
#include <stdio.h>

/*
 * 测试接口,得到value内容
 * arg      : 传入的参数
 *          : 返回得到的结果
 */
const void* 
getvalue(void* arg) {
    const char* key = "heoo-getvalue.c getvalue";
    printf("%s - %sn", key, (void*)arg);
    return key;
}

heoo.c 

#include "heoo.h"
#include <stdio.h>

/*
 * 测试接口,得到key内容
 *      : 返回key的字符串
 */
const char* 
getkey(void) {
    return "heoo.c getkey";
}

/*
 * 测试接口,得到value内容
 * arg      : 传入的参数
 *          : 返回得到的结果
 */
const void* 
getvalue(void* arg) {
    const char* key = "heoo.c getvalue";
    printf("%s - %sn", key, (char*)arg);
    return key;
}

main.c

#include <stdio.h>
#include "heoo.h"

// 测试逻辑主函数
int main(int argc, char* argv[]) {
    // 简单的打印数据
    printf("getkey => %sn", getkey());
    getvalue(NULL);
    return 0;
}

到这里可能认为有个别臃肿, 可是通晓为啥是至关重要的. 会让您对于动态库中度高上0.01分米的.哈哈.

先让地点代码跑起来.

gcc -g -Wall -o main.out main.c heoo.c

测验结果如下

图片 1

测量检验成功,那就从头静态库到动态库扩充之旅.

 

前言

从静态库提及来

率先参照下边编写翻译语句 

gcc -c -o heoo-getkey.o heoo-getkey.c
gcc -c -o heoo-getvalue.o heoo-getvalue.c

对于静态库创设本质正是打包. 所以用linux上二个 ar创设静态库压缩命令.详细用法能够看

  ar详细用法参照    protected]](

 那么大家初步制作静态库

ar rcs libheoo.a heoo-getvalue.o heoo-getkey.o

这正是说我们选择静态库实践编写翻译上面main.c 函数

gcc -g -Wall -o main.out main.c -L. -lheoo

运作的截图如下

图片 2

运作总体符合规律. 对于静态库编写翻译 轻便说美素佳儿下. ar 末尾的 rcs表示 替换创造和加多索引. 具体的看上边的网站.

前面gcc中 -L表示查找库的目录, -l代表搜索的 libheoo库. 还也可以有其余的-I表示查找头文件地点, -D表示加多全局宏.......

对此地点静态库编写翻译还也会有一种艺术如下

gcc -g -Wall -o main.out main.c libheoo.a

实行结果也是一致的.能够将 *.a 明白成四个 *.o合体.

好到那边前言就说完了.那大家起头说正题动态库了.

 

正文

动态库的营造和行使

动态库营造命名如下,依然以heoo.c heoo.h 为例

gcc -shared -fPIC  -o libheoo.so heoo.c

开首编写翻译代码 先介绍一种最轻易易行的多少类似上边静态库最后一种形式.

gcc -g -Wall -o main.out main.c ./libheoo.so

那边是显式编写翻译. 结果如下

图片 3

对于 上边编写翻译 动态库的时候假如 直接利用 libheoo.so. 比方

gcc -g -Wall -o main.out main.c libheoo.so

假定未有配置动态库路线, 查找动态库路线会出标题. 这里就不复现了(因为自己把条件调好了). 汇合会提交化解办法.

下边说libheoo.so 标准的采取格局

gcc -g -Wall -o main.out main.c -L. -lheoo

运维结果如下

图片 4

上面是个常见错误, 系统找不见动态库在那. 须求配置一下, 再编写翻译参照如下

export LD_LIBRARY_PATH="$LD_LIBRARY_PATH;./"
gcc -g -Wall -o main.out main.c -L. -lheoo

地方第一句话是在此时此刻会话层. 增多库查找路线,满含当前文件目录.这几个会话层关闭了就失效了. Linux上shell确实比较重大. 未来推行结果

图片 5

到这里动态库的也都终止了. 一切平常.

贰个精美淫技

问: gcc -l 链接二个库的时候,不过库中留存同名的静态库和动态库. 会链接到那多个库? 

透过上边的那么多测验应该领悟是动态库吧,因为运用动态库会报错.使用静态库未有事.

那么难题来了, 作者想选取静态库如何做.

-static

地点gcc 选项能够扶持我们强制链接静态库!

 

动态库的显示应用

到此处大致是主导了. 扯一点,那一个知识点在window也长期以来知识情形变了,设置变了.链接编写翻译显式加载都有的. 上面是重新操作的代码.

heooso.c

#include <stdio.h>
#include <dlfcn.h>

#define _STR_PATH "./libheoo.so"

// 显示调用动态库, 需要 -ldl 链接程序库
int main(int argc, char* argv[]) {
    const char* (*getkey)(void);
    const void* (*getvalue)(void* arg); 
    /*  
     * 对于dlopen 函数第二个参数
     * RTLD_NOW:将共享库中的所有函数加载到内存
     * RTLD_LAZY:会推后共享库中的函数的加载操作,直到调用dlsym()时方加载某函数
     */
    void* handle = dlopen(_STR_PATH, RTLD_LAZY);
    // 下面得到错误信息,是一种小心的变成方式,每次都检测一下错误是否存在
    const char* err = dlerror();

    if(!handle || err) {
        fprintf(stderr, "dlopen " _STR_PATH " no open! err = %sn", err);
        return -1; 
    }   

    getkey = dlsym(handle, "getkey");
    if((err = dlerror())){
        fprintf(stderr, "getkey err = %sn", err);
        dlclose(handle);
        return -2; 
    }   
    puts(getkey());

    //这种显式调用dll代码,很不安全代码注入太简单了
    getvalue = dlsym(handle, "getvalue");
    if((err = dlerror())){
        fprintf(stderr, "getvalue err = %sn", err);
        dlclose(handle);
        return -3; 
    }   
    puts(getvalue(NULL));

    dlclose(handle);
    return 0;
}

编写翻译代码

gcc -g -Wall -o heooso.out heooso.c -ldl

测量试验结果截图如下

图片 6

 运转总体平常. 成效是促成了.可是大家千万别这么用.否则照旧比较惊恐的.也是一种编制程序思路吧.前边

后记会写一个向后特别的插件机制. 大家能够目睹一下. 低价更加深切的询问Linux系统开拓.算是二个大约的

Linux运用插件技巧的小项目吧.

 

后记

  错误是难免的,接待嘲谑. 最终献上三个linux上哪些通过动态库运转时加载插件的案例.麻雀虽小,五脏俱全.

Makefile

CC = gcc 
DEBUG = -g -Wall
LIB = -ldl
RUNSO = $(CC) -fPIC -shared -o [email protected] $^
RUN = $(CC) $(DEBUG) -o [email protected] $^

#总的任务
all:libheoo.so libheootwo.so libheoothree.so main.out


#简单lib%.so生成
libheoo.so:heoo.c
    $(RUNSO)
libheootwo.so:heootwo.c
    $(RUNSO)
libheoothree.so:heoothree.c
    $(RUNSO)

#生成的主要内容
main.out:main.c
    $(RUN) $(LIB)

# 简单的清除操作 make clean
.PHONY:clean
clean:
    rm -rf *.so *.s *.i *.o *.out *~ ; ls -hl

heoo.h

#ifndef _H_HEOO
#define _H_HEOO

/*
 * 测试接口,得到key内容 
 *      : 返回key的字符串
 */
extern const char* getkey(void);

/*
 * 测试接口,得到value内容
 * arg      : 传入的参数
 *          : 返回得到的结果
 */
extern const void* getvalue(void* arg);

#endif // !_H_HEOO

heootwo.c

#include "heoo.h"
#include <stdio.h>

/*
 * 测试接口,得到key内容 
 *      : 返回key的字符串
 */
const char*  
getkey(void) {
    return "heootwo.c getkey";
}

/*
 * 测试接口,得到value内容
 * arg      : 传入的参数
 *          : 返回得到的结果
 */
const void* 
getvalue(void* arg) {
    const char* key = "heootwo.c getvalue";
    printf("%s - %sn", key, (char*)arg);
    return key;
}

heoothree.c

#include "heoo.h"
#include <stdio.h>

/*
 * 测试接口,得到key内容 
 *      : 返回key的字符串
 */
const char*  
getkey(void) {
    return "heoothree.c getkey";
}

/*
 * 测试接口,得到value内容
 * arg      : 传入的参数
 *          : 返回得到的结果
 */
const void* 
getvalue(void* arg) {
    const char* key = "heoothree.c getvalue";
    printf("%s - %sn", key, (char*)arg);
    return key;
}

 

main.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdbool.h>
#include <errno.h>
#include <unistd.h>
#include <dirent.h>
#include <dlfcn.h>

//塞入的句柄数
#define _INT_HND (3)
// 最多支持108个插件
#define _INT_LEN (108)
// 文件路径最大长度
#define _INT_BUF (512)

// 处理dll,并且将返回的数据保存在a[_INT_HND]中, 这个数组长度必须是
bool dll_add(void* a[], const char* dllpath);
// 处理指定目录得到结果塞入a中, nowpath为NULL表示当前目录
int dll_new(void* a[][_INT_HND], int len, const char* nowpath);
// 释放资源
void dll_del(void* a[][_INT_HND], int len);

/*
 * 动态加载机制
 */
int main(int argc, char* argv[]) {
    int idx, len, i;
    void* a[_INT_LEN][_INT_HND];

    // 当前目录下,处理结果
    len = dll_new(a, _INT_LEN, NULL);
    if(len == 0){ 
        fprintf(stderr, "感谢使用,没有发现合法插件内容!n");
        exit(1);    
    }   

    //数据展示
    puts("------------------------------ 欢迎使用main插件 ----------------------------------");
    for(i=0; i<len; ++i){
        const char* (*getkey)(void) = a[i][1];
        printf("    %d   =>  %sn", i, getkey());
    }   
    printf("    请输入 待执行的 索引[0, %d)n", len);
    if(scanf("%d", &idx)!=1 || idx<0 || idx >= len){
        puts("    fake 错误的命令,程序退出中!");
        goto __exit;
    }
    puts("   执行结果如下:");
    const void* (*getvalue)(void* arg) = a[idx][2];
    puts(getvalue(NULL));

__exit:
    puts("------------------------------ 谢谢使用main插件 ----------------------------------");
    dll_del(a, len);
    return 0;
}

// 处理dll,并且将返回的数据保存在a[_INT_HND]中, 这个数组长度必须是
bool
dll_add(void* a[], const char* dllpath) {
    const char* (*getkey)(void);
    const void* (*getvalue)(void* arg);

    void* handle = dlopen(dllpath, RTLD_LAZY);
    // 下面得到错误信息,是一种小心的变成方式,每次都检测一下错误是否存在
    const char* err = dlerror();

    if(!handle || err) return false;

    getkey = dlsym(handle, "getkey");
    if((err = dlerror())){
        dlclose(handle);
        return false;
    }

    //这种显式调用dll代码,很不安全代码注入太简单了
    getvalue = dlsym(handle, "getvalue");
    if((err = dlerror())){
        dlclose(handle);
        return false;
    }
    // 句柄, key, value, 协议订的
    a[0] = handle;
    a[1] = getkey;
    a[2] = getvalue;
    return true;
}

// 处理指定目录得到结果塞入a中, nowpath为NULL表示当前目录
int
dll_new(void* a[][_INT_HND], int len, const char* nowpath){
    int j = 0, rt;
    DIR* dir;
    struct dirent* ptr;
    char path[_INT_BUF];

    // 设置默认目录
    if(!nowpath || !*nowpath) nowpath = ".";
    // 打开目录信息
    if((dir = opendir(nowpath)) == NULL) {
        fprintf(stderr, "opendir open %s, error:%sn", nowpath, strerror(errno));
        exit(-1);
    }

    //挨个读取文件
    while(j<len && (ptr=readdir(dir))){
        //只处理文件,包含未知文件
        if(DT_BLK == ptr->d_type || DT_UNKNOWN == ptr->d_type){
            rt = snprintf(path, _INT_BUF, "%s/%s", nowpath, ptr->d_name);// 只有确实是 *.so 文件才去出去运行 
            if(rt>3&&rt < _INT_BUF&&path[rt-1]=='o'&&path[rt-2]=='s'&&path[rt-3]=='.') {
                // 添加数据 dao数组 a中
                if(dll_add(a[j], path))
                    ++j;
            }
        }
    }

    closedir(dir);
    return j;
}

// 释放资源
void
dll_del(void* a[][_INT_HND], int len) {
    int i=-1;
    while(++i < len)
        dlclose(a[i][0]);
}

终极运维截图

图片 7

 到此地二个小demo就竣工了. 关于Linux gcc上动态库插件开荒,深入分析完成.O(∩_∩)O哈哈~

引言 入眼陈诉linux上运用gcc编写翻译动态库的一些操作.并且对其浓密的案例解析. 最终介绍...

     注重呈报linux上利用gcc编写翻译动态库的局地操作.何况对其长远的案例解析.
谈到底介绍一下动态库插件本事, 让代码向后包容.关于linux上使用gcc基础编写翻译,

预编写翻译,编写翻译,生成机械码最终链接输出可实行文件流程参照上面.

  gcc编写翻译流程 

而本文着重是剖判动态库相关的知识点. 首先看必要运用的测验素材

 heoo.h **

#ifndef _H_HEOO
#define _H_HEOO

/*
 * 测试接口,得到key内容
 *      : 返回key的字符串
 */
extern const char* getkey(void);

/*
 * 测试接口,得到value内容
 * arg      : 传入的参数
 *          : 返回得到的结果
 */
extern void* getvalue(void* arg);

#endif // !_H_HEOO

 heoo-getkey.c 

#include "heoo.h"

/*
 * 测试接口,得到key内容
 *      : 返回key的字符串
 */
const char*
getkey(void) {
     return "heoo-getkey.c getkey";
}

 heoo-getvalue.c 

#include "heoo.h"
#include <stdio.h>

/*
 * 测试接口,得到value内容
 * arg      : 传入的参数
 *          : 返回得到的结果
 */
const void* 
getvalue(void* arg) {
    const char* key = "heoo-getvalue.c getvalue";
    printf("%s - %sn", key, (void*)arg);
    return key;
}

heoo.c 

#include "heoo.h"
#include <stdio.h>

/*
 * 测试接口,得到key内容
 *      : 返回key的字符串
 */
const char* 
getkey(void) {
    return "heoo.c getkey";
}

/*
 * 测试接口,得到value内容
 * arg      : 传入的参数
 *          : 返回得到的结果
 */
const void* 
getvalue(void* arg) {
    const char* key = "heoo.c getvalue";
    printf("%s - %sn", key, (char*)arg);
    return key;
}

main.c

#include <stdio.h>
#include "heoo.h"

// 测试逻辑主函数
int main(int argc, char* argv[]) {
    // 简单的打印数据
    printf("getkey => %sn", getkey());
    getvalue(NULL);
    return 0;
}

到这里恐怕以为有一些臃肿, 然则明白为何是少不了的. 会令你对于动态库中度高上0.01分米的.哈哈.

先让地方代码跑起来.

gcc -g -Wall -o main.out main.c heoo.c

测验结果如下

图片 8

测量试验成功,这就起来静态库到动态库扩展之旅.

 

前言

从静态库提及来

第一参照下边编写翻译语句 

gcc -c -o heoo-getkey.o heoo-getkey.c
gcc -c -o heoo-getvalue.o heoo-getvalue.c

对于静态库创立本质正是打包. 所以用linux上三个 ar创设静态库压缩命令.详细用法可以看

  ar详细用法参照  

 那么我们最初创设静态库

ar rcs libheoo.a heoo-getvalue.o heoo-getkey.o

那便是说大家应用静态库实践编写翻译下面main.c 函数

gcc -g -Wall -o main.out main.c -L. -lheoo

运营的截图如下

图片 9

运行总体不荒谬. 对于静态库编写翻译 简单说Bellamy(Bellamy)下. ar 末尾的 rcs表示 替换创立和增添索引. 具体的看上面的网址.

末尾gcc中 -L表示查找库的目录, -l表示找出的 libheoo库. 还大概有另外的-I表示查找头文件地方, -D表示加多全局宏.......

对于地点静态库编译还也可以有一种办法如下

gcc -g -Wall -o main.out main.c libheoo.a

施行结果也是大同小异的.能够将 *.a 精晓成多个 *.o合体.

好到这边前言就说完了.那我们开端说正题动态库了.

 

正文

动态库的创设和使用

动态库创设命名如下,仍旧以heoo.c heoo.h 为例

gcc -shared -fPIC  -o libheoo.so heoo.c

起来编写翻译代码 先介绍一种最简便易行的有个别类似上边静态库最终一种情势.

gcc -g -Wall -o main.out main.c ./libheoo.so

此地是显式编写翻译. 结果如下

本文由威尼斯国际官方网站发布于威尼斯国际官方网站,转载请注明出处:linux动态库编写翻译和选用详细剖判

关键词: