href 带参数 打开exe_LabVIEW调用DLL中一、二级指针参数及打包exe运行异常的解决办法...

1. Why?

在实际项目中,经常会用到第三方提供的DLL(动态链接库),调用别人已经封装好的函数来完成项目任务,但是有时候会碰到一些带指针的参数,如 int * p; int ** dp; 相信对部分人来说,指针还是比较让人头疼的(包括我自己...),那如何在LabVIEW正确使用它们呢?最开始的我也是模棱两可,凑合着用,忍受着偶尔会出现的内存错误、无法顺利打包exe的尴尬,故借此次项目的应用深入摸索了一下相关内容,便有了这篇总结。

2. What?

此篇总结主要解决以下几个问题:

  1. 有关DLL中一级指针参数的使用;
  2. 有关DLL中二级指针参数的使用;
  3. 使用GetValueByPointer.xnode导致程序打包exe运行异常的问题;

3. How?

为了说明方便,我自己用C封了几个简单的函数,输出了一个111.dll文件用于测试,函数原型和定义如下图所示:

0ffea9403d9ced4f4b598d8fa9475eb6.png
688b168162e3b31971d74cd5ec01145a.png

1. 一级指针

一级指针相对还是比较容易的,也比较好理解,就是内存中的地址,只是需要注意它的数据类型,配置时要与函数原型声明中的参数类型一致,关于一级指针在labview中的Call Library Function Node中还是得到了很好的支持。以TestFunc3函数为例。

在程序框图面板中放置Call Library Funtion Node.vi , 双击打开配置面板:1. 选择DLL所在的路径>>2. 选择要调用的函数名称>>3. 选择调用约定(这里选用C);

e70b901d5c908ecd20b8f8ec7bc50855.png

进入参数配置面板,函数申明中有一个形参,是unsigned char * 类型,在左侧添加一个参数,然后Type选择数组类型(数组名称也就是首地址),然后数组存放的数据类型(unsigned char也就对应着这里的Unsigned 8-bit Integer),这里是一维的数组,默认的1不改,然后数组格式选用Array Data Pointer,并分配开辟的最小内存大小,这里设定为10,此时在下方可以看到函数原型已经配置成与DLL中的一样,基本上问题就不大了。

23af61da17e1c7f1beecdc9c31788a7c.png

运行,结果与预期的一样。

cb2ab63accdafb91e90f461ed088b2a5.png

如果内存大小没有去分配,是无法得到正确结果的,甚至会报错。将大小改为“”,运行出错。

ca75f107eb7901c8d911b7db2d616c93.png
035d92d6f1c28e22a1512c441b2d7162.png

内存大小除了可以在配置面板中指定,还可以在程序框图中通过初始化一个指定长度的数组去分配大小,显然这种更加灵活,如下图所示。

c400a0e04843c5c3216f1b50dea38f46.png

2. 二级指针

也就是指针的指针,相当于是二重地址,简单理解如下图,&A表示A在内存中的地址,&B表示B在内存中的地址。

1edc594b285d70e8051ae5eb257d9d99.png

在内存中开辟了一块空间存放字符串A“Hello LabVIEW!”,B是个一级指针,存放着A的地址,通过*(&A)便可以解析出字符串A,C是一个二级指针,存放着B的地址,通过二重解析*(*(&B))也可以获取到字符串A。

接下来看看如何在LabVIEW中区配置这个参数类型,LabVIEW中无法直接配置二级指针类型,但我们可以通过一级指针获取到实际数据的地址(&A),然后通过解析地址来获取相应的数据。

以TestFunc2函数为例。

2e015c8e02e28a4002773f7afdd079d5.png

添加一个形参cp,类型选Numeric,数据类型选择Signed 32-bit Integer, 以指针的形式传入。

a6fb8b46609aa2c9bd4d72be1752ff80.png

如果以上面理解的例子来看,通过这样配置之后,cp其实保存的是B的值,也就是变量A的地址&A,那离我们真正想拿到的数据字符串“Hello LabVIEW!”, 还差一步地址解析。

这时候就该GetValueByPointer这个工具登场了,在此路径下可以找到,以2012版本为例(D:Program Files (x86)National InstrumentsLabVIEW 2012vi.lib甥楴楬tyimportslGetValueByPointer)

传入指针参数,然后指定数据类型,这里是一个字符串,便可以获取到正确的结果。

如下图所示。

b0f43fab7d96622dbb75fd8aca4210d2.png

通过GetValueByPointer可以很方便的解析指针,使用也很简单。大家根据实际的数据类型去配置即可。

关于指针的解析,LabVIEW还提供了一个间接的函数,MoveBlock函数,这两种的一个区别是:

  • GetValueByPointer:使用简单,适应大部分数据类型,无法处理某些复杂的类型;
  • MoveBlock:级别较低,控制更多,可以处理任何数据类型;

从目前查看的资料来推测,MoveBlock函数应该是封装到LabVIEW中静态库中,调用方法如下:

  • 打开调用库函数节点,在路径中输入“LabVIEW”;
  • 此时函数名列表会更新出很多函数名,其中就有MoveBlock;

在参数面板,参考如下配置:

6b4ab705a33723b9fa616adb387dc022.png
81e82f1d8f7165fa425f7a1d99fe12c1.png
25b43bfedd5307e05d60b59c9d8e6732.png

我这里配置的adress传入方式是Value,并不是Pointer to Value, 原因就是通过之前介绍的TestFunc2的配置,返回的已经是一个Pointer了,所以这里选择值传入。然后指定目标数据类型和大小,运行变可以获取正确的结果“Hellow LabVIEW!”.

82d6303b92e90c1ec4b4dd99c8f96218.png

3. 打包exe异常

这里打包有问题的情况主要是使用GetValueByPointer,正常打包完成后,运行会发现不报错,但是无法正确得到期望的结果,或者出现如下图的错误。

21dceeebcdbe47a28fdce360fe4902f0.png

出现这些异常的原因就是GetValueByPointer.xonde根本没有正确执行,因为它调用了lvimptsl.dll中封装的GetValueByPointer函数,所以在打包的时候,我们要把lvimptsl.dll包含进去。

  • 把lvimptsl.dll包含到项目中;
  • 打包的时候,在Destinations中新建一个resource文件夹,然后始终包含,文件夹名resource不得修改
  • 在Source File Settings中,把它添加到支持路径resource中;
ad03cc85142cde52e9f39b16cb532a23.png
5c8e0fc7da8ff93a942c66d8fbba1c1d.png
4efde9d7f64ff08cc9e61de3177ac514.png

这样打包之后,就可以正常运行了。

40f44fd94f046962d3581e2034a77ab5.png

在摸索的过程中,参考很多大牛前辈的经验总结,十分感谢,附上几个有价值的参考链接:

https://knowledge.ni.com/KnowledgeArticleDetails?id=kA00Z0000019ZANSA2&l=zh-CN

https://forums.ni.com/t5/LabVIEW/LabVIEW-MoveBlock-and-getting-a-string-from-a-DLL-allocated-by/td-p/3311596?profile.language=zh-CN

https://forums.ni.com/t5/Developer-Center-Resources/Dereferencing-Pointers-from-C-C-DLLs-in-LabVIEW/ta-p/3522795?profile.language=zh-CN

以上内容,如有不对之处还请斧正,感激不尽。欢迎关注我的微信公众号“Retry的LabVIEW”,也希望能帮忙转发、分享到您的圈子,Retry需要您的支持,谢谢。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值