浅议VB6字节数组和字符串的相互转换.doc_第1页
浅议VB6字节数组和字符串的相互转换.doc_第2页
浅议VB6字节数组和字符串的相互转换.doc_第3页
浅议VB6字节数组和字符串的相互转换.doc_第4页
浅议VB6字节数组和字符串的相互转换.doc_第5页
已阅读5页,还剩1页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

浅议VB6字节数组和字符串的相互转换一、 前言 数据类型转换在编程中经常用到,VB6提供了一整套类型转换的函数。但是,在进行类型转换时,有时候仅仅依靠VB提供的函数是不能达到自己的目的的。因此,需要考虑用其他的方法来完成数据类型转换。本文仅谈VB6中字节数组和字符串的相互转换过程中应注意的问题及其解决办法。在Visual Basic中使用Byte数组主要是为了32位API函数的参数传递和函数的返回。在32位的Visual Basic版本中,字符串被假定为Unicode字符,其中每个字符占用两个字节。系统自动地将Unicode的两个连续字节转换成1个字节的ANSI字符。但是,如果该字符串包含二进制数据,其内容将变得很难理解。例如,一个汉字是两个字节,在Visual Basic 6.0中的长度就只是1,这将给我们计算单个汉字的国标码带来一些麻烦。有了Byte数组,这些问题就将迎刃而解。另外,Visual Basic中的字符串和C语言中的字符串有一些不同,本文将给出一个函数,把C字符串转换成Visual Basic字符串。二、 用Byte数组代替字符串Byte数组包含的是0255之间的ASCII码字符,它不会象字符串那样被系统作预处理。你可以在很多API函数中用来Byte数组代替字符串。例如,下面的代码中用GetSystemDirectory这个Windows API函数来取得Windows的系统路径。一共有两段代码,一段代码是传递一个字符串来存储函数返回的系统路径,另一段代码是传递一个Byte数组来代替字符串。为了更好地比较,两段代码的不同部分都用黑体标出。读者可以仔细比较这两段代码的差异,这样您会更深入地理解Byte数组和字符串的差别。把这两段代码的任何一段放入一个窗体中运行,但击窗体的空白区域,你将会在窗体中看到Windows的系统路径。下面是使用字符串的代码:Private Declare Function GetSystemDirectory Lib kernel32 Alias _GetSystemDirectoryA (ByVal lpBuffer As String, ByVal nSize As Long) As LongPrivate Sub Form_Click()Dim n As IntegerDim str As Stringstr = Space$(256)n = GetSystemDirectory(str, 256)str = Left$(str, n)Print strEnd Sub在上面这段代码中,字符串参数lpBuffer返回Windows的系统路径。在函数调用之前,将变量预定义成256个字符,并在函数返回时清除多余的字符。注意:在调用API函数之前,通常都需要预先定义一个字符串或者Byte数组以供API函数存储数据。应该养成这种良好的编程习惯。否则,你的程序有可能崩溃,甚至导致你的系统崩溃。下面是使用Byte数组的代码:Private Declare Function GetSystemDirectory Lib kernel32 Alias _GetSystemDirectoryA (ByRef lpBuffer As Byte, ByVal nSize As Long) As LongPrivate Sub Form_Click()Dim n As IntegerDim Buffer() As ByteDim strA as StringBuffer=Space$(256)n = GetSystemDirectory(Buffer(0), 256)strA=StrConv(Buffer,vbUnicode)strA = Left$(strA, n)Print strAEnd Sub不知道读者注意到没有,第二段代码中的GetSystemDirectory API函数的声明已经改变了。第一个参数的声明由一个ByVal字符串变成了一个ByRef的Byte数组,即由声明:ByVal lpBuffer As String变成了:ByRef lpBuffer As Byte传递字符串时,需要一个ByVal修饰符来把字符串缓冲区传递到API函数中,因为字符串变量实际上指示了字符串内容所在的内存地址。在C语言术语中,这代表了一个指向指针的指针。ByVal意味着被传递的是一个指向实际字符串内容的内存地址。而在传递Byte数组Buffer(0)时,使用ByRef修饰符来传递变量,它相当于传递了数组中第一个字节内容的地址。事实上,这两种结果是一样的。strA=StrConv(Buffer,vbUnicode)这行代码把Byte数组的二进制数据转换成一个合法的Visual Basic字符串。三、 Byte数组和字符串之间的赋值为了简化Byte数组和字符串之间的数据传递,允许你在任何动态Byte数组和任何字符串之间直接互相赋值。例如:Buffer=strAStrA=Buffer注意:当且仅当Byte数组是动态的,而不是固定大小时,你才可以把一个字符串直接赋给一个Byte数组。声明一个动态的Byte数组最简单的方法是在Dim语句中使用空参数,例如:Dim Buffer() as Byte当你把一个字符串赋给一个动态Byte数组时,数组中的字符数将是字符串的字符数目的两倍。这是因为Visual Basic中字符串使用Unicode,并且每个Unicode字符的实际大小是两个字节。当把一个ASCII字符转换成一个Byte数组时,数组中的另一个字节将是0。向Unicode的转换是将每个在缓冲区中的字符转换成2个字节,从而实际上加倍了存储在结果字符串的中字节数目,当你认为函数Len(strA)得到的尺寸大小和Unicode转换后的Ubound(Buffer)函数所返回的尺寸大小相同时,上述特点就不很明显了。但是,函数LenB(strA)确实返回一个2倍于Len(strA)返回值的数值。这是因为Len函数返回的是字符串中字符的数目,而LenB函数返回的是字符串中字节的数目。一个Unicode串的字符长度仅仅是该串中实际字节数目的一半,这是因为每个Unicode字符2个字节。四、 字符串转换成VB字符串当我们在VB中调用Win32 API函数时,如果函数的返回值是一个字符串,那一般有如下三种情况:1. 函数预先要求你提供一个有固定空间的字符串,以供存储函数的返回值。2. 函数的返回是一个以Null结尾的C字符串,而不是正规的VB字符串。3. Win32 API函数有时候会返回另一种类型的字符串。这种类型的字符串在单个缓冲区内保存了多个字符串值,每个值之间用Null隔开,结尾的是两个Null,一个Null是最后一个字符串值的结尾符,另一个Null是整个字符串的结尾符。这其实就是我们通常在C中遇到的字符串数组。第一种情况很好办,只无原则预先定义好一个空间足够大的字符串,然后把API函数的返回值赋于这个字符串就可以了。例如,如果你已经知道函数返回值最多不会走过256个字符,可以这样编码如下:Dim sAPIReturn as stringSAPIReturn=Space$(256)SAPIReturn=API_Function()对于第二和第三种情况,就必须把返回的C字符串成标准的VB字符串。下面这个函数CStringToVBString把一个以Null结尾的C字符串成VB字符串。Public Function CStringToVBString(psCString As String) As string参数psCString是一个待转换的C字符串函数返回Null左边所有的字符dim sReturn as stringdim iNullCharPos As IntegeriNullCharPos=InStr(psCString,vbNullChar)if iNullCharPos 0 thensReturn =left(psCString, iNullCharPos -1)elsesReturn =pscstringend ifCStringToVBString=sReturnEnd function下面这个过程把一个含有多个C字符串的缓冲区转换成一个字符串数组。Public Sub MultiCStringToStringArray(psMultiCString As String, psaStrings() As String)参数psMultiCString是待转换的多个C字符串参数psaStrings是返回的VB字符串数组,调用之前它必须是一个动态的空数组Dim iNullPos As IntegerDim iPrevPos As IntegerDim iIdx As Integer初始化字符串数组iIdx = 0ReDim psaStrings(0 To iIdx + 1)psaStrings(iIdx + 1) = DoiNullPos = InStr(iPrevPos + 1, psMultiCString, vbNullChar)If iNullPos iPrevPos + 1 Then把找到的C字符串赋值给字符串数组psaStrings(iIdx) = Mid$(psMultiCString, (iPrevPos + 1), (iNullPos - 1) - iPrevPos)iIdx = iIdx + 1ReDim Preserve psaStrings(0 To iIdx)iPrevPos = iNullPosElse找到了两个Null字符,去掉最后一个,然后退出ReDim Preserve psaStrings(0 To iIdx - 1)Exit DoEnd IfLoopEnd Sub当调用Win32 API函数时,使用这两个简单的函数,你可以消除很多冗余的代码,加快开发步伐。注意:当你为API的返回值预先分配字符串的空间时,一定不要忘了空间内必须包含Null结束符。另外,建议你在使用API时,最好对每个变量都进行声明,加上下面这句代码:Optio

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论