【澳门葡京备用网址】linux下制作共享库,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 - %s\n", 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 - %s\n", key, (char*)arg);
    return key;
}

main.c

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

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

到那里也许认为有点臃肿, 但是明白为啥是少不了的. 会让你对于动态库中度高上0.0一分米的.哈哈.

先让地方代码跑起来.

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

到此处动态库的也都得了了. 一切平常.

3个秀气淫技

问: 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 = %s\n", err);
        return -1; 
    }   

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

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

    dlclose(handle);
    return 0;
}

编写翻译代码

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

测试结果截图如下

澳门葡京备用网址 6

 运维总体不荒谬. 功能是兑现了.不过我们千万别这么用.不然依旧相比较惊险的.也是1种编制程序思路吧.后边

后记会写三个向后十分的插件机制. 大家能够目睹一下. 福利更深远的垂询Linux系统开辟.算是3个简易的

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

 

后记

  错误是免不了的,应接嗤笑. 最后献上一个linux上怎么通过动态库运营时加载插件的案例.麻雀虽小,5脏俱全.

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 - %s\n", 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 - %s\n", 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   =>  %s\n", 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:%s\n", 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编写翻译动态库的1部分操作.并且对其深刻的案例分析.
最后介绍…

【澳门葡京备用网址】linux下制作共享库,linux动态编写翻译剖析。引言

接触linux时间十分长,总是认为底气不足,许多东西总是认为迷迷糊糊,其实是因为没找得到linux
C的两把钥匙: makefile和动态库、共享库。linux
C中差不多全部的顺序都以以库的样式提交,如要要铺排,必须驾驭怎么着修改makefile。最后生成的组件也将以.a可能.so的花样加载到越来越大的应用程序中。下边包车型客车篇章是斯科学普及里华嵌嵌入式培训焦点 李家凯先生的博客,小编做了一些修改。

    
重视讲述linux上应用gcc编写翻译动态库的有个别操作.并且对其深刻的案例分析.
终极介绍一下动态库插件本事, 让代码向后包容.关于linux上选择gcc基础编写翻译,

    
重视讲述linux上利用gcc编写翻译动态库的有个别操作.并且对其浓密的案例分析.
聊到底介绍一下动态库插件手艺, 让代码向后包容.关于linux上采纳gcc基础编写翻译,

Linux下静态库和动态库(共享库)的炮制与行使

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

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

Linux操作系统支持的函数库分为静态库和动态库,动态库又称共享库。linux系统有几个根本的目录存放相应的函数库,如/lib
/usr/lib。 

  gcc编写翻译流程 

  gcc编写翻译流程 

静态函数库: 

而本文着重是分析动态库相关的文化点. 首先看必要选用的测试素材

而本文器重是分析动态库相关的文化点. 首先看须要运用的测试素材

那类库的名字一般是libxxx.a;利用静态函数库编写翻译成的文本十分的大,因为整个函数库的兼具数据都会被重组进目标代码中,他的长处就一目驾驭了,即编写翻译后的施行顺序不须求外部的函数库辅助,因为具备应用的函数都早已被编译进可施行文件了。当然那也会成为他的欠缺,因为假如静态函数库退换了,那么你的主次必须另行编译,而且体量也十分大。 

 heoo.h **

 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
#ifndef _H_HEOO
#define _H_HEOO

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

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

#endif // !_H_HEOO

这类库的名字一般是libxxx.so,动态库又称共享库;相对于静态函数库,动态函数库在编写翻译的时候并从未被编写翻译进目的代码中,你的程序实行到相关函数时才调用该函数库里的相应函数,由此动态函数库所发出的可执行文件相当小。由于函数库未有被重组进你的顺序,而是程序运转时动态的报名并调用,所以程序的运作情形中务必提供对应的库。动态函数库的改换并不影响你的次序,所以动态函数库的进级相比较便利。而且一旦多个应用程序都要采取同一函数库,动态库就卓殊适合,能够削减应用程序的体量。

 heoo-getkey.c 

 heoo-getkey.c 

