OLE2A 和 A2OLE

OLE2A转换变乱码的问题

 

正文

 

从网上找了个人的日志。它将的是OLE2T,我觉得应该差不多吧。

以下摘自章鱼花园博客

以前VC6的代码,从xml文件中读取属性文字(中文),返回BSTR类型,用OLE2T转换,然后显示,一直工作很正常,用来做本地化,比修改Res文件要方便的多。

同样的代码,在VS2003里面居然不能工作了,调试,发现BSTR返回正常,而通过OLE2T转换居然就全转成“???”了,晕。

做了无数尝试,发现用unicode编译就正常。可是原来的VC6项目不论是否Unicode编译都正常的,要把一个项目全部转成Unicode还是有相当工作量的,似乎也并没有必要。

寻找解决方法,看了一堆ATL的代码,ATL7提供了OLE2T的代替品COLE2T,用一个重载是COLE2T(bstr, codepage),发现给把第二个参数设置成CP_ACP,即ANSI code page就转换正常了。原来是codepage的问题,ATL搞了一大堆代码来获取转换用的codepage,看得人晕,而我要的只是ANSI code page。

继续找,终于找到了,ATL的代码通过一条宏定义进行了分支。如果定义了宏_CONVERSION_DONT_USE_THREAD_LOCALE,则对当前code page的请求简单返回CP_ACP,否则,就搞了一大堆代码从当前线程中查询。那么我们要做的就是定义这个宏就可以了。试了一下,果然如此

最终解决方案,在stdafx.h的所有包含文件前面加上:

#define  _CONVERSION_DONT_USE_THREAD_LOCALE



搞定

OLE2A的潜在危险

分类: Miscellaneous 2007-06-28 23:20 503人阅读 评论(0) 收藏 举报

我们经常会在OLE(W)字符和ANSI(A)字符间转换。通常都使用ATL所提供的OLE2A/W2A/A2W/A2OLE等宏。可是这些宏是ATL的老版本(3.0),它存在一个潜在的问题。因为这些宏使用了local变量(在栈上分配)。而对于不同的cpulocal variable的尺寸是有限制的,例如:对于x86,这一尺寸为4k字节,即: 1个内存page。因此,如果转换的字符串大于这一尺寸,就会出现stack overflow,引起程序崩溃。C++运行时的_chkstk函数将完成这一检测。

在新版(7.0)ATL中,已经提供了一系列新的宏,并且在堆上分配。比如:应该使用COLE2T来取代OLE2A等。除此以外,还提供了异常处理等新的特性。

MSDN 2003上,可以在下面的位置找到详细说明,或者输入OLE2A查看:

ms-help://MS.MSDNQTR.2003FEB.2052/vclib/html/_atl_String_Conversion_Macros.htm

12.   使用SafeArray问题
学会使用SafeArray也是很重要的,因为在ADO编程中经常要用。它的主要目的是用于automation中的数组型参数的传递。因为在网络环境中,数组是不能直接传递的,而必须将其包装成SafeArray。实质上SafeArray就是将通常的数组增加一个描述符,说明其维数、长度、边界、元素类型等信息。SafeArray也并不单独使用,而是将其再包装到VARIANT类型的变量中,然后才作为参数传送出去。在VARIANT的vt成员的值如果包含VT_ARRAY|...,那么它所封装的就是一个SafeArray,它的parray成员即是指向SafeArray的指针。SafeArray中元素的类型可以是VARIANT能封装的任何类型,包括VARIANT类型本身。  
使用SafeArray的具体步骤:
方法一:
  包装一个SafeArray:
(1).   定义变量,如:
  VARIANT   varChunk;
  SAFEARRAY   *psa;
        SAFEARRAYBOUND   rgsabound[1];
(2).   创建SafeArray描述符:
  uIsRead=f.Read(bVal,ChunkSize);//read   array   from   a   file.
  if(uIsRead==0)break;
  rgsabound[0].cElements   =uIsRead;
  rgsabound[0].lLbound   =   0;
  psa   =   SafeArrayCreate(VT_UI1,1,rgsabound);
(3).   放置数据元素到SafeArray:
  for(long   index=0;index <uIsRead;index++)                    
  {
    if(FAILED(SafeArrayPutElement(psa,&index,&bVal[index])))
      ::MessageBox(NULL, "出毛病了。 ", "提示 ",MB_OK   |   MB_ICONWARNING);
  }
  一个一个地放,挺麻烦的。
(4).   封装到VARIANT内:
  varChunk.vt   =   VT_ARRAY|VT_UI1;
  varChunk.parray   =   psa;
  这样就可以将varChunk作为参数传送出去了。

  读取SafeArray中的数据的步骤:
(1).   用SafeArrayGetElement一个一个地读
  BYTE   buf[lIsRead];
  for(long   index=0;index <lIsRead;index++)                  
  {                      
    ::SafeArrayGetElement(varChunk.parray,&index,buf+index);      
  }
  就读到缓冲区buf里了。
方法二:
  使用SafeArrayAccessData直接读写SafeArray的缓冲区:
(1).   读缓冲区:
  BYTE   *buf;
  SafeArrayAccessData(varChunk.parray,   (void   **)&buf);
  f.Write(buf,lIsRead);
  SafeArrayUnaccessData(varChunk.parray);
(2).   写缓冲区:
  BYTE   *buf;
  ::SafeArrayAccessData(psa,   (void   **)&buf);
  for(long   index=0;index <uIsRead;index++)                    
  {
    buf[index]=bVal[index];    
  }
  ::SafeArrayUnaccessData(psa);

  varChunk.vt   =   VT_ARRAY|VT_UI1;
  varChunk.parray   =   psa;

这种方法读写SafeArray都可以,它直接操纵SafeArray的数据缓冲区,比用SafeArrayGetElement和SafeArrayPutElement速度快。特别适合于读取数据。但用完之后不要忘了调用::SafeArrayUnaccessData(psa),否则会出错的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值