海康VisionMaster-全局变量与全局脚本

VM是海康机器人自主研发的机器视觉软件,将一系列的图像算法、逻辑工具和通讯协议等封装成图形化模块,致力于帮助客户提供快速搭建视觉应用、解决视觉难题,能够满足视觉定位、尺寸测量、缺陷检测以及信息识别等机器视觉应用。
转自:机器视觉技术交流社区

1 引言

全局是指一切从系统整体及其全过程出发的思想。这种全局概念同样应用在VM中,在VM的快捷工具条中有一组全局工具,比如相机管理、控制器管理、全局变量、通信管理、全局触发、全局脚本等等,这里主要介绍全局变量和全局脚本。
全局变量,有过编程基础的朋友们相信也不陌生,是一种可以被局部获取或设置的变量。在VM中,全局变量可以被模块的运行参数订阅(获取全局变量的值),也可以绑定模块的结果(设置全局变量的值)。另外,还有一些其它比较经典的应用,如接收通讯数据,流程之间数据交换,计数等等。
全局脚本,支持调用算法平台SDK的C#版本(VM3.x)。在VM中,全局脚本可以控制流程的运行,快捷工具条上的单次执行按钮(方案中的所有流程执行一次)和连续执行按钮(方案中的所有流程连续执行)被点击之后都会进入全局脚本函数中。全局脚本常见应用有:指定流程执行一次、多流程控制执行、模块参数配置、模块运行结果获取、全局变量设置、全局通信等等。

2 全局变量介绍

全局变量是在所有的函数外部定义的变量,是可以被本方案中所有流程调用或修改的变量,可自定义变量名称、类型和当前值,它在整个工程文件内都有效。单击图标可进行全局变量的配置。如图中所示,添加了一个变量,名称命名为var0,注释可不写,类型为float。
在这里插入图片描述
相关功能如下:

  • 添加变量:点击+添加变量后可新增全局变量。

  • 导入/导出:可以固定格式文件导入或导出全局变量信息。

  • 搜索:当全局变量较多时,可快速搜索。

  • 置顶/上移/下移:可对变量的位置进行上下调整。

  • 保存变量:可将当前设置的全局变量进行保存。

  • 输入来源:即设置全局变量的值,可通过对对应类型的模块结果数据进行绑定,只可以绑定一个,如下图所示。
    在这里插入图片描述

  • 目标输出:即获取全局变量的值,可通过对对应类型的模块运行参数数据进行绑定,可绑定多个,可以如下图所示。也可以双击某个模块打开参数配置,进而订阅全局变量。
    在这里插入图片描述

  • 初始化:开启的作用,则是接收通讯发送过来的固定格式的字符串,实现对全局变量的值的设置。如变量var0,当通讯发过来SetGlobalValue:var0=99,可以将该变量值设为99。
    在这里插入图片描述

3 全局脚本接口和示例

单击图标则可以全局脚本代码界面,菜单栏的功能分别为导入C#代码、导出C#代码、打开(导入)示例代码、打开工程目录、打开引用的程序集、保存代码,菜单栏下方则是C#编辑区,C#编程区中Init()函数为初始化函数,Porcess()为处理函数。
在这里插入图片描述

  1. 全局脚本接口
    全局脚本中可以使用的设置和获取接口如下所示包含全局变量、脚本连续运行事件间隔、通讯接口等等。其它的接口函数调用可以查看《VisionMaster算法平台SDK开发指南V3.x.chm》,路径:VisionMaster4.2.0\Development\V3.x\Documentations。
