运行软件弹出错误 access violation at address 004096da write of address 00401000 怎么解决

Windows用户可能经常会看到类似于错误提示:“Error:Access violation at address 836556F8. Read of address 836556F8”。作为一个Delphi程序开发者,遇到这种错误的机会比其他用户更多(^_^)。


  究竟什么是“Access Violation”?如何在设计期避免它的出现?


  Access Violation(非法访问),General Protection Fault(一般保护性错误)或者Invalid Page Fault(无效页面错误),虽然说法不一样,但本质上总是由同一种错误引起的。Access Violation常常在计算机用户运行的程序试图存取未被指定使用的存储区时遇到。
Access violation at address <十六进制值>
in module <应用程序名>
Read of address <十六进制值>


  一旦Windows要在它被分配的存储区之外写数据信息,它就会覆盖其他程序甚至操作系统的命令或数据。一旦发生了这种情况,操作系统将会瘫痪或者以某种形式关闭,你必须重新启动计算机。例如,在Windows NT/2000下一个程序遇到这种错误时,Dr. Watson出现并且停止了该程序,捕获了一些快速的细节状态,再把它们用文本形式记录下来。Access Violation是某些最令人气恼的Windows程序遇到的错误之一。本文的目的就是让你找到Delphi中Access Violation的解决之道。首先声明一点,Access Violation和Microsoft Access没有任何关系。

 用Delphi开发程序时,我们可以把遇到的Access Violation分成两大类:运行期和设计期。 


一、设计期的Access Violation 


1.硬件原因
  在启动或关闭Delphi IDE以及编译一个Delphi工程时容易出现设计期的Access Violation。在你的计算机运行中出现Access Violation信息可能由各种各样的原因引起,包括系统BIOS、操作系统或者是硬件驱动线,有些声卡、显卡、网卡实际上也会导致这种错误。为什么这么说?计算机里的每一块卡都有它的设备驱动程序。对于不同的制造商、不同版本的Windows或者不同版本的Delphi都可能会遇到不同的问题。如下的几个步骤可能有助于你解决遇到的这些问题: 


  1. 按照必要的步骤来证实你安装的驱动程序之间没有冲突。 


  2. 有时降低显示分辨率可能会使某些古怪的显卡驱动程序稳定一些。 


  3. 如果使用双处理器的主板,则保证对每个处理器的修改步骤一样。 


  4. 对于计算机上的所有硬件注意使用最新的驱动程序。 


2.软件原因
  尽管Intel的计算机中Windows是最流行的操作系统,由于Windows系统天生的脆弱性和BUG,应用程序的误操作可能导致操作系统的迅速瘫痪(有时操作系统本身也会莫名其妙的瘫痪)。选择一个更稳定的程序开发环境是解决之道,如下几个步骤可以帮助你防止某些Access Violation的发生: 


  (1)尽管Windows 9X相当流行,Windows NT/2000还是从多方面被证实是一个稳定得多的环境,几乎对于所有的Windows代码平台而言都是这样。 


  (2) 确保对于Windows NT/2000已经安装了最新的service pack。每次安装完新版的service pack,你会发现机器变得稳定了。 


  (3) 为你使用的各种版本的Delphi装上当前的更新或补丁(BDE、ADO……),这 是提前预防错误的好办法。尽量使用最新的Delphi补丁——Access Violation错误数量尤其是设计期的错误数会大大减少。 


  (4)如果你在IDE中经常随机遇到Access Violation错误,很有可能是你安装了一个不好的控件、包或者一个向导,它不是你使用的版本的Delphi所编写或编译的。试着一个一个卸载定制的控件(或者包)直到问题被解决,然后联系控件厂商关注这个问题的结果。 


  (5) 检查一下计算机里是否有没用的东西和程序冲突。奇怪的软件程序和测试版的产品常常会导致Access Violation错误。 


  (6) 如果系统设置有错误,那么Access Violation错误可能也会经常出现。如果你不停地遇到一个错误提示信息一样的Access Violation,记录下这些细节,然后通知可能导致这个错误的软件制造厂商。 


  这些就是我对设计期Access Violation错误的全部建议。 


二、运行期的Access Violation
  Delphi常见的运行期Access Violation错误有哪些?如何防止?


  任何软件
开发都会遇到这样的情况:你写好程序并测试,然后到处发送,结果用户告诉你它失败了。


  你可能考虑用编译指令{$D}编译你的程序——Delphi可以建立一个有助于定位Access Violation错误的源代码的镜像文件。工程选项对话框(Project|Options|Linker & Compiler)让你指定你所需要的一切。对于单元文件,debug信息和单元的对象代码一起记录在unit文件里了。编译使用这个单元的程序时,debug信息会增加单元文件的大小而且会增加额外的内存开销,但是它不会影响最终可执行文件的大小和运行速度。包含debug信息和镜像文件(Project|Options|Linker)选项的产品只有在{$D+} 编译指令下才会完成行信息。


  Access violation通常只在程序的某一个方面表现出来。当问题第一次出现时,考虑一下用户进行了什么操作是很重要的,然后从这里寻找突破口。从用户的角度来看,你的程序中止了他们的工作,由他们来告诉你出现的问题似乎让你延期解决这个问题了。然而,与用户交流是你发现问题和改善程序的惟一有效方法。


  现在你将可以知道在只给你冲突地址的情况下,如何轻松发现准确路径、源代码文件、发生Access violation错误的行:
