`
nigelzeng
  • 浏览: 605732 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

malloc函数引起的意外错误

    博客分类:
  • C
阅读更多

在实现一个简单的字符对应函数的过程中发现了这个不算是BUG的错误:

这个错误出现的前提是这样的:

 

Main函数如下:

 

int main(int argc, char **argv)
{
	int num = 4;
	char *nodeName=NULL;
	nodeName = getNodeName(num);
	printf("\n\nnodeName is:\n%s\n\n",nodeName);
	char *aliasVal = (char *)malloc(10);
	getAliasVal(aliasVal, num);
	printf("\n\nnodeName is:\n%s\n\n",nodeName); //不能在这里输出nodeName,
	exit(0);

}
 

 

getNodeName函数:

 

char *getNodeName(int num)
{
	printf("\n\ntest num:%d\n\n",num);
/*	char *nodeName=NULL;*/
	char *nodeName = (char *)malloc(10);
	// 生成节点名
	switch (num)
	{
		case 1:
			nodeName = "port1";
			printf("\n\nin the switch:%s\n\n",nodeName);
			break;
		case 2:
			nodeName = "port2";
			break;
		case 3:
			nodeName = "port3";
			break;
		case 4:
			nodeName = "port4";
			printf("\n\nin the switch:%s\n\n",nodeName);
			break;
		case 5:
			nodeName = "port5";
			break;
		case 6:
			nodeName = "port6";
			break;
		case 7:
			nodeName = "port7";
			break;
		case 8:
			nodeName = "port8";
			break;
	}
	printf("\n\nOUT OF the switch:%s\n\n",nodeName);
	return nodeName;
}

 

getAliasVal函数

 

char *getAliasVal(char *aliasVal, int num)
{
	char b[1];
	int tmp = num - 1;
	if (tmp == 0)
	{
		strcpy(aliasVal, "ttyM0");
	}
	else
	{
		// 整数转字符串,方便拼接
		itoa(tmp, b);
		// 生成alias属性的值
	/*	char *aliasVal = (char *)malloc(10);*/
		strcpy(aliasVal, "ttyM");
		strcat(aliasVal, b);
	}
	printf("\n\nresult:\n%s\n\n",aliasVal);	
	return aliasVal;
}

 

 

错误产生在main函数的第9行那段,

该段代码不能正确产生输出,而前一句(第6行)是可以正确输出的,具有可重复性。

在getNodeName中nodeName也都能正确输出。

 

起初是怀疑nodeName指针出的问题,

但是用GDB进行简单的调试之后,发现nodeName指针并没有错误。

 

于是定位到malloc()函数这里,

因为malloc()函数之前可以正确输出,而malloc()之后就乱了,而且每次执行的结果都不一样。(指针的地址每次不一样。)

 

解决办法目前我只想到取消使用malloc()函数,我推测的原因是malloc把内存重新整理过才进行的分配,所以指针会乱。深层一点的代码我暂时没时间阅读……

做完毕业设计先……

0
3
分享到:
评论
4 楼 nigelzeng 2012-05-15  
wuzhi3078 写道
这个是{}作用域引起的。
char *getAliasVal(char *aliasVal, int num)
{
	char b[1];
	int tmp = num - 1;
	if (tmp == 0)
	{
		strcpy(aliasVal, "ttyM0");
	}
	else
	{
		// 整数转字符串,方便拼接
		itoa(tmp, b);
		// 生成alias属性的值
		char *aliasVal = (char *)malloc(10);/*如果你在这里声明,编译器会以为aliasVal是这个{}里面的,不是参数那个*/
		strcpy(aliasVal, "ttyM");
		strcat(aliasVal, b);
	}
	printf("\n\nresult:\n%s\n\n",aliasVal);/*当else{}结束了,里面那个aliasVal变量也不见了,(没有free,分配的内存是存在的).这里的aliasVal其实是参数那个,因为参数那个aliasVal在传参前分配了内存,但是没有赋值,内存是乱码,printf打印的就是乱码。你看下面的例子*/
	return aliasVal;
}

char* GetVal(char *a, int m)
{
	{
		char *a = (char*)malloc(10);
		strcpy(a, "EEEEEEE\n");
		//return a;
	}
	return a;
}

void main()
{
	char *a;
	a =(char *) malloc(10);
	strcpy(a, "ABCDEF\n");
	a = GetVal(a, 1);
	printf("%s", a);
}

输出的是ABCDEF。就是说出了{}返回的是参数那个.如果把里面那个注释,返回的就是里面那个,输出EEEEEE. 如果没有大括号你会发现,变量重名编译出错,这跟全局变量,局部变量有点相似的


嗯,我试了一下,正如你所说。呵呵,谢谢你!
3 楼 wuzhi3078 2012-05-14  
这个是{}作用域引起的。
char *getAliasVal(char *aliasVal, int num)
{
	char b[1];
	int tmp = num - 1;
	if (tmp == 0)
	{
		strcpy(aliasVal, "ttyM0");
	}
	else
	{
		// 整数转字符串,方便拼接
		itoa(tmp, b);
		// 生成alias属性的值
		char *aliasVal = (char *)malloc(10);/*如果你在这里声明,编译器会以为aliasVal是这个{}里面的,不是参数那个*/
		strcpy(aliasVal, "ttyM");
		strcat(aliasVal, b);
	}
	printf("\n\nresult:\n%s\n\n",aliasVal);/*当else{}结束了,里面那个aliasVal变量也不见了,(没有free,分配的内存是存在的).这里的aliasVal其实是参数那个,因为参数那个aliasVal在传参前分配了内存,但是没有赋值,内存是乱码,printf打印的就是乱码。你看下面的例子*/
	return aliasVal;
}

char* GetVal(char *a, int m)
{
	{
		char *a = (char*)malloc(10);
		strcpy(a, "EEEEEEE\n");
		//return a;
	}
	return a;
}

void main()
{
	char *a;
	a =(char *) malloc(10);
	strcpy(a, "ABCDEF\n");
	a = GetVal(a, 1);
	printf("%s", a);
}

输出的是ABCDEF。就是说出了{}返回的是参数那个.如果把里面那个注释,返回的就是里面那个,输出EEEEEE. 如果没有大括号你会发现,变量重名编译出错,这跟全局变量,局部变量有点相似的
2 楼 nigelzeng 2012-05-14  
wuzhi3078 写道
/*  char *aliasVal = (char *)malloc(10);*/ 

这里才不认为是malloc的问题。
我认为是char *aliasVal重名的问题。不知道编译器为什么认为是符号。
(char *)aliasVal = (char *)malloc(10)

这样是通过的。我不知道你单单是测试malloc的bug还是干嘛。都没free。
估计是每个编译器的问题不一样吧,我用VC6。
谢谢你的博文!


是char *aliasVal的重名问题导致后面的nodeName输出错误吗?
free操作是我没有考虑到……
我接触linux c语言不久,所以很多地方还不明白,还请你不吝赐教。
1 楼 wuzhi3078 2012-05-14  
/*  char *aliasVal = (char *)malloc(10);*/ 

这里才不认为是malloc的问题。
我认为是char *aliasVal重名的问题。不知道编译器为什么认为是符号。
(char *)aliasVal = (char *)malloc(10)

这样是通过的。我不知道你单单是测试malloc的bug还是干嘛。都没free。
估计是每个编译器的问题不一样吧,我用VC6。
谢谢你的博文!

相关推荐

    C、C++笔试面试宝典

    不过需要注意的是,引用传递可能会引起对于原始数据的意外修改,因此在使用时要特别小心。 在准备C和C++的笔试面试时,掌握上述概念和知识点对于应对各类问题非常关键。理解new/delete和malloc/free的不同用法,...

    C/C++常见错误汇总

    17. **预处理器宏错误**:宏可以引起意外的代码扩展,尤其是涉及到函数式宏和条件编译时。使用时需谨慎,尽量用模板或inline函数代替宏。 18. **递归深度限制**:无限递归或过深的递归会导致栈溢出。确保递归有明确...

    学习C++需要了解的十八个基础知识.docx

    同时,布尔变量不应与任何值进行比较,以免导致错误,整型变量之间的比较需确保类型匹配,浮点数比较时最好有明确的比较值限制,指针变量应当与NULL进行比较,而不是与布尔型或整型比较。 其次,深入理解const和#...

    C C++高质量编程(林锐)

    - 使用CONST可以避免意外修改,提高代码的安全性。 **11.2 提高程序的效率** - 优化算法和数据结构是提高程序效率的关键。 - 适当使用缓存、提前计算等技术也可以提升性能。 **11.3 一些有益的建议** - 遵循良好的...

    c库函数查询--很强大

    这些函数为程序设计提供了便利,例如`printf`用于格式化输出,`scanf`用于从标准输入读取数据,`strlen`计算字符串长度,`strcpy`复制字符串,`malloc`和`free`管理动态内存,`pow`进行幂运算等。 了解C库函数的一...

    编写优质无错代码(3).docx

    这种风格被称为"防御性编程",可以减少因意外赋值引起的错误。 8. **内存安全**:避免返回栈上的局部变量地址,如`DoSomeThing`函数。因为函数结束时,栈上的内存将被回收,返回的指针会变成无效的。 综上所述,...

    最常见的50个CC面试题.pdf

    - `malloc/free`是C语言中的内存管理函数,分配和释放内存,不调用构造和析构函数。 - `new/delete`是C++中的操作符,分配内存时可以调用构造函数,释放时调用析构函数,且可以重载。 6. **宏MIN的副作用**: - ...

    Effective_C++

    - **解释**: `malloc`和`free`是C语言中的函数,用于动态内存分配和释放。相比之下,C++提供了`new`和`delete`操作符,它们不仅可以分配内存,还能够调用构造函数和析构函数,更加安全和高效。此外,`new`和`delete`...

    改善程序设计技术的50个有效做法

    C++推荐使用行内注释`//`,而不是块注释`/*...*/`,因为后者容易引起未关闭的注释错误。然而,当需要多行注释时,可以适当使用块注释,但需谨慎使用。 5. **正确使用new和delete**: 当动态分配数组时,使用方...

    各大IT公司笔试真题汇总_华为面试笔试题目(附答案

    7. 静态全局变量:在给定的例子中,我们可以看到静态全局变量的作用域是局限于一个源文件内的,它可以避免在其它源文件中引起错误。 知识点:静态全局变量、作用域、源文件 8. 静态局部变量:在给定的例子中,我们...

    effective C++

    3. **尽量用new和delete代替malloc和free**:C++的new和delete操作符提供了自动类型转换和构造/析构函数调用,这在处理对象实例化和内存管理时非常重要。malloc和free只管理原始字节,不调用对象的构造和析构。 4. ...

    c/c++语言编程修养、习惯

    使用`sizeof`操作符时,直接指定类型而非变量可以避免因指针解引用引起的错误。 #### 三十一、不要忽略Warning 编译器警告往往是潜在问题的指示,及时解决这些问题可以预防未来可能出现的重大错误。 #### 三十二...

    嵌入式系统和linux工程师面试题.docx

    19. `const`在C语言中表示常量,推荐使用它代替`#define`,因为`const`是类型安全的,不会引起意外的类型转换。 20. `volatile`表示变量的值可能在程序未显式修改时发生变化,如硬件I/O操作。编译器不会优化这类...

    常见嵌入式笔试题

    此外,ISR不应进行复杂计算或调用可能引起额外中断的函数,以防止死锁或资源竞争。 #### 数据类型转换和溢出问题 代码片段`void foo(void)`展示了当混合使用有符号和无符号整型变量进行算术运算时可能出现的问题。...

Global site tag (gtag.js) - Google Analytics