linux的进程与库之间的通信两种方式

文章目录

  • 前言
  • 一、进程A与算法库b的通信方式之一:动态dlopen加载算法库b,编译的时候是需要加载该头文件就可以,无需连接该算法库b
    • 具体的实施细节:
  • 二、进程A与算法库b的通信方式之二:进程A编译的时候连接上算法库b和该头文件
    • 具体的实施细节:
  • 总结


前言

像平常主要是做视觉算法开发的,有时候算法效果实现了,但是要想要用产品上需要工程的实践来串联起算法和整个产品的连接关系。大部分AI的产品,算法一般实现之后,对外只会留两个接口:1)输入图像、激光雷达、Imu等数据;2)输出算法的结果:机器人的状态估计位置、速度、方向、图像的检测推理后筛选的坐标等;但是这仅仅是视觉算法内部的计算,产品它需要与其它模块交互,像与控制器交互、与显示的app交互;算法与外部模块的交互就会涉及到,通信链路的工程化,你的数据要通过协议传给接受方,所以涉及到自己的算法如何与自己通信进程进行交互问题,下面列些自己常用的两种进程与算法库的通信交互方法,各有利弊。


一、进程A与算法库b的通信方式之一:动态dlopen加载算法库b,编译的时候是需要加载该头文件就可以,无需连接该算法库b

具体的实施细节:

1、构建进程A和算法库b之前的一个公用头文件,必须需要的三个函数以及两个数据结构、一个是进程传给算法库的数据结构,一个是算法库传给进程的数据结构
2、定义三个函数:初始化函数、进程A回调算法库b数据函数、算法库b回调进程A的数据函数
3、具体的头文件.h和具体的.cpp实现
4.公共头文件

公共头文件,代码如下:

//从进程A中获取,算法库b开关
typedef struct
{//enableState 为0 不开启算法库b; 为1开启算法库bint enableState;
}StateFromA;//发送给进程A,算法库b内部运行的状态
typedef struct
{//sendState 1算法库b开启但未执行(检测)、2算法库b执行中(检测)、3算法库b执行完成 4、算法库b失败int sendState;
}StateToA;//typedef int (*b_flyCtrl_cb_func) (void* pData,  int len);
//typedef int (*b_PodCtrl_cb_func) (void* pData,  int len);
typedef int (*b_SendState_cb_func) (void* pData,  int len);
typedef int (*b_getbStateFromA_fn)(StateFromA* State);
typedef int (*b_init_fn)(b_flyCtrl_cb_func cb_func1,b_PodCtrl_cb_func cb_func2,b_SendState_cb_func cb_func3);//进程A调用算法库b检测算法库的初始化函数,并将算法库b状态和控制参数的回调函数的指针传入算法库中,
extern "C"
//进程A可以同时实现三个回调函数,将算法库的内部数据回调出去;
具体实现是在进程A中实现该三个函数,算法库b只需要调用typedef新定义的三个回调函数的指针函数别名就可以,把相应需要回调的函数结构数据填写进去就可以实现
//int b_init(b_flyCtrl_cb_func cb_func,b_PodCtrl_cb_func cb_func2,b_SendState_cb_func cb_func3);
int b_init(b_SendState_cb_func cb_func3);//进程A调用,算法库b获取进程A是否开启算法库b的状态信息
extern "C"
int b_getbStateFromA(StateFromA* bState);

算法库b的.cpp,代码如下:

	//从进程A获取算法b开关状态信息,结构体StateFromA* g_bStateInfoFromA = NULL;//进程A从算法b获取算法b内部的运行状态信息,结构体StateToA* g_bStateInfoFromb = NULL;g_bCtrl_cb ;  //通过该控制1回调函数指针,将控制1的结构体数据传给进程Ag_bCtrlPod_cb ;//通过该控制2回调函数指针,将控制2的结构体数据传给进程AStateFromA  g_ASendState_cb;//通过该进程A回调函数指针,将算法库b状态的结构体数据传给进程Aextern "C"