“Search - Find Error…”。


  当一个运行期Access violation出现时,你的用户得到的错误信息类似于如下情况:
Access violation at address <十六进制值>
in module <应用程序名>
Read of address <十六进制值>


  如果你的程序在Delphi IDE里包含debug信息编译,你可以定位到导致这个错误源代码这一行。
在Delphi程序中,一个最普遍导致Access Violation错误的原因是使用了一个没有被创建的对象。如果第二个地址<十六进制值>是FFFFFFF或0000000,十有八九就是你访问? 了一个没有被建立的对象。例如,你调用了一个表单的事件,但这个表单不是自动创建的,也没有代码实例化。


?procedure TfrMain.OnCreate(Sender: TObject);
var BadForm: TBadForm;
begin
//这里将会产生Access violation
BadForm.Refresh;
end;


  假设BadForm在工程选项“Available Forms”窗口列表里——这个窗口是需要手工创建和释放的。在上面的代码里调用BadForm窗口的Refresh方法就会导致Access violation。


  如果你在Debugger选项窗口使“Stop on Delphi Exceptions”生效,那么就会弹出下面的信息:
The message states that the EAccessViolation has occurred. The EAccessViolation is the exception class for invalid memory access errors. 


  这是你在设计程序时将会看到的信息,下一个信息框将会出现,然后程序失败了:
Access violation at address 0043F193
in module ’Project1.exe’
Read of address 000000.


  第一个十六进制数0043F193是发生Access violation的编译代码(Project1.exe)的运行期错误的地址。在IDE里选择菜单项“Search|Find Error…”,在对话框里输入错误发生的地址(0043F193)后点击“OK”按钮。Delphi将会重新编译你的工程文件,然后显示发生运行期错误的那一行代码,这里就是BadForm.Refresh这一行了。


  下面列出了Delphi环境下导致Access violation错误的大部分常见原因。这个列表不是也不可能覆盖所有可能出现的Access violation的情况。请在论坛上发送你的Access violation信息,大家可以试着一起解决这个问题——真正的实际事例一般情况下比列出来的错误隐晦得多。


1. 调用一个不存在的对象
  如上所述,大部分Access violation的合理原因是使用了没有被创建或者已经被释放的对象。为了防止这种类型的Access violation的发生,请确保你访问的任何对象都首先被创建了。例如,当一个Table定位在一个没有被创建的data module(从auto-crete窗口里移走了)里,你可能在窗体的OnCreate事件里打开这个表。


  在下面的代码里,在调用一个已经被删除了的对象(b:TBitmap)事件后,一个Access violation出现了:
var b:TBitmap;
begin
b:=TBitmap.Create;
try
//对b对象进行一些操作
finally
b.free;
end; 
...
//由于b已经被释放,一个Access violation错误将会出现
b.Canvas.TextOut(0,0,’这是一个 Access Violation’);
end;


2. 不存在的API参数
  如果你试图给Win API函数传递一个不存在的参数将会出现一个Access violation错误。解决此类Access violation错误的最好方法是查阅Win API帮助,看看这个API函数调用的参数信息以及参数类型。例如,总是保证不给一个缓冲参数传递一个无效指针。


3. 让Delphi释放
  当一个对象拥有另一个对象时,让它给你做删除工作。因为默认情况下,所有的窗体(自动创建的)都属于Application对象。当一个应用程序结束时,它释放了Application对象,也就释放了所有窗体。例如,如果你在程序开始时自动创建了两个窗体(Form1/Unit1和Form2/Unit2),下面的代码就会导致Access violation错误的出现:
unit Unit1;
...
uses unit2;
...
procedure TForm1.Call_Form2
begin
Form2.ShowModal;
Form2.Free;
//Access violation错误将会出现 
Form2.ShowModal; 
end;


4. 杀死异常
  永远不要破坏临时异常对象(E),处理一个异常会自动释放异常对象。如果你自己手动释放了异常对象,程序会试图再次释放它,那么就会出现Access violation错误:
Zero:=0;
try
dummy:= 10 / Zero;
except
on E: EZeroDivide do
MessageDlg(’不能用0做除数!’,mtError, [mbOK], 0);
E.free. Access violation错误将会出现
end;


5. 检索一个空字符串
  一个空字符串是没有任何数据的。就是说,检索一个空字符串相当于访问一个不存在的对象,这将导致Access violation错误:
var s: string;
begin
s:=’’;
s[1]:=’a’; 
//Access violation错误将会出现
end;


