读取与修改其他程序的数据Read/WriteProcessMemory[通俗易懂]

读取与修改其他程序的数据Read/WriteProcessMemory[通俗易懂]要修改或读取其他进程的数据,首先要知道几个知识:一、1.windows系统为每个程序分配4GB的虚拟内存,虚拟内存由“页文件”实现。     2.每个程序的4GB空间的前2GB是程序的私有空间,后2GB是系统的空间。     3.每个页文件4KB。     4.在程序私有的2GB中,windows98系列的系统的程序的可用地址为4MB–2GB

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

要修改或读取其他进程的数据,首先要知道几个知识:

一、1.windows系统为每个程序分配4GB的虚拟内存,虚拟内存由“页文件”实现。

      2.每个程序的4GB空间的前2GB是程序的私有空间,后2GB是系统的空间。

      3.每个页文件4KB。

      4.在程序私有的2GB中,windows 98系列的系统的程序的可用地址为4MB–2GB

                                       windows 2000系列的系统的程序的可用地址为64KB–2GB

         因此还需要先判别程序运行的操作系统。

二、1.查阅MSDN可知,Read/WriteProcessMemory都需要一定的权限,

         因此打开句柄时必须赋予相应权限。

 

这个程序是《Windows程序设计》——王艳平著的第二章的一个程序

程序的具体实现过程是:

     创建一个子进程执行一个自己写的测试程序,

     然后得到有读权限的子进程的句柄,搜索要改的数据的内存,最后修改。

 

原程序稍有错误,原程序没有取得读和写权限,所以GetLastError会返回998——内存位置访问无效。

                                                                                               和5——拒绝访问。  

 

修改后程序正确执行,但是在读取一些不可用内存地址时会有229错误(会有很多,是正常的)

——仅完成部分的 ReadProcessMemory 或WriteProcessMemory 请求。

 

 

修改后的源代码如下:

/****以下是02MemRepair.cpp中的代码****/

#include <stdio.h>
#include <windows.h>

BOOL FindFirst(DWORD dwValue); // 在目标进程空间进行第一次查找
BOOL CompareAPage(DWORD dwBaseAddr,DWORD dwValue);// 比较目标进程内存一页中比较
BOOL FindNext(DWORD dwValue); // 在目标进程空间进行第二三四查找
void ShowList();    // 打印搜索出的地址
DWORD g_arList[1024];   // 存放查找到的地址列表
int g_nListCnt;     // 有效地址个数
HANDLE g_hProcess;    // 目标进程句柄
DWORD dError;     // 错误代码

int main(int argc,char* argv[])
{

 // 启动测试程序
 char szFileName[] = “..//02TestMemRepair//debug//02TestMemRepair.exe”;
 STARTUPINFO si = {sizeof(STARTUPINFO)};
 PROCESS_INFORMATION pi;
 ::CreateProcess(NULL,szFileName,NULL,NULL,FALSE,CREATE_NEW_CONSOLE,NULL,NULL,
  &si,&pi);
 ::CloseHandle(pi.hThread);
 // 打开句柄要给予能读的权限,据MSDN知最少是PROCESS_VM_READ
 g_hProcess = ::OpenProcess(PROCESS_ALL_ACCESS,FALSE,pi.dwProcessId);
 
 // 输入要修改的值
 int iVal; 
 printf(” 想要修改的数据现在的值是:”);
 scanf(“%d”,&iVal);

 // 进行第一次查找
 if(!FindFirst(iVal))
 { 
  printf(“FindFirst失败”);
  getchar();

  return FALSE;
 }

 // 打印搜索出的结果
 ShowList();
 
 // 若搜索结果不唯一,再次进行搜索
 while(g_nListCnt>1)
 {

  printf(“/n 搜索结果不唯一,改变数据的值后再次搜索,新值为:”);
  scanf(“%d”,&iVal);
  if(FindNext(iVal))
   ShowList();
  else
  {

   printf(” FindNext失败”);
   return FALSE;
  }
 }

 // 搜索结果唯一,进行修改
 printf(“/n 想要修改成的值为:”);
 scanf(“%d”,&iVal);
 if(::WriteProcessMemory(g_hProcess,(LPVOID)g_arList[0],&iVal,sizeof(iVal),NULL))
  printf(“/n 修改成功!/n”);
 else
 {

  printf(” WriteProcessMemory失败”);
  return FALSE;
 }
 
 ::CloseHandle(g_hProcess);
 return 0;
}

BOOL FindFirst(DWORD dwValue)
{

 const DWORD dwOneGB = 1024*1024*1024; // 1GB=1073741824 
 const DWORD dwOnePage = 4*1024;   // 4KB=4096
 if(g_hProcess == NULL)
 {

  printf(” 要读取的进程不存在/n”);
  return FALSE;
 }
 // 查看操作系统类型,以决定开始地址
 DWORD dwBase;
 OSVERSIONINFO vi = {sizeof(OSVERSIONINFO)};
 ::GetVersionEx(&vi);
 if(vi.dwPlatformId==VER_PLATFORM_WIN32_WINDOWS)
 {

  printf(” 操作系统为98系列/n”);
  dwBase = 4*1024*1024; // 98系列,4MB=4194304
 }
 else
 {

  printf(” 操作系统为NT系列/n”);
  dwBase = 65536;// NT系列,64KB=65536
 }
 

 // 在开始地址到2GB的地址空间进行查找
 for(;dwBase<2*dwOneGB;dwBase+=dwOnePage)
 { // 每次在一页中查找
  
  if(CompareAPage(dwBase,dwValue))
  {

   printf(” CompareAPage成功/n”);
  }
  else
  {

   printf(” CompareAPage失败/t%d/n”,GetLastError());
   continue;
  }
 }
 return TRUE;
}

