在vs2005的mfc程序中使用LabView的Slider控件,添加变量后,调用赋值成员函数
_variant_t var(50.99);
m_BatterySlide1.SetValue(var);
程序报错:
程序中断处:
case VT_VARIANT:
//VARIANT is always passed by ref
*pArg = *va_arg(argList, VARIANT*);
break;
查看Slider类的成员函数:
void SetValue(VARIANT propVal)
{
SetProperty(0x101, VT_VARIANT, propVal);
}
在网上查这个问题,发现
VC6的向导产生的封装类对VARIANT类型的属性的处理是传地址
void CAxCtrlAV::SetFoo(const VARIANT& propVal)
{
SetProperty(0x1, VT_VARIANT, &propVal);
}
VC7的向导产生的封装类对VARIANT类型的属性的处理是传值
void CAxCtrlAV::SetFoo(VARIANT propVal)
{
SetProperty(0x1, VT_VARIANT, propVal);
}
因为VC7的va_list/va_start的 行为改为替换栈层次而不是实际上保存参数的地址,对属性的访问最终会在COleDispatchDriver::InvokeHelperV中导致程序崩溃。解决该问题的方法是手工将VC7的类向导产生的封装类的定义和实现更改为VC6风格的,或者不使用VARIANT做为属性的类型。
链接:http://blog.csdn.net/jiangsheng/article/details/353978
按照此方法,修改SetVlaue成员函数为:
void SetValue(const VARIANT& propVal)
{
SetProperty(0x101, VT_VARIANT, &propVal);
}
再次执行程序,运行成功!
进一步思考,这样赋值仍不够直观,为Slider控件赋值,只需传float型变量,如果SetValue函数能直接接收float类型参数,则使用起来更加直观方便。
观察这样的函数,接收参数类型与SetProperty函数的第二个参数VT_VARIANT&