Chinaunix首页 | 论坛 | 博客
  • 博客访问: 537055
  • 博文数量: 576
  • 博客积分: 40000
  • 博客等级: 大将
  • 技术积分: 5020
  • 用 户 组: 普通用户
  • 注册时间: 2008-10-13 14:47
文章分类

全部博文(576)

文章存档

2011年(1)

2008年(575)

我的朋友

分类:

2008-10-14 14:58:38

调试方法和技巧

作者:

便于调试的代码风格:

  1. 不用全局变量
  2. 所有变量都要初始化,成员变量在构造函数中初始化
  3. 尽量使用const
  4. 详尽的注释

VC++编译选项:

  1. 总是使用/W4警告级别
  2. 在调试版本里总是使用/GZ编译选项,用来发现在Release版本中才有的错误
  3. 没有警告的编译:保证在编译后没有任何警告,但是在消除警告前要进行仔细检查

调试方法:

1、使用 Assert(原则:尽量简单)
  assert只在debug下生效,release下不会被编译。

例子:

char* strcpy(char* dest,char* source)
{
	assert(source!=0);
	assert(dest!=0);
	char* returnstring = dest;
	
	while((*dest++ = *source++)!= ‘\0’)
	{
		;
	}
	return returnstring;
}      
2、防御性的编程

例子:
char* strcpy(char* dest,char* source)
{
	if(source == 0)
	{
		assert(false);
		reutrn 0;
	}

	if(dest == 0)
	{
		assert(false);
		return 0;
	}
	char* returnstring = dest;
	while((*dest++ = *source++)!= ‘\0’)
	{
		;
	}
	return returnstring;
}      
3、使用Trace

以下的例子只能在debug中显示,

例子:

a)、TRACE
CString csTest = “test”;
TRACE(“CString is %s\n”,csTest);

b)、ATLTRACE

c)、afxDump

CTime time = CTime::GetCurrentTime();
#ifdef _DEBUG
afxDump << time << “\n”;
#endif

4、用GetLastError来检测返回值,通过得到错误代码来分析错误原因

5、把错误信息记录到文件中

异常处理

  程序设计时一定要考虑到异常如何处理,当错误发生后,不应简单的报告错误并退出程序,应当尽可能的想办法恢复到出错前的状态或者让程序从头开始运行,并且对于某些错误,应该能够容错,即允许错误的存在,但是程序还是能够正常完成任务。

调试技巧

1、VC++中F5进行调试运行

a)、在output Debug窗口中可以看到用TRACE打印的信息
b)、 Call Stack窗口中能看到程序的调用堆栈

2、当Debug版本运行时发生崩溃,选择retry进行调试,通过看Call Stack分析出错的位置及原因
3、使用映射文件调试

a)、创建映射文件:Project settings中link项,选中Generate mapfile,输出程序代码地址:/MAPINFO: LINES,得到引出序号:/MAPINFO: EXPORTS。
b)、程序发布时,应该把所有模块的映射文件都存档。
c)、查看映射文件:见” 通过崩溃地址找出源代码的出错行”文件。

4、可以调试的Release版本

  Project settings中C++项的Debug Info选择为Program Database,Link项的Debug中选择Debug Info和Microsoft format。

5、查看API的错误码,在watch窗口输入@err可以查看或者@err,hr,其中”,hr”表示错误码的说明。
6、Set Next Statement:该功能可以直接跳转到指定的代码行执行,一般用来测试异常处理的代码。
7、调试内存变量的变化:当内存发生变化时停下来。

常见错误

1、在函数返回的时候程序崩溃的原因

a)、写自动变量越界
b)、函数原型不匹配

2、MFC

a)、使用错误的函数原型处理用户定义消息

正确的函数原型为:

afx_msg LRESULT OnMyMessage(WPARAM wParam,LPARAM lParam);

3、谨慎使用TerminateThread:使用TerminateThread会造成资源泄漏,不到万不得已,不要使用。

4、使用_beginthreadex,不要使用Create Thread来常见线程。

参考资料:
《Windows程序调试》


--------------------next---------------------

我新建一个dialog,new一个class可以,新建第二dialog,再新建一个类时,就说找不到对应的头文件,在ClassWizard的类列表里也没有第二个类,怎么回事?望高手指点!
( xzm_99059124 发表于 2004-12-9 21:13:00)
 
~~ ( wangjia163 发表于 2004-12-8 8:29:00)
 
凡事不可一概而论,作为一个好的程序员应该有自己的判断力。象回调函数这些不用全局变量来传递变量的话如何有用?没有警告的编译,有些警告根本就是无关大局的,象warning C4786,变量类型展开后超过255个字符,debug信息被截断,这在CList和CMap等应用中是很常见的,但它是无害的。TerminateThread并不是万不得已不能用,如果那样的话,M$干脆就可以去掉这个API了,以挂起状态创建的线程就只能在外部这样杀掉,因为线程根本就没有运行,在内部如何能够停止?

总结一句,教条主义不可取啊!我们要培养自己的判断力,而不是照本宣科,那样的话我们就只能永远跟随在别人的后面。 ( underyun 发表于 2004-12-7 9:29:00)
 
.......................................................

--------------------next---------------------

阅读(312) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~