//int b_init(b_flyCtrl_cb_func cb_func,b_PodCtrl_cb_func cb_func2,b_SendState_cb_func //cb_func3);//进程A的回调函数入口,将回调函数指针通过初始化函数传入到算法库中
int b_init(b_SendState_cb_func cb_func3);
{//g_bCtrl_cb = cb_func1;  //通过该控制1回调函数指针,将控制1的结构体数据传给进程A//g_bCtrlPod_cb = cb_func2;//通过该控制2回调函数指针,将控制2的结构体数据传给进程Ag_bSendState_cb = cb_func3;//通过该进程A回调函数指针,将算法库b状态的结构体数据传给进程A//算法库b,输入数据结构体申请内存空间//给全局算法库b的开启状态结构体申请内存空间,使用函数指针的别名来定义全局算法b是否能够开启状态的结构体b_SendState_cb_func g_ASendState_cb;
return 0
}//相当于算法库b的输入口,获取了外部的状态
//算法库b获取进程A中是否需要开启算法b的状态,进程A只需要将算法b需要的结构体数据填写进去,算法b库这边的函数就会自动的更新该算法b的开启状态
extern "C"
int b_getbStateFromA(StateFromA* bState);
{if (StateToA== NULL)//所以进程A先调用,算法库b的初始化函数,里面会给指针申请内存{fprintf(stderr, "%s %d: ERROR! bStateFromb is NULL, return FAILURE.\n", __FUNCTION__, __LINE__);return -1;}else{//enableState: 为0 不开启算法库b; 为1开启算法库bg_bStateInfoFromA ->enableState = bState->enableState;return 0;}
}int main()
{//相当于算法库b的输出口,发出去了了算法库b自己的运行状态	//发送算法b内部运行状态给进程A,通过A的回调函数接口(别名)去实现memset(g_bStateInfoFromb , 0, sizeof(StateToA));//发送算法b的内部运行状态之前,先			      结构体数据都清零//调用进程A的回调函数接口(回调函数别名),将算法库b内部的运行状态传给进程Ag_bSendState_cb((void *)g_bStateInfoFromb , (int)sizeof(StateToA)); 
}

进程A的.cpp,代码如下:

int ALibLoad(struct *ip)
{//动态装载算法库b动态库 void *bHandle = dlopen("/usr/lib/libb.so",RTLD_LAZY);if(bHandle ==NULL){printf("dlopen失败:%s.",dlerror());return -1;}ip->bInitFuncPtr     = (b_init_fn)dlsym(bHandle , "b_init");if( NULL == ip->bInitFuncPtr ){        pritnf("dlsym b_init 失败:%s.",dlerror());return -1;}ip->bGetStateFuncPtr           = (b_getbStateFromA_fn)dlsym(bHandle , "b_getbStateFromA");if( NULL == ip->bGetStateFuncPtr           ){        printf("dlsym b_getbStateFromA失败:%s.",dlerror());return -1;}
}
int AInit(struct *ip)
{//进程A回调算法库b运行状态结构体初始化memset(&ip->bGetStateFuncPtr, 0x00, sizeof(b_getbStateFromA));if (sem_init(&g_sembSendStateEvent, 0, 0)){binocularlog("g_sembSendStateEventinit failed, %d: %s", errno, strerror(errno));return -1;}//将算法库b的初始化函数调用起来
//ip->bInitFuncPtr((b_flyCtrl_cb_func)bCtrlCbFun,(b_PodCtrl_cb_func)bPodCtrlCbFun,(b_SendState_cb_func)bStateCbFun);
ip->bInitFuncPtr((b_SendState_cb_func)bStateCbFun);
}//进程A回调算法库b的运行状态函数实现
int bStateCbFun(void* pData,  int len)
{int rtn_cb = -1;StateToA*curbStateInfo = (StateToA*)pData;StateToA*tmpbStateInfo = new StateToA;memcpy(tmpbStateInfo , curbStateInfo , sizeof(StateToA));g_bSendStateDataQueue.push(tmpbStateInfo );sem_post(&g_sembSendStateEvent);rtn_cb = 0;return rtn_cb;
}

二、进程A与算法库b的通信方式之二:进程A编译的时候连接上算法库b和该头文件

具体的实施细节:

1、定义好进程A和算法库b使用的共同头文件
2、该头文件是以类的封装形式展现出来,包括类的初始化函数、类的输入数据接口、类的输出数据接口
3、具体实现如下
公共头文件,代码如下:

//从进程A中获取,算法库b开关
typedef struct
{//enableState 为0 不开启算法库b; 为1开启算法库bint enableState;
}StateFromA;//发送给进程A,算法库b内部运行的状态
typedef struct
{//sendState 1算法库b开启但未执行(检测)、2算法库b执行中(检测)、3算法库b执行完成 4、算法库b失败int sendState;
}StateToA;class B 
{B();~B();
public:int get (StateFromA bState);int output(StateToA &curbState);
private:
StateFromA  m_bState;
StateToA  m_curbState;}

算法库b的.cpp,代码如下:

int B::get (StateFromA bState)
{m_bState = bState;//外部数据传给算法库b的内部私有变量m_curbState = 1;return 0;
}int B::output(StateToA &curbState)
{curbState = m_curbState;//将算法库b运行状态传输给外部调用者return 0;
}

进程A的.cpp,代码如下:

int main ()
{StateFromA g_bState;//定义进程开启算法库b的结构体数据变量g_bState = 0;//开启算法库bStateToA g_curbState;//定义进程获取算法库b的运行状态结构体数据变量B bobject;bobject.get (g_bState);//发送给算法库b的开启标志位bobject.output(g_curbState);//获取算法库b内部的运行状态return 0;
}

总结

进程与库通信两种方式的优缺点 1、进程与库编译相互依赖, 优点:库里面可以自行使用类的函数来完成整理系统的处理 缺点:库的头文件把整个类的处理函数都暴露给了进程,外部可以看到库的详细信息、编译需要依赖库的存在 2、进程编译不依赖库 优点:进程编译不依赖库的存在、库的内部处理函数不会暴露给进程 缺点:进程与库需要操作加载库,使用函数指针来调用库的数据,会形成很多单独的通信接口

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/256683.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Day03——类、值和对象

1.js数字-NaN和Infinity 1、在js中,有一种特殊的数值,叫做NaN(Not a Number),表示本来要返回数值的操作却未返回数值的情况,例如0除以0的操作,在其它语言中会报错误或异常,而在js中&…

ASIHTTPRequest详解 [经典3]

大文件断点续传 0.94 以后支持大文件的断点下载,只需要设置: [ request setAllowResumeForFileDownloads:YES ]; [ request setDownloadDestinationPath:downloadPath ]。 就可以了。 ASIHTTPRequest会自动保存访问过的UR&#xf…

python矩阵相关操作