功能 函数方法 参数说明
获取全局变量int型 GetGlobalVariableIntValue(string paramName, ref int paramValue) 输入:变量名paramName;输出:变量值paramValue
获取全局变量float型 GetGlobalVariableFloatValue(string paramName, ref float paramValue) 输入:变量名paramName;输出:变量值paramValue
获取全局变量string型 GetGlobalVariableStringValue (string paramName, ref string paramValue) 输入:变量名paramName;输出:变量值paramValue
设置全局变量int型 SetGlobalVariableIntValue (string paramName, int paramValue) 输入:变量名 paramName,变量值paramValue
设置全局变量float型 SetGlobalVariableFloatValue (string paramName, float paramValue) 输入:变量名 paramName,变量值paramValue
设置全局变量string型 SetGlobalVariableStrignValue (string paramName, string paramValue) 输入:变量名 paramName,变量值paramValue
获取连续运行时间间隔 GetScriptContinusExecuteInterval ()
设置连续运行时间间隔 SetScriptContinusExecuteInterval (uint nMilliSecond) 输入:时间间隔nMilliSecond,单位ms
初始化全局通信 StartGlobalCommunicate ()
注册通信接收事件 RegesiterReceiveCommunicateDataEvent()
通信数据接收事件 UserGlobalMethods_OnReceiveCommunicateDataEvent(ReceiveDataInfo dataInfo) 输入:通讯信息dataInfo
通信发送数据 SendCommDeviceData(string data,int deviceID) 输入:待发送数据data,设备deviceID
  1. 全局脚本中示例
    在vm中的全局脚本示例中,保存了常见应用的示例代码,根据所需选择任意一个就是导入到当前方案中,随之可进行参考或修改代码。
    在这里插入图片描述

4 全局脚本引用和调试

  1. 添加引用
    在全局脚本界面,打开引用的程序集,根据需求进行程序集动态库的添加,仅支持C#程序集添加,到需要的第三方程序集路径下找到想要添加的.dll,点击打开即可添加,添加完成后在全局脚本中调用即可。

  2. 调试步骤
    在全局脚本界面,打开工程目录后就会自动跳转全局脚本所在的文件夹,选择.sln文件使用VS打开,设置断点并且重新生成,然后点击【调试】中的【附加到进程】,再选择GlobalScript.exe附加,最后在VM中运行方案的单次执行来查看是否能进入断点。小小提示:断点调试时,每次修改代码都需要重新编译,VS每次重新编译后,只有第一次运行会进入函数Init()。
    在这里插入图片描述

5 案例演示

需求:结合全局变量和全局脚本,完成通讯设置全局变量的值。此处则是通讯发送物理坐标,修改全局变量的值。
第一步,在VM通讯管理设置号通讯设备。

在这里插入图片描述

第二步,在全局变量中设置中对应的变量。
在这里插入图片描述

第三步,打开全局脚本中的示例-全局通信,适当修改代码。在全局脚本中根据输入的字符串进行分割设置对应的全局变量。

效果如下所示,实现通讯输入的点位信息赋值到全局变量中,那么其它模块将可以订阅全局变量的值。
在这里插入图片描述

完整代码如下所示:

using System;
using VM.GlobalScript.Methods;
using System.Windows.Forms;
using iMVS_6000PlatformSDKCS;
using System.Runtime.InteropServices;

/******************************
 * 示例说明: 接收全局通信模块数据示例
 *     前提: 全局通信模块中开启有通信设备
 * 控制逻辑: 1.接收来自全局通信模块接收到的数据
 *           2.如果接收到数据字符0,则执行流程1一次
 * ***************************************/
public class UserGlobalScript : UserGlobalMethods, IScriptMethods
{
    
    
    /// <summary>
    /// 初始化函数
    /// </summary>
    /// <returns>成功:返回0</returns>
    public int Init()
    {
    
    
        //二次开发SDK初始化
        InitSDK();
        //设置与全局通信模块的通信端口
        StartGlobalCommunicate();
        //注册通信数据接收事件
        RegesiterReceiveCommunicateDataEvent();

        return 0;
    }

