TinyXml Print 函数 Bug

最近做一个项目,用到了TinyXml操作XML,有一个地方需要用print函数把XML文本打印出来,代码如下:

m_pDbCfg->trees[m_pDbCfg->treeCount-1].cbuffer = (char*)malloc( MAX_BUFFERSIZE );
int idx = 0;
int isize = MAX_BUFFERSIZE;//由于无法预计最大长度,<span style="font-family: Arial, Helvetica, sans-serif;">看了一下Print函数,打印时对长度做出了判断,当长度不够时,realloc了一个新的大小,便</span>将<span style="font-family: Arial, Helvetica, sans-serif;">MAX_BUFFERSIZE定义为512,</span>
treeElement->Print( m_pDbCfg->trees[m_pDbCfg->treeCount-1].cbuffer, &idx, &isize, 0 );

程序结束时,我打算free掉m_pDbCfg->trees[m_pDbCfg->treeCount-1].cbuffer,问题来了,有时会崩溃,有时正常,真是让人觉得莫名其妙,找同事看了看,一开始在外边自己代码找问题,感觉TinyXml这个库辣么多人用,应该不会出问题嘛。但是,外边没找到问题,后来调试到Print函数里边,突然发现,realloc 存在问题,把代码贴出来,如下(代码太多就不全部贴出来了,能说明问题就行):

void TiXmlElement::Print( char* cbuffer, int *idx, int *size, int depth ) const
{
	int i;
	for ( i=0; i<depth; i++ )
	{
		sprintf( &cbuffer[*idx], "    " ); 
		*idx += strlen("    "); 
		if(*idx > *size-128)
		{ 
			*size += 1024; 
			cbuffer = (char *)realloc(cbuffer, *size); 
		}
	}
<span style="font-family: Arial, Helvetica, sans-serif;">}</span>

问题是什么呢?那就是realloc之后,指针的地址可能会发生变化,cbuffer = (char*)realloc(cbuffer, *size);也即是说,此时返回的cbuffer(局部变量)与实参不一致,至于原因可见realloc函数介绍,如下:
原型::void *realloc(void *mem_address, unsigned int newsize);
功能:先判断当前的指针是否有足够的连续空间,如果有,扩大mem_address指向的地址,并且将mem_address返回,如果空间不够,先按照newsize指定的大小分配空间,将原有数据从头到尾拷贝到新分配的内存区域,而后释放原来mem_address所指内存区域(注意:原来指针是自动释放,不需要使用free),同时返回新分配的内存区域的首地址。即重新分配存储器块的地址。
所以,如果空间不够,那么m_pDbCfg->trees[m_pDbCfg->treeCount-1].cbuffer指针所指内容就是无效的,当我使用m_pDbCfg->trees[m_pDbCfg->treeCount-1].cbuffer去free时,程序就会崩溃。
解决:可以把print函数的参数cbuffer定义为指针的指针,或者指针的引用,这样,realloc产生的地址就能赋值给实参了,由于指针还要解引用,改动代码太多,本人将cbuffer改为指针的引用,问题成功解决。
感慨:这么成熟的一个库,都会出这样的问题,一定要敢于怀疑,大胆假设,小心求证。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
tinyxml2是一个轻量级的XML解析器,非常适合在嵌入式系统中使用。下面是一些常用的tinyxml2函数的详细解释: 1. XMLDocument XMLDocument对象表示一个XML文档。它包含了一些常用的函数,如LoadFile()和SaveFile()。 - LoadFile(const char* filename):从文件中加载XML文档。 - SaveFile(const char* filename):将XML文档保存到文件中。 - Clear():清除XMLDocument对象中所有的内容。 - NewElement(const char* name):创建一个新的XMLElement对象并将其添加到XMLDocument中。 2. XMLElement XMLElement对象表示XML文档中的一个元素。它包含了一些常用的函数,如FirstChildElement()和Attribute()。 - FirstChildElement():返回第一个子元素。 - NextSiblingElement():返回下一个兄弟元素。 - Attribute(const char* name):返回指定名称的属性值。 - SetAttribute(const char* name, const char* value):设置指定名称的属性值。 - GetText():返回元素的文本内容。 - SetText(const char* text):设置元素的文本内容。 - Value():返回元素的名称。 3. XMLNode XMLNode对象是XMLElement和XMLDocument的基类。它包含了一些常用的函数,如FirstChild()和NextSibling()。 - FirstChild():返回第一个子节点。 - NextSibling():返回下一个兄弟节点。 - ToElement():将XMLNode对象转换成XMLElement对象。 - ToDocument():将XMLNode对象转换成XMLDocument对象。 4. XMLText XMLText对象表示XML文档中的一个文本节点。它包含了一些常用的函数,如SetValue()和Value()。 - SetValue(const char* value):设置文本节点的值。 - Value():返回文本节点的值。 5. XMLComment XMLComment对象表示XML文档中的一个注释节点。它包含了一些常用的函数,如SetValue()和Value()。 - SetValue(const char* value):设置注释节点的值。 - Value():返回注释节点的值。 6. XMLDeclaration XMLDeclaration对象表示XML文档中的一个声明节点。它包含了一些常用的函数,如SetValue()和Value()。 - SetValue(const char* value):设置声明节点的值。 - Value():返回声明节点的值。 7. XMLAttribute XMLAttribute对象表示XML文档中的一个属性节点。它包含了一些常用的函数,如SetValue()和Value()。 - SetValue(const char* value):设置属性节点的值。 - Value():返回属性节点的值。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值