首页 > 其他分享 >access violation at address in module Read of address

access violation at address in module Read of address

时间:2023-07-02 14:32:21浏览次数:47  
标签:Access 错误 violation Delphi access Violation Windows address


Access Violation(非法访问)错误的解决方法(转自网上)2009-02-03 16:33        Access

Violation(非法访问),General Protection Fault(一般保护性错误)或者Invalid Page Fault(无

效页面错误),虽然说法不一样,但本质上总是由同一种错误引起的。Access Violation常常在计算机

用户运行的程序试图存取未被指定使用的存储区时遇到。
Access violation at address <十六进制值>
in module <应用程序名>
Read of address <十六进制值>

  Windows用户可能经常会看到类似于错误提示:“Error:Access violation at address 836556F8

(004096da). Read of address 836556F8(00401000)”。作为一个Delphi程序开发者,遇到这种错

误的机会比其他用户更多(^_^)。

  一旦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,Delphi,access,Violation,Windows,address
From: https://blog.51cto.com/u_16174476/6605260

相关文章

  • java.net.BindException: Address already in use: JVM_Bind <null> 的解决方案
    问题描述在学习SSM整合中,启用Tomcat插件时出现以下错误java.net.BindException:Addressalreadyinuse:JVM_Bind<null>通过查阅资料发现是端口被占用了解决方案通过命令查看进程,这里我的是8080端口号被占用了netstat-ano再运行命令去杀死占用端口进程taskk......
  • System.ObjectDisposedException: Cannot access a disposed context instance
    @@abpconsoleprojectSystem.ObjectDisposedException:Cannotaccessadisposedcontextinstance.Acommoncauseofthiserrorisdisposingacontextinstancethatwasresolvedfromdependencyinjectionandthenlatertryingtousethesamecontextinstanc......
  • Cross-thread operation not valid: Control 'txtMessage' accessed from a thread ot
    WinformTextBox Cross-threadoperationnotvalid:Control'txtMessage'accessedfromathreadotherthanthethreaditwascreatedon. (330条消息)解决Cross-threadoperationnotvalid的问题_GuanXX的博客-CSDN博客  privatevoidsafeSetText(stringt......
  • C#复制Access数据库表结构和数据到另一个数据库表
    一、参考资料(以下为AccessSQL语句)参考原文链接1.1.复制表select*intotable2fromtable1--创建了一张新表:table2,把table1表中的所有数据连同表结构都一并复制到table2中--可以再通俗的理解为,先复制了个一模一样的表,然后把表名改成table2了truncatetabletable2-......
  • Access control allow origin 简单请求和复杂请求
    错误信息:XMLHttpRequestcannotloadhttp://web.image.myqcloud.com/photos/v2/10008653/bhpocket/0/?sign=4FcLKd5B8…p4SkFVUEJtZ1omZT0xNDQ0NzExMDE5JnQ9MTQ0NDcwNzQxOSZyPTEzMDMyMDgzOTAmdT0wJmY9.No'Access-Control-Allow-Origin'headerispresentontherequ......
  • 解决:远程连接mysql:报异常,1044 - Access denied for user ‘root‘@‘%‘ to database
    Navicat报错:使用Navicat远程连接,Docker中的mysql5.6时报异常,:1044-Accessdeniedforuser'root'@'%'todatabase'xxx'问题原因:(1)根本原因:远程连接用户权限不足!(2)直接原因:应该是创建远程连接用户‘root@%’时,没有添加访问数据库的权限。解决办法:#这里为刚才创建的root@......
  • ACCESS注入
    简介ACCESS是微软所推出的数据库,具有强大的数据处理和统计分析能力,常用在各类管理软件上。现如今ACCESS数据库使用占比已经较少,但当我们在测试中遇到这个数据库时,也需要有相关的手法进行测试。当然我们也可以祈祷别遇见他,因为后续的各种猜解会让人难以接受,就像我们暗恋的那个......
  • 【五期邹昱夫】CCF-B(IEEE Access'19)Badnets: Evaluating backdooring attacks on deep
    "Gu,Tianyu,etal."Badnets:Evaluatingbackdooringattacksondeepneuralnetworks."IEEEAccess7(2019):47230-47244."  本文提出了外包机器学习时选择值得信赖的提供商的重要性,以及确保神经网络模型安全地托管和从在线存储库下载的重要性。并展示了迁移学习场......
  • F5APM第七期Network Access模式配置
    F5APM第七期NetworkAccess模式配置’......
  • F5APM第六期Portal Access模式配置
    F5APM第六期PortalAccess模式配置Portal_conn......