BOOL CompareAPage(DWORD dwBaseAddr,DWORD dwValue)
{

 // 读取一页的内存
 BYTE arByte[4096];
 if(::ReadProcessMemory(g_hProcess,(LPCVOID)dwBaseAddr,arByte,4096,NULL))
 {

  printf(” ReadProcessMemory成功/n”);
  int* pdw;
  for(int i=0;i<4096;i++)
  {

   pdw = (int*)&arByte[i];
   if(*pdw==dwValue) // 如果等于要查找的值
   {

    if(g_nListCnt>=1024)
    {

     printf(” 找到的地址过多/n”);
     return FALSE;
    }
    //添加到全局变量中
    g_arList[g_nListCnt++] = dwBaseAddr+i;
   }
  }
 }
 else
 {

  printf(” ReadProcessMemory失败/t%d/n”,::GetLastError());
  return FALSE;  // 这页不可读
 }
 // 在这一页中查找
 return TRUE;
}

BOOL FindNext(DWORD dwValue)
{

 int nOrgCnt = g_nListCnt;
 g_nListCnt = 0;
 DWORD dwSecondRead;  // 存储从地址中取出的值,与新值dwValue比较
 // 在已经搜索出来的地址中再次查找
 for(int m=0;m<nOrgCnt;m++)
 {

  if(::ReadProcessMemory(g_hProcess,(LPCVOID)g_arList[m],&dwSecondRead,sizeof(DWORD),NULL))
  {

   if(dwSecondRead==dwValue)
    g_arList[g_nListCnt++] = g_arList[m];
  }
  else
  {

   printf(” ReadProcessMemory失败”);
   return FALSE;
  }
  return TRUE;
 }
}

void ShowList()
{

 for(int i=0;i<g_nListCnt;i++)
 {

  printf(“%d:%08X/n”,i+1,g_arList[i]);
 }
 printf(“一共找到%d个地址/n”,g_nListCnt);
}

/****以上是02MemRepair.cpp中的代码****/ 

 

/****以下是02TestMemRepair.cpp中的代码****/
#include <stdio.h>
int g_nNum;  // 全局变量测试
int main(int argc,char* argv[])
{

 int i = 198; // 局部变量测试
 g_nNum = 1003;
 while(1)
 {

  //
  printf(” i = %d,addr = %08X;  g_nNum = %d,addr = %08X/n”,++i,&i,–g_nNum,&g_nNum);
  getchar();
 }
 return 0;
}

/****以上是02MemRepair.cpp中的代码****/

 

/****以下是02TestMemRepair.cpp中的代码****/

 

 #include <stdio.h>
int g_nNum;  // 全局变量测试
int main(int argc,char* argv[])
{

 int i = 198; // 局部变量测试
 g_nNum = 1003;
 while(1)
 {

  //
  printf(” i = %d,addr = %08X;  g_nNum = %d,addr = %08X/n”,++i,&i,–g_nNum,&g_nNum);
  getchar();
 }
 return 0;
}

/****以上是02TestMemRepair.cpp中的代码****/

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/194418.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • json到底是什么(c# json)

    &gt;如果点进来的刚巧是个高手,那……你就当是过来做个鉴定的吧。很多从事数据分析工作的伙伴都是从“最简单”的神器—Excel开始入门的,对编程语言、计算机世界的很多工具都不甚了解。要做结构化的数据,就赶紧跑去自学sql,学习sqlserver或是mysql数据库。要做半结构化、非结构化的数据又得去学习诸如mongo数据库的东西。&gt;天天听对面的程序员小哥说我给你个打包个j…

    2022年4月16日
    49
  • Android中如何使用单选对话框[通俗易懂]

    Android中如何使用单选对话框[通俗易懂]给Button设置OnClick事件设置intid=0;finalString[]s={“单选A”,”单选B”,”单选C”,”单选D”,”单选E”};AlertDialog.BuilderDanItem=newAlertDialog.Builder(MainActivity.this);DanItem.setTitle(“单选列表”);…

    2022年5月30日
    36
  • Pycharm踩坑(一) FileNotFoundError: [Errno 2] No such file or directory: ‘../data/users.txt‘ 目录结构「建议收藏」

    Pycharm踩坑(一) FileNotFoundError: [Errno 2] No such file or directory: ‘../data/users.txt‘ 目录结构「建议收藏」Python使用Pycharm运行程序提示:FileNotFoundError:[Errno2]Nosuchfileordirectory:’../data/users.txt’目录结构test/├──data└──testcases└──test.py在test.py中运行代码:withopen(‘../data/users.txt’)asf…

    2022年6月24日
    77
  • HTML与XML的区别(转)

    什么是HTML  HTML的全拼是Hypertext Markup Language, 中文也就是超文本链接标示语言。HTML(HyperTextMark-upLanguage)即超文本标记语言,是WWW的描述语言。设计HTML语言的目的是为了能把存放在一台电脑中的文本或图形与另一台电脑中的文本或图形方便地联系在一起,形成有机的整体,人们不用考虑具体信息是在当前电脑上还是在网络的其它电脑上。我们只需

    2022年2月25日
    53
  • Java Web 网络商城案例演示十五 订单详情功能(提交订单支付界面)

    Java Web 网络商城案例演示十五 订单详情功能(提交订单支付界面)订单详情功能(提交订单支付界面)原理分析步骤实现:1、准备工作:order_list.jsp当中修改链接提交当前订单编号<ahref=”${pageContext.request.contextPath}/OrderServlet?method=findOrderByOid&oid=${o.oid}”>付款</a>2、OrderServlet…

    2022年5月27日
    46
  • MySQL优化原理

    MySQL优化原理

    2021年11月4日
    41

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号