_bstr_t与BSTR

作者:无名 - 开源软件 -

问题:

BSTR   a   =   _bstr_t("a");

BSTR   b   =   _bstr_t("b");  

  CString   c   ;  

  c   =   a;  

  MessageBox(c);  

  c   =   b;  

  MessageBox(c);  

  为什么消息框中显示的都是   b  

  如果这样:  

    _bstr_t   bstr1("a");  

  BSTR   a   =   bstr1;  

  _bstr_t   bstr2("b");  

  BSTR   b   =   bstr2;  

  CString   c   ;  

  c   =   a;  

  MessageBox(c);  

  c   =   b;  

  MessageBox(c);  

  消息框中显示就对了!

Answer:

首先要明确,_bstr_t是对BSTR的封装。  

   

  Code   1:  

   

  BSTR   a   =   _bstr_t("a");  

  //这里,_bstr_t("a")只是临时对象,当它初始化(包含一个内部的BSTR)并赋值给BSTR   a之后,它就被撤销了。而它赋的值是BSTR的地址。  

  BSTR   b   =   _bstr_t("b");  

  //第二次,仍然产生一个临时对象,它也初始化...过程跟上面一样。而且,由于前面的_bstr_t对象已被撤销,所以它产生BSTR的地址也同上。  

  //所以,实际上a,b指向同一个地址。而该地址的内容被写了两次。  

   

  Code   2:  

   

  _bstr_t   bstr1("a");  

  BSTR   a   =   bstr1;  

  _bstr_t   bstr2("b");  

  BSTR   b   =   bstr2;  

  //这里的两个_bstr_t   bstr1,bstr2都是局部对象,同时存在。所以,他们内部包含的BSTR的地址不一样。这样分别赋值就没问题。

 

 

#if   defined(WIN32)   &&   !defined(OLE2ANSI)  

  typedef   WCHAR   OLECHAR;  

  #else  

  typedef   char   OLECHAR;  

  #endif  

  typedef   OLECHAR*   BSTR;  

   

  实际上BSTR就是直接指向字符串的首地址(双字节或者ANSI字符串)。  

  而BSTR之前的四个字节是该字符串的长度,由编译器生成的代码自动去读这个长度,而不是用"\0"来标识字符串的结束。

 

BSTR字符串有以下几个要点要强调一下:  

  1.一个BSTR字符串变量实际上是一个指针变量。它占用32bit即4个字节,就像其它的指针一样。而且,它指向一个Unicode格式的字符数组。但是,我们不能把字符串与BSTR字符串等同起来。我们必须用它自己确切的名字――“BSTR“。  

  2.一个BSTR字符串变量指向的字符串数组必须由4个字节的保留字开始(保存字符串数组的字节数而不是字符数),由2个空字符结束。  

  3.由于空字符在Unicode格式的字符串中的任何位置都有可能出现,所以,以空字符声明一个Unicode格式的字符串的结束并不合适。因此,在4个保留字中保存字符串的长度是至关重要的。  

  4.我们再强调一下,BSTR字符串指针实际指向的是Unicode格式的字符数组的首地址,而不是开头的4个字节。接下来我们就会看到,在这儿这样不厌其烦地强调BSTR字符串变量的特征是为了与马上就要解释的VC++中的string类型作个比较。  

  5.前4个字节的length记录的是字符数组的字节数(注意,不是字符数),包括结尾的空字节。因为数组是Unicode格式的,所以实际字符是length的一半。  

  在这儿强调一下,一个Unicode格式的空字符其实是占用2个字节的空间,而不是1个字节。在Unicode格式的数组中测试空字符时要当心这一点。  

  我们一般在惯例上说BSTR字符串“help”是“一个BSTR类型的字符串”。一般公认为,一个BSTR字符串变量指向的一个字符数组中包括至少两个空字符。  

  就Visual   Basic来说,BSTR字符串未尾的两个空字节可能没什么用处,但是对Win32来说,它们却是至关重要的。原因在于,Win32版本的Unicode   String(它称为LPWSTR)定义为指向一个空字符结尾的Unicode格式的字符串的指针。  

  从这个原因上解释BSTR字符串为什么要以空字符结尾就合情合理了。下面让我们讨论一下C++类型的string变量。  

  两句代码:  

  Dim   str   as   String  

  str=”help”  

  str表示的是一个BSTR字符串变量的名字,而不是一个Unicode格式的字符数组。换句话说,str是一个保存地址xxxx的变量的名字。  

  以下是个小小的实验,它表明Visual   Basic中string变量是指向字符数组的指针而不是字符数组。下面定义了一个结构,它的成员变量的类型是string。  

  Private   Tyep   utTest  

  astring   As   String  

  bstring   As   String  

  End   Type  

  Dim   uTest   As   utTest  

  Dim   s   as   String  

  s=”testing”  

  uTest.astring=”testing”  

  uTest.astring=”testing”  

  Debug.Print   Len(s)  

  Debug.Print   Len(uTest)  

  这几句代码的执行结果是:  

  7  

  8  

  对string变量来说,Len函数返回的是字符串数组的字符个数。所以7个字符的字符串“testing”返回7。对结构变量uTest来说,Len函数返回的是该结构占用的内存空间。所以返回值8清楚地表明了每一个BSTR变量在内存中占用4个字节。因为BSTR是一个Win32的指针!  

  C类型的LPSTR和LPWSTR   字符串  

  Visual   C++使用LPSTR和LPWSTR字符串。  

  LPSTR类型字符串的定义是:指向一个空字节结尾的ANSI格式的字符串数组的指针。但是,因为我们是以空字节的位置来判断LPSTR字符串的终止的,所以,在LPSTR中是不允许字符串中还有第二个空字节存在。同样,LPWSTR是一个指向空字节终止的Unicode格式的字符串的指针,它的中间也不允许有空字节存在。LPWSTR中的W指Wide,它是微软对Unicode的另一种说法。  

   

  可能我们也会碰到LPCSTR和LPCWSTR类型的字符串。其中的C表示Constant(常量)。这种字符串是不能被API函数修改的。除此以外,LPCSTR都与LPSTR相同。同理,LPCWSTR除不能修改外,其它的都与LPWSTR相同。  

  再说LPTSTR,LPTSTR一般都用在条件编译中,就象TCHAR一样。以下是一个例子代码:  

  #ifdef     UNICODE  

  typedef   LPWSTR   LPTSTR;             //   在Unicode下LPTSTR与LPWSTR是相同的  

  typedef   LPCWSTR   LPCTSTR;         //   在Unicode下LPCTSTR与LPCWSTR是相同的  

  #else        

  typedef   LPSTR   LPTSTR;               //在ANSI下LPTSTR与LPSTR是相同的  

  typedef   LPCSTR   LPCTSTR;           //在ANSI下LPTCSTR与LPCSTR是相同的  

  #endif  