6. 直接引用指针
你必须间接引用指针,否则你会改变指针地址并可能会破坏其他存储单元 :
procedure TForm1.Button1Click(Sender: TObject);
var
p1 : pointer;
p2 : pointer;
begin
GetMem(p1, 128);
GetMem(p2, 128);
//下一行导致Access violation错误
Move(p1, p2, 128);
//下一行方法正确
Move(p1^, p2^, 128);
FreeMem(p1, 128);
FreeMem(p2, 128);
end;
  这些就是我对运行期Access Violation错误的全部建议,我希望你们也能对你们程序出现的Access Violation错误提出一些看法。


  老猫点评:相信所有读者都遇到过“Access violation” 的错误,如果不是自己的程序,我们有很多人就把责任都推在Bill Gates的头上。如果你自己的程序出现了这个尴尬的错误,面对用户的询问,我们该如何解释?本文就是最好的答案

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 这个错信息意味着在程序中出现了访问地址(address)0x00000000的错。具体来说,发生了一个试图在地址0x00000000上进行写入操作的访问违例(access violation)。 在计算机程序中,每个变量或数据都存储在内存的某个地址上。程序需要通过对这些地址的读取和写入来操作数据。然而,访问地址0x00000000是非法的,因为它是一个空地址(null address)或无效地址(invalid address)。 通常情况下,这种错是由以下几种情况引起的: 1. 空指针引用(Null Pointer Reference):在代码中试图通过空指针来访问或修改数据,这会导致访问违例错。 2. 未初始化的指针(Uninitialized Pointer):使用一个未初始化的指针进行读写操作,指针的值为0x00000000,因而无法访问有效地址。 3. 越界访问(Out-of-Bounds Access):尝试访问数组或其他数据结构的超出其边界范围的元素,导致访问了无效地址。 解决这种错通常需要检查代码中的指针使用和数据访问逻辑。确保指针被正确初始化,并避免在访问数据之前检查其是否为空指针。另外,还需确保进行数组访问时不会越界。 在调试错时,可以使用调试器检查程序执行过程中的内存访问情况,以确定错出现的具体位置。此外,对于大型项目,可以考虑使用内存检测工具或静态代码分析工具来帮助找出潜在的内存访问问题。 ### 回答2: "access violation at address 00000000.write of adress 00000000" 是一个表示访问错的错消息,通常出现在计算机软件中。这个错消息指示在程序运行过程中,试图写入一个无效的内存地址0x00000000,导致访问冲突。具体原因和解决方法可能因软件和编程环境而异,以下是一些可能的原因和解决方法: 1. 空指针引用:错消息中的0x00000000可能表示空指针,即未分配有效内存地址的指针。解决方法是检查代码中的指针和内存分配,确保指针不为NULL。 2. 资源文件缺失或损坏:某些软件可能需要依赖于特定文件或库,如果这些文件丢失或损坏,将导致访问错解决方法是重新安装软件或修复文件。 3. 内存错:如果程序访问了无效的内存地址,可能是由于内存错引起的。解决方法是进行内存检查,确保程序没有访问无效的内存地址。 4. 驱动程序冲突或过时:某些软件可能需要特定的驱动程序支持。如果驱动程序过时或与其他软件冲突,可能会导致访问错解决方法是更新或卸载冲突的驱动程序。 总结来说,"access violation at address 00000000.write of adress 00000000" 错消息表明程序试图写入无效的内存地址,解决方法包括检查空指针引用、修复或重新安装相关文件、进行内存检查以及更新驱动程序等。具体解决方法需要根据具体软件和错情况来确定。 ### 回答3: access violation是指访问违规,在计算机程序中发生这种错通常会导致程序崩溃或异常退出。access violation at address 00000000.write of address 00000000是指在地址00000000处发生了写访问的违规操作。 发生这种错通常有以下几种原因: 1. 空指针引用:程序试图访问一个未初始化或者被释放的指针所指向的内存地址,由于该地址为空,所以发生了写访问的违规操作。 2. 内存越界:程序试图写入超出分配内存范围的地址,这可能是由于数组越界、栈溢出或者堆操作不正确引起的。 3. 内存保护:操作系统对某些内存地址进行了保护,例如对只读内存进行写操作,或者对已释放的内存进行访问。 4. 软件:程序设计存在逻辑错或者编码错,导致了访问地址错的情况。 如果想要解决这个错,我们可以采取以下几种方法: 1. 检查代码逻辑,查找空指针引用或者内存越界的情况,并进行修复。 2. 使用调试工具定位错位置和原因,例如使用断点调试器查看引发错的代码行和相关变量。 3. 确认程序的内存分配方式是否正确,避免对已释放的内存进行访问。 4. 确保使用的库或者模块是最新版本,并检查是否有已知的错或者安全漏洞。 总之,解决access violation at address 00000000.write of address 00000000错需要仔细分析错原因,修复代码逻辑并保持软件的稳定性和安全性。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值