variant_t与_bstr_t
----------部分资源来自网络,感谢作者,特此声明
1. short/long/float/double/char*可通过构造函数或直接赋值转换成_variant_t类型或_bstr_t类型。
_variant_t 可隐式转换为short/long/float/double,所以可以直接用=。
_bstr_t 可隐式转换成char*,所以也可以直接用=。
_variant_t 与 _bstr_t 可互相直接转换。
可以看看comutil.h类的实现 基本就一堆构造函数和重载了很多运算符 就会明白了
2. variant_t 和 _bstr_ BSTR(Basic STRing,Basic字符串)是一个OLECHAR*类型的Unicode字符串。它被描述成一个与自动化相兼容的类型。由于操作系统提供相应的 API函数(如SysAllocString)来管理它以及一些默认的调度代码,因此BSTR实际上就是一个COM字符串,但它却在自动化技术以外的多种 场合下得到广泛使用。图1描述了BSTR的结构,其中DWORD值是字符串中实际所占用的字节数,且它的值是字符串中Unicode字符的两倍。
在OLE、ActiveX和COM中,VARIANT数据类型提供了一种非常有效的机制,由于它既包含了数据本身,也包含了数据的类型,因而它可以实现各种不同的自动化数据的传输。下面让我们来看看OAIDL.H文件中VARIANT定义的一个简化版:
struct tagVARIANT {
VARTYPE vt;
union {
short iVal; // VT_I2.
long lVal; // VT_I4.
float fltVal; // VT_R4.
double dblVal; // VT_R8.
DATE date; // VT_DATE.
BSTR bstrVal; // VT_BSTR.
…
short * piVal; // VT_BYREF|VT_I2.
long * plVal; // VT_BYREF|VT_I4.
float * pfltVal; // VT_BYREF|VT_R4.
double * pdblVal; // VT_BYREF|VT_R8.
DATE * pdate; // VT_BYREF|VT_DATE.
BSTR * pbstrVal; // VT_BYREF|VT_BSTR.
};
};
显然,VARIANT类型是一个C结构,它包含了一个类型成员vt、一些保留字节以及一个大的union类型。例如,如果vt为VT_I2,那么我们可以从iVal中读出VARIANT的值。同样,当给一个VARIANT变量赋值时,也要先指明其类型。例如:
VARIANT va;
:: VariantInit(&va); // 初始化
int a = 2002;
va.vt = VT_I4; // 指明long数据类型
va.lVal = a; // 赋值
_variant_t 是一个用于COM的VARIANT类,它的功能与COleVariant相似。不过在Visual C++.NET的MFC应用程序中使用时需要在代码文件前面添加下列两句:
#i nclude "comutil.h"
#pragma comment( lib, "comsupp.lib" )
_bstr_t是是C++对BSTR的封装,它的构造和析构函数分别调用SysAllocString和SysFreeString函数,其他操作是借用BSTR API函数。与_variant_t相似,使用时也要添加comutil.h和comsupp.lib。
CString转换成BSTR
通常是通过使用CStringT::AllocSysString来实现。例如:
CString str("This is a test");
BSTR bstrText = str.AllocSysString();
…
SysFreeString(bstrText); // 用完释放
BSTR转换成CString
一般可按下列方法进行:
BSTR bstrText = ::SysAllocString(L"Test");
CStringA str;
str.Empty();
str = bstrText;
关于数据类型转换由于COM对象是跨平台的,它使用了一种通用的方法来处理各种类型的数据,因此Cstring 类和COM对象是不兼容的,我们需要一组API来转换COM对象和C++类型的数据。_vatiant_t和_bstr_t就是这样两种对象。它们提供了 通用的方法转换COM对象和C++类型的数据。
3.下面是我自己的几段代码(vc2005中调试成功)
写入:
UpdateData();
try
{
// 写入各字段值
m_pset->AddNew();
m_pset->PutCollect(long(0),_variant_t(m_name));
m_pset->PutCollect(long(1),_variant_t(m_pass));
m_pset->Update();
MessageBox(_T("插入成功!"),_T("提示"),MB_OK);
}
catch(_com_error *e)
{
MessageBox(e->ErrorMessage(),_T("数据库读取错误"),MB_OK);
}
数据读取:
_try
{
_variant_t vfield;
vfield=m_pset->GetCollect(long(0));
m_name=vfield.bstrVal;
vfield=m_pset->GetCollect(long(1));
m_pass=vfield.bstrVal;
/*
m_name=(LPCSTR)_bstr_t(m_pset->GetCollect("UserName"));
m_pass=(LPCSTR)_bstr_t(m_pset->GetCollect("PassWord"));
*/
//UpdateData(false);
}
catch(_com_error&e)
{
MessageBox(e.ErrorMessage(),_T("数据库读取错误"),MB_OK);
}