上边来介绍linux静态函数库的开创和使用: 

#include "heoo.h"

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

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

例程add.h add.c sub.h sub.c main.c: 

 heoo-getvalue.c 

 heoo-getvalue.c 

add.h 

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

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

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

#ifndef ADD_H 

heoo.c 

heoo.c 

#define ADD_澳门葡京备用网址 ,H 

#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 - %s\n", key, (char*)arg);
    return key;
}
#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 - %s\n", key, (char*)arg);
    return key;
}

int add(int x,int y); 

main.c

main.c

#endif 

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

// 测试逻辑主函数
int main(int argc, char* argv[]) {
    // 简单的打印数据
    printf("getkey => %s\n", getkey());
    getvalue(NULL);
    return 0;
}
#include <stdio.h>
#include "heoo.h"

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

add.c 

到此地大概感到有些臃肿, 然而清楚为啥是要求的. 会让你对于动态库中度高上0.01分米的.哈哈.

到那里只怕认为有点臃肿, 不过知道为啥是必备的. 会让你对于动态库中度高上0.0一分米的.哈哈.

#include <stdio.h>

先让地点代码跑起来.

先让地点代码跑起来.

#include “add.h” 

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

int add(int x,int y)

测试结果如下

测试结果如下

澳门葡京备用网址 8

澳门葡京备用网址 9

return (x+y);

测试完了,那就起来静态库到动态库扩大之旅.

测试完了,那就起来静态库到动态库扩展之旅.

 

 


前言

前言

sub.h 

从静态库提及来

从静态库聊起来

#ifndef SUB_H 

首先参照上面编写翻译语句 

第3参照上边编写翻译语句 

#define SUB_H 

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

int sub(int x,int y); 

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

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

#endif 

  ar详细用法参照  

  ar详细用法参照  

sub.c 

 那么我们开头制作静态库

 那么我们初阶创设静态库

#include <stdio.h>

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

#include “sub.h” 

那么大家利用静态库施行编写翻译上边main.c 函数

那么大家采用静态库实施编写翻译上边main.c 函数

int sub(int x,int y)

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

运行的截图如下

运作的截图如下

return (x-y);

澳门葡京备用网址 10

澳门葡京备用网址 11

运作总体平常. 对于静态库编写翻译 轻易说美赞臣下. ar 背后的 rcs表示 替换创制和加多索引. 具体的看上边的网站.

运营总体平常. 对于静态库编写翻译 轻易说爱他美(Aptamil)下. ar 末尾的 rcs表示 替换创制和加多索引. 具体的看上边的网站.


末尾gcc中 -L表示查找库的目录,
-l表示寻找的 libheoo库. 还有别的的-I表示查找头文件地方,
-D代表增添全局宏…….

后边gcc中 -L表示查找库的目录,
-l代表搜索的 libheoo库. 还有其它的-I表示查找头文件地方,
-D表示增加全局宏…….

main.c 

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

对于地点静态库编写翻译还有壹种方法如下

#include <stdio.h>

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

#include “sub.h” 

实施结果也是平等的.能够将 *.a 通晓成几个 *.o合体.

施行结果也是均等的.能够将 *.a 了然成多个 *.o合体.

#include “add.h”

好到那里前言就说完了.那大家初阶说正题动态库了.

好到此处前言就说完了.那大家起初说正题动态库了.

int main() 

 

 

正文

正文

int a,b;

动态库的构建和利用

动态库的构建和平运动用

a = add(1,2);

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

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

b = sub(10,5);

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

printf(“a=%d,b=%d\n”,a,b);

发轫编写翻译代码 先介绍壹种最简易的有点类似下边静态库最终壹种格局.

起先编译代码 先介绍1种最简便的有点类似上边静态库最终①种格局.

return 0; 

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

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

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


