您的位置:首页 > Web前端

CString:Getbuffer和Releasebuffer的作用

2016-09-02 16:37 393 查看
首先看MSDN中的解释:
CString::GetBuffer 

LPTSTR GetBuffer( int nMinBufLength ); 

  throw( CMemoryException ); 

返回值:一个指向对象的(以空字符结尾的)字符缓冲区的LPTSTR指针。 

参数:nMinBufLength  字符缓冲区的以字符数表示的最小容量。这个值不包括一个结尾的空字符的空间
说明: 

此成员函数返回一个指向CString对象的内部字符缓冲区的指针。返回的LPTSTR不是const,因此可以允许直接修改CString的内容。 

如果你使用由GetBuffer返回的指针来改变字符串的内容,你必须在使用其它的CString成员函数之前调用ReleaseBuffer函数。 

在调用ReleaseBuffer之后,由GetBuffer返回的地址也许就无效了,因为其它的CString操作可能会导致CString缓冲区被重新分配。 

如果你没有改变此CString的长度,则缓冲区不会被重新分配。 

当此CString对象被销毁时,其缓冲区内存将被自动释放。 

注意:如果你自己知道字符串的长度,则你不应该添加结尾的空字符。但是,当你用ReleaseBuffer来释放该缓冲区时,你必须指定最后的字符串长度。如果你添加了结尾的空字符,你应该给ReleaseBuffer的长度参数传递-1,ReleaseBuffer将对该缓冲区执行strlen来确定它的长度。
CString对象在内存中用一个计数器来维持可用缓冲区的大小

void ReleaseBuffer( int nNewLength = -1 )
{
if( nNewLength == -1 )
{
nNewLength = StringLength( m_pszData );
}
SetLength( nNewLength );
}


很明显ReleaseBuffer的作用就是更新字符串的长度。 CString内,GetLength获取字符串长度并不是动态计算的,而是在赋值操作后计算并保存在一个int变量内的,当通过GetBuffer直接修改CString时,那个int变量并不可能自动更新,于是便有了ReleaseBuffer.
示例:下面的例子说明了如何用CString::GetBuffer和CString::ReleaseBuffer(最好逐步调试观察值和地址的变化)



#include <stdio.h>
#include <afxwin.h>

void main(void)
{
CString s("abcd");

int i = s.GetLength();
printf("%s\n", s);
printf("s length1 is %d\n", i);

LPTSTR p = s.GetBuffer(6);
strcpy( p, "12345");
printf("%s\n", s);
int j = s.GetLength();
printf("s length2 is %d\n", j);

s.ReleaseBuffer();
printf("%s\n", s);
int k = s.GetLength();
printf("s length3 is %d\n", k);

strcpy( p, "a1b2c3");     // 这里新指定的字串若为"a1b2c3e"则会在此中断
printf("%s\n", s);
int m = s.GetLength();
printf("s length4 is %d\n", m);
}


运行结果为:



注意: 
1、CString对象内部的最后保留了'\0'字符,但是必须通过Getbuffer返回其指针后访问它才不会发生异常,如果用GetAt(GetLength())访问就会发生异常. 

2、GetBuffer函数与GetbufferSetLength中的参数如果大于字符串的实际长度,则自动开辟新增加的长度。当调用ReleaseBuf函数后,则新增加的缓冲区如果未被存放内容那么多于的内存就会被释放。如果参数小于字符串的实际长度,并且大于等于0,则利用返回的指针访问不会发生异常,并且如果未做任何修改操作,那么当调用ReleaseBuf函数后,还是保持原来的字符串内容部变。如果小于0则会发生异常。

3、通过指针p和字符串s 实行字符串动态增加的效果是完全不一样的,因为字符串s里有很多成员函数为之服务。 
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: