debug assertion failed!是啥问题?高分求助

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了debug assertion failed!是啥问题?高分求助相关的知识,希望对你有一定的参考价值。

就是这个问题 帮忙解决了再加100

直观地说,就是store.cpp的第519行出错了。

但并不一定是说,519行有什么语法错误,也可能根源在此之前。

该警告表明程序在选定点的行为与预期行为不符合。

在程序跟踪调试过程出现此警告,最常见的错误就是引用了不存在的变量或函数,或者变量未经过声明、函数未经定义,又或者把某个局部变量当做全局变量引用。此类错误通常归结为所谓的“野指针”。

另一类可能是编程习惯不严谨,导致的内存泄漏。

查错应首先着手于519行调用了什么,如果是变量或函数,则此变量在之前是否未声明、函数未定义,又或者曾经存在但在519行之前又被删去;如果519行是与创建资源相关,那么资源创建与其特性是否考虑周到。比如,反复创建同一类资源,在前一资源未释放内存条件下又重复创建。如此,可能导致分配的内存很快耗尽,程序崩溃。

一、assertion的作用

assertion用于差错,C++通过assert宏提供断言功能,其功能定义如下:

    assert([表达式]);//表达式为假,程序终止在这一行,并报错。

    断言是一种判断条件,在程序执行中的特定点条件表达式必须为真。在条件不满足时系统停止程序的执行并报告错误。

    C++通过assert宏提供断言功能,要使用assert宏必须包含头文件#include <cassert>

    assert要执行一个可以得出true或者false的表达式。通常,关系表达式,逻辑表达式或者返回为bool值的函数都可以用于assert表达式

二、程序在选定点的行为与预期行为不符合,可能的原因包括野指针和内存泄漏

野指针——野指针指向一个已删除的对象或未申请访问受限内存区域的指针。与空指针不同,野指针无法通过简单地判断是否为 NULL避免,而只能通过养成良好的编程习惯来尽力减少。对野指针进行操作很容易造成程序错误。

内存泄漏(Memory Leak)——指程序中己动态分配的堆内存由于某种原因程序未释放或无法释放,造成系统内存的浪费,导致程序运行速度减慢甚至系统崩溃等严重后果。

泄漏原因

在C语言中,从变量存在的时间生命周期角度上,把变量分为静态存储变量和动态存储变量两类。静态存储变量是指在程序运行期间分配了固定存储空间的变量而动态存储变量是指在程序运行期间根据实际需要进行动态地分配存储空间的变量。在内存中供用户使用的内存空间分为三部分:

    程序存储区

    静态存储区

    动态存储区。

    程序中所用的数据分别存放在静态存储区和动态存储区中。静态存储区数据在程序的开始就分配好内存区,在整个程序执行过程中它们所占的存储单元是固定的,在程序结束时就释放,因此静态存储区数据一般为全局变量。动态存储区数据则是在程序执行过程中根据需要动态分配和动态释放的存储单元,动态存储区数据有三类函数形参变量、局部变量和函数调用时的现场保护与返回地址。由于动态存储变量可以根据函数调用的需要,动态地分配和释放存储空间,大大提高了内存的使用效率,使得动态存储变量在程序中被广泛使用。

    开发人员进行程序开发的过程使用动态存储变量时,不可避免地面对内存管理的问题。程序中动态分配的存储空间,在程序执行完毕后需要进行释放。没有释放动态分配的存储空间而造成内存泄漏,是使用动态存储变量的主要问题。一般情况下,开发人员使用系统提供的内存管理基本函数,如malloc、recalloc、calloc、free等,完成动态存储变量存储空间的分配和释放。但是,当开发程序中使用动态存储变量较多和频繁使用函数调用时,就会经常发生内存管理错误,例如:

    分配一个内存块并使用其中未经初始化的内容;

    释放一个内存块,但继续引用其中的内容;

    子函数中分配的内存空间在主函数出现异常中断时、或主函数对子函数返回的信息使用结束时,没有对分配的内存进行释放;

    程序实现过程中分配的临时内存在程序结束时,没有释放临时内存。内存错误一般是不可再现的,开发人员不易在程序调试和测试阶段发现,即使花费了很多精力和时间,也无法彻底消除。

    产生方式的分类

    以产生的方式来分类,内存泄漏可以分为四类:

    常发性内存泄漏:发生内存泄漏的代码会被多次执行到,每次被执行时都会导致一块内存泄漏。

    偶发性内存泄漏:发生内存泄漏的代码只有在某些特定环境或操作过程下才会发生。常发性和偶发性是相对的。对于特定的环境,偶发性的也许就变成了常发性的。所以测试环境和测试方法对检测内存泄漏至关重要。

    一次性内存泄漏:发生内存泄漏的代码只会被执行一次,或者由于算法上的缺陷,导致总会有一块且仅有一块内存发生泄漏。

    隐式内存泄漏:程序在运行过程中不停的分配内存,但是直到结束的时候才释放内存。严格的说这里并没有发生内存泄漏,因为最终程序释放了所有申请的内存。但是对于一个服务器程序,需要运行几天,几周甚至几个月,不及时释放内存也可能导致最终耗尽系统的所有内存。所以,我们称这类内存泄漏为隐式内存泄漏。从用户使用程序的角度来看,内存泄漏本身不会产生什么危害,作为一般的用户,根本感觉不到内存泄漏的存在。真正有危害的是内存泄漏的堆积,这会最终耗尽系统所有的内存。从这个角度来说,一次性内存泄漏并没有什么危害,因为它不会堆积,而隐式内存泄漏危害性则非常大,因为较之于常发性和偶发性内存泄漏它更难被检测到。

1、野指针的病例:

在画图的时候经常会定义一些画刷啊,画笔啊之类的指针,一般都是这么干的:CPen myPen=new CPen
然后用完后就会来一句:delete myPen,发现程序居然会报错!把这个delete语句删除就没问题了,这是为什么呢?难道不需要释放吗?
而且观察到用来申请内存的new貌似和平常的不一样,是紫色的,就像宏定义一样,但在new上右键又转不到类似宏的定义上去,这又是为什么呢?

已经把画笔还原成了原来的oldPen,但是在最后delete oldPen就会报错,只能改成上面那句oldPen=NULL;

oldPen 是个"pen"  select 回去后, 表示 还回 , 必须是 存在的!
把它delete 掉了就不存在了!

所以不要管 pOldPen(不要=0,不要delete) , 这是个局部变量 ,出函数就没有了。

2、内存泄漏的病例:

非静态内部类创建静态实例造成的内存泄漏

例如,有时候我们可能会在启动频繁的Activity中,为了避免重复创建相同的数据资源,可能会出现如下写法:

public class MainActivity extends AppCompatActivity

private static TestResource mResource = null;

@Override
protected void onCreate(Bundle savedInstanceState)
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
if(mResource == null)
mResource = new TestResource();

//...


class TestResource
//...


这样在Activity内部创建了一个非静态内部类的单例,每次启动Activity时都会使用该单例的数据。虽然这样避免了资源的重复创建,但是这种写法却会造成内存泄漏。因为非静态内部类默认会持有外部类的引用,而该非静态内部类又创建了一个静态的实例,该实例的生命周期和应用的一样长,这就导致了该静态实例一直会持有该Activity的引用,从而导致Activity的内存资源不能被正常回收。
解决方法:将该内部类设为静态内部类或将该内部类抽取出来封装成一个单例,如果需要使用Context,就使用Application的Context。

参考技术A

    原因

这种问题通常是出现了野指针,比如某个指针指向一个局部变量,而在该变量作用域外使用该指针引用了该对象。 

    解决方法

1、"Project"->"setting"->"点击c/c++"->"Category选项中选择Preprocessor" ->"在Undefined symbols:填写_DEBUG" 重新编译运行即可 

2、按F5运行你的程序 ;在出错时,选择“重试” ;按ALT+7调出“调用栈”窗口;双击从上往下的最近一个自己定义的函数,系统会自动把该函数所在的文件显示出来,此时程序就暂停在光标处。一般来说错误就出在这附近。你可以通过查看变量的值来确认

3、对于MSComm的关联变量在创建前需要先销毁窗口,如下程序段: 

mycomm.DestroyWindow();           //mycomm为MSComm的关联变量 

4、在OnCreate()中对控件进行了修改和初始化,应放在OnInitDialog中。

资料来源:百度文库

参考技术B 完全不懂,在别的地方帮你找的。不知道对不对。
可以跟踪到 mfc提供的源代码内部,(注:如果打开了mfc源代码,设置了断点,但是跟不进去,那就需要更新PDB文件,具体网上搜)
打开 wincore.cpp文件(D:\Program Files\Microsoft Visual Studio .NET 2003\Vc7\atlmfc\src\mfc)。查看 312 行,所在函数如下:
CWnd* PASCAL CWnd::FromHandle(HWND hWnd)

CHandleMap* pMap = afxMapHWND(TRUE); //create map if not exist
ASSERT(pMap != NULL);
CWnd* pWnd = (CWnd*)pMap->FromHandle(hWnd);

#ifndef _AFX_NO_OCC_SUPPORT
pWnd->AttachControlSite(pMap);
#endif

ASSERT(pWnd == NULL || pWnd->m_hWnd == hWnd);
return pWnd;


断言就是 Assert( pWnd == NULL || pWnd->m_hwnd == hWnd );也就是读取句柄映射表错误,有2种可能:
1你传入的窗口句柄为空,也就是生成了窗口对象但是没有使用 Create创建窗口。
2.窗口所在线程不是当前所在线程,那么使用FromHandle读取映射窗口指针或者映射临时窗口指针必然会出错。窗口都是线程相关的噢
你点击菜单命令出错,你就有可能你的菜单命令执行的代码有 FromHandle语句,你看看,一定就是它
参考技术C 出现这种情况很可能是使用了野指针,比如某个指针指向一个局部变量,而在该变量作用域外使用该指针引用了该对象。
Assertion 是断言,用于假定某个条件肯定会满足,在debug模式下,当该条件不满足时则发生阻塞而弹出该对话框;在release下会忽略断言。
参考技术D 多半是非法指针访问。。未初始化,等等。。

出现这种情况,就点调试。。。然后就可以根据函数调用堆栈,找到是哪里出错了。。(菜单查看->调试窗口->call stack; 或者variable 窗口上的下拉菜单也可以看追问

是这样的一个问题

关于Debug Assertion Failed问题

Debug Assertion Failed

program:D\...

LINE:1044

Expression:_CrtIsValidHeapPointer(pUserData)

for information on how your program can cause an assertion 

failure,see the Visual C++ documentation on asserts.

(Press Retry to debug the application)

学习链表时遇到了这样的问题,暂时没找到原因。。

 

以上是关于debug assertion failed!是啥问题?高分求助的主要内容,如果未能解决你的问题,请参考以下文章

MFC中的Debug Assertion Failed 如何查找原因

debug assertion failed winocc.cpp line:138 好不容易下载到一个程序,不能打开,大大们帮我一下吧!急!

用Microsoft Visual Studio 2010编译下面这个程序总是提示Debug Assertion Failed

如何在 Visual Studio 中跳过 Debug Assertion Failed 并直接中断

vc++ 调试时出现Debug Assertion Failed!该怎么办?

opencv检错:程序运行过程正常,当跳出函数时出现断言错误(Debug Assertion Failed)