当前位置:首页 > > 充电吧
[导读]因为我是做嵌入式开发的,每次设备程序更新后都需要修改上位机,并且多个上位机,修改起来特麻烦,又不想用C#主要是底层使用的是C语言,配置解析通信等在单片机里面写好后可以直接复制到C++中使用,比较方便,

因为我是做嵌入式开发的,每次设备程序更新后都需要修改上位机,并且多个上位机,修改起来特麻烦,又不想用C#主要是底层使用的是C语言,配置解析通信等在单片机里面写好后可以直接复制到C++中使用,比较方便,我使用VC++开发比较方便,但是资料少,因此折腾了几晚上.

每次新产品都需要配一个上位机,并且本地配置与远程配置都需要重新开放配置程序,因此就想办法把配置模块变为一个动态的控件,一次开发后续2个程序都可以同时使用,使用了很多种方法,最后还是使用反射方式.


一.首先新建一个窗体控件DLL

将需要的界面从源程序拷贝过来



//对外接口函数,所有参数均为Object ^类型


		//输入的配置,用于设置输入配置的缓冲区
public:void Object_SetInConfig(Object ^Parameter)
		{
			if(Parameter == nullptr)
			{
				System::Windows::Forms::MessageBox::Show("内存不足!","错误",
					System::Windows::Forms::MessageBoxButtons::OK,System::Windows::Forms::MessageBoxIcon::Error);
				return;
			}
			SetInConfig((void*)Convert::ToInt32(Parameter));	//调用函数设置输入的配置参数

		}

	   //检查参数是否合法
public:Object ^Object_XF_CheckParameter(void)
	   {
		   Object ^temp = gcnew Object;

		   temp = this->CheckParameter();
		   return temp;
	   }

	   //存储配置
public:Object ^Object_SaveConfig(Object ^Parameter)
	   {
		   Object ^temp = gcnew Object;

		   if(Parameter == nullptr)
		   {
			   System::Windows::Forms::MessageBox::Show("内存不足!","错误",
				   System::Windows::Forms::MessageBoxButtons::OK,System::Windows::Forms::MessageBoxIcon::Error);
			   temp = false;
			   return temp;
		   }
		   temp = this->SaveConfig((void*)Convert::ToInt32(Parameter));
		   return temp;
	   }

	   //获取配置文件大小
public:Object ^Object_GetConfigSize(void)
	   {
		   Object ^temp = gcnew Object;

		   temp = this->GetConfigSize();
		   return temp;
	   }




二.动态加载调用


//通过方法名称获得方法
		System::Reflection::Assembly ^assembly;
		System::Type^ type;
		Object ^obj;
		System::Reflection::MethodInfo ^XF_SetInConfig;				//显示读取的配置
		System::Reflection::MethodInfo ^XF_CheckParameter;			//参数无误
		System::Reflection::MethodInfo ^XF_SaveConfig;				//存储配置
		System::Reflection::MethodInfo ^XF_GetConfigSize;			//获取配置大小	
		System::Reflection::MethodInfo ^XF_DefaultConfig;			//加载默认
		//动态加载DLL文件
		void LoadDLL(String ^pDLL)
		{
			try
			{
				this->assembly = System::Reflection::Assembly::LoadFrom(USER_LIB.GetRunningDirectory()+"\device\"+this->pDevDLL);	//加载DLL
			}
			catch (System::IO::FileNotFoundException^ e)
			{
				System::Windows::Forms::MessageBox::Show("找不到依赖的设备配置文件: "+this->pDevDLL+" 程序无法继续运行!","程序发生错误",
					System::Windows::Forms::MessageBoxButtons::OK,System::Windows::Forms::MessageBoxIcon::Error);
				Application::Exit();	//程序退出
				return;
			}
			this->type = this->assembly->GetType("XF_RTU_N_V1_0_CONFIG.XF_RTU_N_V1_0_CONFIGControl");
			this->obj = this->assembly->CreateInstance("XF_RTU_N_V1_0_CONFIG.XF_RTU_N_V1_0_CONFIGControl");
			
			this->panel1->Controls->Add((System::Windows::Forms::Control ^)this->obj);	//将DLL的控件加载到panel1并显示
			this->PerformLayout();

			//通过方法名称获得方法
			this->XF_SetInConfig = this->type->GetMethod("Object_SetInConfig");				//显示读取的配置
			this->XF_CheckParameter = this->type->GetMethod("Object_XF_CheckParameter");	//参数检查
			this->XF_SaveConfig = this->type->GetMethod("Object_SaveConfig");				//存储配置
			this->XF_GetConfigSize = this->type->GetMethod("Object_GetConfigSize");			//获取配置大小
			this->XF_DefaultConfig = this->type->GetMethod("DefaultConfig");				//加载默认
		}



//使用反射调用函数

1.无参数,无返回函数调用最简单


this->XF_DefaultConfig->Invoke(this->obj, nullptr);

2.带返回参数的函数调用,此处返回的是bool类型



if((bool)this->XF_CheckParameter->Invoke(this->obj, nullptr) == true)//检查参数
			 {
				 System::Windows::Forms::MessageBox::Show("配置参数无误!","提示",
					 System::Windows::Forms::MessageBoxButtons::OK,System::Windows::Forms::MessageBoxIcon::None);
				 this->toolStripStatusLabel1->Text = "参数检查无误";
			 }

3.带参数的函数调用需要使用cli::array< Object ^>^


将指针转化为int类型传入到参数表d中,此处只有1个形参,因此为

cli::array< Object ^>(1),多个按照实际填写.



cli::array< Object ^>^  d = gcnew cli::array< Object ^>(1);
				 d[0] = (int)&RTU_Config;	//获取指针并转化为int
				 this->XF_SaveConfig->Invoke(this->obj, d);			//存储配置



三.本地配置上位机与其它程序实现统一,一次编写,2个地方均可以使用


本地配置程序加载的配置控件




远程后台加载的同样的控件

四.可实现同一个程序完成多个功能
同一个程序动态加载不同控件实现不同功能

五.通过与ini配置文件结合,可以在不修改程序代码的情况下增加新设备支持,类似于插件

[设备数量]
NUM=4

[设备类型]
TYPE0=XF-RTU(老版)
TYPE1=XF-RTU-N(标准版)
TYPE2=XF-RTU-N(双DTU版)
TYPE3=XF-RTU-M(精简版)

[设备说明]
INF0=第一代RTU
INF1=第二代低功耗RTU
INF2=第二代低功耗RTU
INF3=低功耗简版RTU口

[配置控件]
DLL0=XF_RTU_老版本.dll
DLL1=XF_RTU_N_V1_0_CONFIG.dll
DLL2=XF_RTU_N_V1_0_CONFIG.dll
DLL3=XF_RTU_N_V1_0_CONFIG.dll









本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