    /// <summary>
    /// 运行函数
    /// 单次执行:该函数执行一次
    /// 连续执行:以一定时间间隔重复执行该函数
    /// </summary>
    /// <returns>成功:返回0</returns>
    public int Process()
    {
    
    
        MessageBox.Show("进入Process");
        //m_operateHandle 二次开发SDK操作句柄
        if (m_operateHandle == IntPtr.Zero)
        {
    
     return ImvsSdkPFDefine.IMVS_EC_NULL_PTR; }

        //默认执行全部流程,如果自定义流程执行逻辑,请移除DefaultExecuteProcess方法

        int nRet = DefaultExecuteProcess();

        return nRet;
    }
    /// <summary>
    /// 通信数据接收函数
    /// </summary>
    public override void UserGlobalMethods_OnReceiveCommunicateDataEvent(ReceiveDataInfo dataInfo)
    {
    
    
        if (dataInfo == null || dataInfo.DeviceData == null)
        {
    
     return; }
        //接收到的数据转成字符串
        string str = System.Text.Encoding.Default.GetString(dataInfo.DeviceData);
        MessageBox.Show(str);
        string[] strlist = str.Split(',');
        SetGlobalVariableFloatValue("X", Convert.ToSingle(strlist[1]));
        SetGlobalVariableFloatValue("Y", Convert.ToSingle(strlist[2]));
        SetGlobalVariableFloatValue("R", Convert.ToSingle(strlist[3]));

        //这里的deviceIndex和全局通信模块中的一致
        if (dataInfo.DeviceID == 1)
        {
    
    
            //解析收到的数据
            if (strlist[0] == "C")
            {
    
    
                //执行流程1 一次
                ImvsPlatformSDK_API.IMVS_PF_ExecuteOnce_V30_CS(m_operateHandle, 10000, null);
            }
        }
    }

    /// <summary>
    /// SDK回调函数
    /// </summary>
    public override void ResultDataCallBack(IntPtr outputPlatformInfo, IntPtr puser)
    {
    
    
        base.ResultDataCallBack(outputPlatformInfo, puser);
        ImvsSdkPFDefine.IMVS_PF_OUTPUT_PLATFORM_INFO struInfo = (ImvsSdkPFDefine.IMVS_PF_OUTPUT_PLATFORM_INFO)Marshal.PtrToStructure(outputPlatformInfo, typeof(ImvsSdkPFDefine.IMVS_PF_OUTPUT_PLATFORM_INFO));
        switch (struInfo.nInfoType)
        {
    
    
            //获取模块结果数据
            case (uint)ImvsSdkPFDefine.IMVS_CTRLC_OUTPUT_PlATFORM_INFO_TYPE.IMVS_ENUM_CTRLC_OUTPUT_PLATFORM_INFO_MODULE_RESULT:
                {
    
    
                    ImvsSdkPFDefine.IMVS_PF_MODULE_RESULT_INFO_LIST_P resultInfo = (ImvsSdkPFDefine.IMVS_PF_MODULE_RESULT_INFO_LIST_P)Marshal.PtrToStructure(struInfo.pData, typeof(ImvsSdkPFDefine.IMVS_PF_MODULE_RESULT_INFO_LIST_P));
                    break;
                }
            ///获取流程运行状态
            case (uint)ImvsSdkPFDefine.IMVS_CTRLC_OUTPUT_PlATFORM_INFO_TYPE.IMVS_ENUM_CTRLC_OUTPUT_PLATFORM_INFO_WORK_STATE:
                {
    
    
                    ImvsSdkPFDefine.IMVS_PF_MODULE_WORK_STAUS stWorkStatus = (ImvsSdkPFDefine.IMVS_PF_MODULE_WORK_STAUS)Marshal.PtrToStructure(struInfo.pData, typeof(ImvsSdkPFDefine.IMVS_PF_MODULE_WORK_STAUS));
                    break;
                }
            default:
                break;
        }
    }
}

6 总结

在推崇图形化交互的VM视觉平台里,尽量避免用户再去过多的编辑代码。全局变量则在VM中起到了很好的桥梁作用,然而全局脚本中所实现的示例代码功能,则是依靠VM自身的已有的功能也能实现,比如示例代码模块参数设置,用户则是直接双击模块就可以打开模块的参数配置窗口;示例代码中的全局通信接收数据,用户也可以用快捷工具条中的通讯管理实现(自带了接收事件进行协议解析)等,这些其实都可以不用代码实现,仅仅在相应窗口中进行配置即可。另外,根据经验,有以下两点需要注意:

  1. 无法进入到断点位置。建议调试过程中,打开工程目录后关闭脚本模块编辑界面,只使用vs修改代码或重新生成,然后附加进程。
  2. 除了使用断点查看变量值,还可以使用MessageBox.Show(string paramValue)函数来查看相关变量。

猜你喜欢

转载自blog.csdn.net/Lminmine/article/details/125415454