澳门葡京备用网址 12

澳门葡京备用网址 13

任由是静态函数库照旧动态函数库,都以由*.o目标文件生成。 

对于 上面编写翻译 动态库的时候纵然 直接利用 libheoo.so. 举个例子

对此 下面编写翻译 动态库的时候假使 直接行使 libheoo.so. 举例

所以先 gcc -c add.c 

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

gcc -c sub.c 

要是没有陈设动态库路线,
查找动态库路线会出难题. 那里就不复现了(因为本人把情状调好了). 会师会交到化解办法.

假定没有配备动态库路径,
查找动态库路线会出难题. 那里就不复现了(因为自身把条件调好了). 会合会提交消除办法.

生成add.o sub.o

上边说libheoo.so 规范的选用办法

下边说libheoo.so 标准的应用方法

静态函数库由ar命令成立 

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

本例:ar -cr libaddsub.a add.o sub.o

运维结果如下

运作结果如下

【注意那里的称谓是libaddsub,库都要以libxxx.a或许libxxx.so的花样命名】

澳门葡京备用网址 14

澳门葡京备用网址 15

-c create的意思 

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

地方是个常见错误, 系统找不见动态库在那. 须要安插一下, 再编写翻译参照如下

-r
replace的情趣,表示当插入的模块名早已在库中设有,则替换同名的模块。假使若干模块中有3个模块在库中不存在,ar展现2个荒谬消息,并不替换别的同有名的模特块。暗中同意的意况下,新的成员扩大在库的结尾处,能够行使其它任选项来改动扩大的岗位。 

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

到此静态函数库制造实现。 

上边第一句话是在脚下会话层. 增多库查找路线,包涵当前文件目录.这几个会话层关闭了就失效了. Linux上shell确实很首要.
以后实践结果

上边第一句话是在当下会话层. 增添库查找路线,包括当前文件目录.这几个会话层关闭了就失效了. Linux上shell确实很重大.
以往实行结果

使用格局:通过gcc -o main main.c -L.
-laddsub编写翻译main.c就会把静态函数库整合进main。 
【注意】上面包车型大巴库名称叫是怎么着,不是libaddsub

澳门葡京备用网址 16

澳门葡京备用网址 17

其中 

到此地动态库的也都停止了. 一切符合规律.

到那边动态库的也都得了了. 一切日常.

-L钦赐静态函数库的职位供查找,注意L前边还有’.’,表示静态函数库在本目录下搜索。 

3个精密淫技

三个秀气淫技

-l则钦命了静态函数库名,由于静态函数库的命名方式是lib***.a,其中的lib和.a忽略。 

问: gcc
-l 链接一个库的时候,然则库中留存同名的静态库和动态库. 会链接到那些库? 

问: gcc
-l 链接多少个库的时候,但是库中存在同名的静态库和动态库. 会链接到那么些库? 

依照静态函数库的特色,此处删除libaddsub.a后main照旧得以运作,因为静态库的始末早已结合进来了。 

通过地点的那么多测试应该明了是动态库吧,因为使用动态库会报错.使用静态库未有事.

经过上面包车型地铁那么多测试应该理解是动态库吧,因为运用动态库会报错.使用静态库未有事.

动态函数库的成立和动用 

那么难题来了, 小编想使用静态库怎么做.

那就是说难题来了, 作者想采用静态库怎么办.

gcc -shared -fpic -o libaddsub.so add.c sub.c 

-static

-static

-fpic:发生地点非亲非故代码

地点gcc 选项能够支持大家强制链接静态库!

下边gcc 选项能够扶持大家强制链接静态库!

-shared:生成共享库

 

 

用上述命令生成libaddsub.so 动态函数库。

动态库的显得采取

动态库的来得应用

gcc -o out main.c -L. -laddsub 【注意名称是怎么写的】