1.创建mxn的矩阵 import numpy as np from numpy import *x np.array([[2,1],[1,2],[2,2])#创建了一个3x2的数组 x np.mat(x)#将数组转化为矩阵2.数组与矩阵的转化 from numpy import * import numpy as np#数组转化为矩阵 matrix np.mat(array)#矩阵转化为数组 array ma…

关于F5的一些基础话题

负载均衡器通常称为四层交换机或七层交换机。四层交换机主要分析IP层及TCP/UDP层,实现四层流量负载均衡。七层交换机除了支持四层负载均衡以外,还有分析应用层的信息,如HTTP协议URI或Cookie信息。一、F5配置步骤:1、F5组网规划(1)…

传输协议上的字节解析问题

1、 低字节在前,高字节在后 2、0x01高字节 , 0x00 低字节,如何组合成一个字节 3、0x01 0000 0000 0000 0001 4、0x00 0000 0000 0000 0000 5、按照组合规则,低字节在前,高字节在后组合成的新数据 6、0x01 0x00 0x0…

POJ C++程序设计 编程题#7:字符串排序

编程题&#xff03;7&#xff1a;字符串排序 来源: 北京大学在线程序评测系统POJ (Coursera声明&#xff1a;在POJ上完成的习题将不会计入Coursera的最后成绩。) 总时间限制: 1000ms 内存限制: 1024kB 描述 请按照要求对输入的字符串进行排序。 #include <iostream> #inc…

Delphi格式化函数Format、FormatDateTime和FormatFloat详解

转自&#xff1a;http://outofmemory.cn/code-snippet/7631/Delphi-format-hua-function-Format-FormatDateTime-FormatFloat-explainindetail 1.Format 根据指定所需要的格式&#xff0c;格式化字符串。 原型&#xff1a; function Format(const Format: string const Args: a…

OSPF地址汇总配置

配置过程在R1上&#xff1a;1. 给 Loopback 0 接口分配IP。2. 给ethernet 0/0 接口分配 IP3. 开启OSPF路由协议&#xff0c;进程号为110&#xff0c;并宣告直连网段。在R2上&#xff1a;1. 给ethernet 0/0 接口分配 IP2. 开启OSPF路由协议&#xff0c;进程号为110&#xff0c;并…

论文写作与学术规范课堂笔记01——4.30

论文质量的判断标准 &#xff08;原创&#xff09;创新性&#xff1a;观点、方法、材料 写作质量&#xff1a;逻辑性、语言清晰、数学数据、格式、数据可视化 论文写作的提升路径 模仿 经验

requirejs学习之-- 初始化(一)

为了规范在项目中使用的javascript代码&#xff0c;我们使用了requirejs框架。 初始阶段&#xff0c;我们在按钮的点击事件中调用创建的模块&#xff0c;代码如下&#xff1a; function button_click() {_this this;var args _this["Command_Params"] || (_this.co…

ROS系统中的多个版本Boost问题

1、删除多余的boost版本&#xff0c;只需要删除该版本的libboost*库以及Boost的头文件&#xff0c;不需要使用sudo apt-get rm --pugre libboost-dev sudo apt-get armove libboost-dev;这种卸载会把很多Boost依赖的库删掉&#xff1b;想要单纯的只删除当前版本的Boost库&#…

iOS的通知

首先,什么是通知呢,通知跟代理的功能是一样的,都是传值,调方法,但是我个人觉得通知比代理还是要简便的,而且功能更强大,如果你代理弄懂了,那么通知也就非常容易理解了,我认为通知比代理书写更容易,而且功能更强大.代理是一对一传值,但是通知可以一对多或多对多,好了不多说了给大…

ValueStack基础:OGNL

ValueStack基础&#xff1a;OGNL 要了解ValueStack&#xff0c;必须先理解OGNL(Object Graphic Navigatino Language)&#xff01; OGNL是Struts2中使用的一种表达式语言&#xff0c;它可以用于JSP的标签库中&#xff0c;以便能够方便的访问各种对象的属性&#xff1b;它用于界…

4.H - 组合

题目连接&#xff1a;http://acm.hust.edu.cn/vjudge/contest/125308#problem/H 题目大意&#xff1a;下面是一个二人小游戏&#xff1a;桌子上有M堆扑克牌&#xff1b;每堆牌的数量分别为Ni(i1…M)&#xff1b;两人轮流进行&#xff1b;每走一步可以任意选择一堆并取走其中的任…

结构体指针需要申请指针内存,结构体对象不需要申请对象内存

struct frame_info { char* data;//图像数据 int bufsize;//图像大小 }&#xff1b; 1、struct frame_info* finfoDT new struct frame_info; finfoDT->data new 1280*720 2、struct frame_info finfoDT new struct frame_info; finfoDT.data new 1280*720

图论——连通图

Tyvj 2059 元芳看电影 描述 神探狄仁杰电影版首映这天&#xff0c;狄仁杰、李元芳和狄如燕去看电影。由于人实在是太多了&#xff0c;入场的队伍变得十分不整齐&#xff0c;一个人的前面可能会出现并排的好多人。“元芳&#xff0c;这队伍你怎么看&#xff1f;”“大人&#xf…

linux-ftools查看Linux 的cached里面有哪些内容

最近&#xff0c;公司有几台java服务器经常出现can not allocate memory 的情况。导致SSH登录失败。 因此&#xff0c;有必要做点什么。 我们可以使用linux-ftools查看Linux 的cached里面有哪些内容&#xff08;工具: https://code.google.com/p/linux-ftools/&#xff09;&…

matlab常用函数——软件常用函数

一、软件操作函数 1)命令窗口函数: clc:清空命令窗口,使用向上箭头翻看命令。 open:打开文件,文本文件(*.doc),可执行文件(*.exe),图形文件(*.fig),超文本文件(*.html,*.htm),MATLAB数据库文件(*.mat),simulink模型文件*.mdl),MATLAB p文件(*.p),…

stanford corenlp的TokensRegex

最近做一些音乐类、读物类的自然语言理解&#xff0c;就调研使用了下Stanford corenlp&#xff0c;记录下来。 功能 Stanford Corenlp是一套自然语言分析工具集包括&#xff1a; POS(part of speech tagger)-标注词性NER(named entity recognizer)-实体名识别Parser树-分析句子…