IT人知识库 原文地址:http://www.itpeo.net/15310/3487418.html





标签

MFC下判断文件是否存在

BOOL IsFileExist(LPCTSTR lpFileName) {  &n... ...

Dll的编写与Dll的显示调用和隐式调用

Dll的编写:     现在新建的Dll工程中创建一个新的类,我在这个... ...

Lucene3.0之查询处理

1、 查询的三种方式 ①     顺序查询:简单,但查询效率低 ②&nbs... ...

Lucene3.0之索引

1 倒排索引     倒排索引是满足实际应用而设计的一种数据结构。倒排索引的每一个元... ...

android callable与runnable的区别

编写多线程程序是为了实现多任务的并发执行,使用线程或者线程池,Thread,ExcutorService.线程或... ...

rfedfre

经测试Google热气球WIFI上网速度和3G媲美

Google的“Project Loon”的项目是将热气球作为WIFI热点为地面提供上网服务,这个惊人的想法到底... ...

OpenERP的外贸企业ERP方案

一、系统功能 OpenERP是欧洲中小企业使用人数最多的ERP软件,是世界上排名第一的开源ERP软件。... ...

OpenERP快速入门书免费发布

     《OpenERP应用和开发基础》第一版、第二版分别于2009年底... ...

泛型和反射机制

泛型:由于类型转换的错误,Java虚拟机在编译时不会报错,而在运行时报异常,所以安全隐患非常明显,在集合类中添加对... ...

rfedfre

Image Filter

    给图片加上滤镜,实现图片处理各种效果,例如:旧照片处理、曝光处理、... ...

SoapObject解析

解析webservice返回的单个实体类可通用性使用: public static Map&l... ...

关于ExtJS4.1创建Window拖动时会出现两个相同的Window

今天测试项目时发现新开的窗口在拖动操作下会出现两个一样的Window,而且也没有任何报错,我的项目类似桌面应用,父... ...

rfedfre

智和信通SugarNMS无线微波网管解决方案

无线/微波设备网管 无线/微波设备网管软件,是基于智和网管平台,面向无线控制设备、无线接入设备、无线终端... ...

rfedfre

智和信通SugarNMS EPON网管解决方案

EPON网管软件介绍 EPON网管软件,是基于智和网管平台,面向GPON、EPON、XPON、OLT、O... ...

rfedfre

超棒的悬浮层式的页面元素注解jQuery插件 - Chardin.js

日期:2013-5-2  来源:GBin1.com 在线演示 还记... ...

rfedfre

基于javascript和canvas(画布)的开源HTML5应用开发框架 - iio Engine

日期:2013-5-5  来源:GBin1.com 随着HTML... ...

maven 打包的作用域

maven 打包时针对dependency的不同作用域进行了单独的配置. Maven依赖的Scop... ...

学院/专业/班级三级联动下拉菜单

/* * 学校学院班级三级联动  author:Renault  QQ:115233... ...

C++ STL编程轻松入门 3

2 牛刀小试:且看一个简单例程   2.1 引子   如果你是一个纯粹的实用主义者,也许一开始就可以... ...

rfedfre

OSChina 周二乱弹 —— 倒了八辈子煤了

各位 OSCer 早,刚刚出门吃早餐,走在大马路突然发现裤子穿反了,劳资果断倒着走~~!呵呵,深深地被自己的机智所... ...