到那里基本上是重视了. 扯一点,那个知识点在window也壹律知识情状变了,设置变了.链接编译显式加载都有的. 上边是双重操作的代码.

到此地基本上是注重了. 扯一点,这一个知识点在window也同样知识环境变了,设置变了.链接编写翻译显式加载都有的. 上边是再度操作的代码.

此时还不可能立刻./out,因为在动态函数库使用时,会招来/usr/lib
/lib目录下的动态函数库,而此时我们转换的库不在里边。 

heooso.c

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 = %s\n", err);
        return -1; 
    }   

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

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

    dlclose(handle);
    return 0;
}
#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 = %s\n", err);
        return -1; 
    }   

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

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

    dlclose(handle);
    return 0;
}

最直接最简易的法门正是把libaddsub.so拉到/usr/lib 或/lib中去。 

编写翻译代码

编写翻译代码

还有1种艺术,要是libaddsub.so在/home/linux/addsub

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

export LD_LIBRARY_PATH=/home/linux/addsub:$LD_LIBRARY_PATH

测试结果截图如下

测试结果截图如下

其它仍是可以够在/etc/ld.so.conf文件里投入我们转移的库的目录,然后/sbin/ldconfig。 

澳门葡京备用网址 18

澳门葡京备用网址 19

/etc/ld.so.conf是非凡重大的3个索引,里面存放的是链接器和加载器找寻共享库时要检查的目录,私下认可是从/usr/lib
/lib中读取的,所以想要顺利运转,我们也得以把大家库的目录加入到这几个文件中并推行/sbin/ldconfig。

 运维总体通常. 功用是促成了.可是豪门千万别这么用.不然还是相比较危急的.也是一种编程思路吧.后边

 运维总体常常. 作用是落成了.可是豪门千万别这么用.不然还是相比危险的.也是一种编制程序思路吧.后边

 

后记会写1个向后10分的插件机制. 大家能够目睹一下. 有益越来越深切的精晓Linux系统开辟.算是三个简短的

后记会写多少个向后十分的插件机制. 我们能够目睹一下. 方便更加深远的刺探Linux系统开拓.算是3个差不离的

Linux运用插件才干的小项目吧.

Linux运用插件才具的小项目吧.

 

 

后记

后记

  错误是难免的,迎接揶揄. 最终献上五个linux上哪些通过动态库运营时加载插件的案例.麻雀虽小,伍脏俱全.

  错误是在所难免的,迎接嘲讽. 最终献上三个linux上怎样通过动态库运维时加载插件的案例.麻雀虽小,⑤脏俱全.

Makefile

Makefile

CC = gcc 
DEBUG = -g -Wall
LIB = -ldl
RUNSO = $(CC) -fPIC -shared -o $@ $^
RUN = $(CC) $(DEBUG) -o $@ $^

#总的任务
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
CC = gcc 
DEBUG = -g -Wall
LIB = -ldl
RUNSO = $(CC) -fPIC -shared -o $@ $^
RUN = $(CC) $(DEBUG) -o $@ $^

#总的任务
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

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
#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

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 - %s\n", key, (char*)arg);
    return key;
}
#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 - %s\n", key, (char*)arg);
    return key;
}

heoothree.c

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 - %s\n", key, (char*)arg);
    return key;
}
#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 - %s\n", key, (char*)arg);
    return key;
}

 

 

main.c

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   =>  %s\n", 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:%s\n", 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]);
}
#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   =>  %s\n", 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:%s\n", 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]);
}

末段运营截图

末尾运维截图

澳门葡京备用网址 20

澳门葡京备用网址 21

 到此地三个小demo就告竣了. 关于Linux
gcc上动态库插件开荒,剖析达成.O(∩_∩)O哈哈~

 到那边1个小demo就完工了. 关于Linux
gcc上动态库插件开辟,剖析实现.O(∩_∩)O哈哈~

相关文章

发表评论

电子邮件地址不会被公开。 必填项已用*标注

*
*
Website