c#尝试写入或者读取受保护的内存_C#解决"Additional information: 尝试读取或写入受保护的内存。这通常指示其他内存已损坏。"...

今天小编用C#和VB.NET调用DLL动态链接库的时候,都出现了"Additional information: 尝试读取或写入受保护的内存。这通常指示其他内存已损坏。"这样的错误,Visual Studio 2015的具体错误代码如下:An unhandled exception of type 'System.AccessViolationException' occurred in WindowsFormsApplication1.exe

Additional information: 尝试读取或写入受保护的内存。这通常指示其他内存已损坏。

小编在C#中开始是这样的引入DLL的,参数分别是long类型与结构体类型,本身和DLL api里面的类型一样,应该是没问题的,如图所示:

于是小编欣喜若狂的开始使用这个hst_open方法了,小编是这样调用的,于是就出现了上述的错误,如图:

解决方案:

小编最后用C++语言同样的调用DLL动态链接库做了实验,发现虽然没有报这种错误,但是发现hst_open方法的这两个参数类型竟然是错误的!后来小编将它的两个参数类型做了一些改变,例如将hst_open方法第一个参数类型改成int类型,将第二个参数改成ref引用类型,如图所示:

小编改变DLL方法参数类型之后,在C#调用DLL动态链接库方法的时候,传值的时候自然也要做一些改变,引用类型的值要加入“ref”关键字,值类型(如com)则直接传入数字或变量名即可,如图所示:

如果在VB.NET语言中调用DLL动态链接库出现"Additional information: 尝试读取或写入受保护的内存。这通常指示其他内存已损坏。"的错误,该如何解决呢?

其实万变不离其宗,出现这样的错误也是因为在调用dll的时候参数类型不对,一定要注意参数是ByVal(值类型)还是ByRef(引用类型),于是同样的错误,小编将参数改成如图所示就可以了,如下:

在结构体前面加入ByRef引用类型标志,第一个参数即使是值类型,也要看是Integer还是Long,因为类型的正确与否对于DLL的调用是非常重要的,然后在VB.NET中想要调用它,传值时可以直接传入值和结构体,如图:

中心思想:

调用dll动态链接库的时候,参数类型是非常重要的,如果类型不对,就会出现本文这样的错误。

来源网站:太平洋学习网,转载请注明出处:http://www.tpyyes.com/a/csharp/704.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
对于读写保护内存C#可以通过调用Windows API函数来实现。需要使用到的函数有OpenProcess、ReadProcessMemory和WriteProcessMemory。 首先需要打开进程,使用OpenProcess函数打开目标进程,获取到进程的句柄。进程句柄可以通过进程ID来获取,具体代码如下: ``` [DllImport("kernel32.dll")] public static extern IntPtr OpenProcess(int dwDesiredAccess, bool bInheritHandle, int dwProcessId); int processId = 1234; // 目标进程ID IntPtr processHandle = OpenProcess(ProcessAccessFlags.All, false, processId); ``` 其中,dwDesiredAccess参数指定了进程的访问权限,这里设置为All表示获取进程的完全访问权限。bInheritHandle参数表示是否继承句柄,这里设置为false。dwProcessId参数是目标进程的ID。 接着,可以使用ReadProcessMemory函数来读取目标进程的内存。该函数的参数包括进程句柄、读取内存的起始地址、存储读取结果的缓冲区、需要读取的字节数以及实际读取的字节数。具体代码如下: ``` [DllImport("kernel32.dll")] public static extern bool ReadProcessMemory(IntPtr hProcess, IntPtr lpBaseAddress, byte[] lpBuffer, int nSize, out int lpNumberOfBytesRead); IntPtr address = new IntPtr(0x12345678); // 要读取内存地址 byte[] buffer = new byte[4]; // 存储读取结果的缓冲区 int bytesRead = 0; // 实际读取的字节数 bool success = ReadProcessMemory(processHandle, address, buffer, buffer.Length, out bytesRead); ``` 需要注意的是,由于读取内存保护的,可能会引发访问异常。可以通过设置进程的访问权限或使用VirtualProtectEx函数来解决这个问题。 最后,可以使用WriteProcessMemory函数来写入目标进程的内存。该函数的参数和ReadProcessMemory函数类似,具体代码如下: ``` [DllImport("kernel32.dll")] public static extern bool WriteProcessMemory(IntPtr hProcess, IntPtr lpBaseAddress, byte[] lpBuffer, int nSize, out int lpNumberOfBytesWritten); IntPtr address = new IntPtr(0x12345678); // 要写入内存地址 byte[] buffer = new byte[] { 0x01, 0x02, 0x03, 0x04 }; // 要写入的数据 int bytesWritten = 0; // 实际写入的字节数 bool success = WriteProcessMemory(processHandle, address, buffer, buffer.Length, out bytesWritten); ``` 需要注意的是,写入内存也是保护的,可能会引发访问异常。同样可以通过设置进程的访问权限或使用VirtualProtectEx函数来解决这个问题。 以上是C#读写保护内存的基本方法,需要注意的是,这种操作可能会引发系统崩溃或数据丢失等问题,应谨慎使用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值