C# 操作地址 從記憶體中讀取寫入數據(初級)

来源:https://www.cnblogs.com/flamegreat/archive/2020/01/15/12195949.html
-Advertisement-
Play Games

本示例以植物大戰僵屍為例, 實現功能為 每1秒讓陽光刷新為 9999.本示例使用的游戲版本為 [植物大戰僵屍2010年度版], 使用的輔助查看記憶體地址的工具是 CE. 由於每次啟動游戲, 游戲中陽光地址都是變的, 唯一不變的基址1, 我們要通過CE工具找到基址1的地址, 可以算出陽光的地址. 基址2 ...


本示例以植物大戰僵屍為例, 實現功能為 每1秒讓陽光刷新為 9999.本示例使用的游戲版本為 [植物大戰僵屍2010年度版], 使用的輔助查看記憶體地址的工具是  CE.

由於每次啟動游戲, 游戲中陽光地址都是變的, 唯一不變的基址1, 我們要通過CE工具找到基址1的地址, 可以算出陽光的地址.

基址2的地址 = 基址1中的值 + 偏移1;

陽光的的地址 = 基址2中的值 + 偏移2;

以下為簡單示例:  視窗界面一個按鈕 和 一個定時器

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
using System.Runtime.InteropServices;
using System.Diagnostics;
 
namespace ZhiWuDaZhanJiangShi
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }
 
        #region API
 
        //從指定記憶體中讀取位元組集數據
        [DllImportAttribute("kernel32.dll", EntryPoint = "ReadProcessMemory")]
        public static extern bool ReadProcessMemory(IntPtr hProcess,IntPtr lpBaseAddress,IntPtr lpBuffer,int nSize,IntPtr lpNumberOfBytesRead);
 
        //從指定記憶體中寫入位元組集數據
        [DllImportAttribute("kernel32.dll", EntryPoint = "WriteProcessMemory")]
        public static extern bool WriteProcessMemory(IntPtr hProcess,IntPtr lpBaseAddress,int[] lpBuffer,int nSize, IntPtr lpNumberOfBytesWritten );
 
        //打開一個已存在的進程對象,並返回進程的句柄
        [DllImportAttribute("kernel32.dll", EntryPoint = "OpenProcess")]
        public static extern IntPtr OpenProcess(int dwDesiredAccess, bool bInheritHandle, int dwProcessId);
 
        //關閉一個內核對象。其中包括文件、文件映射、進程、線程、安全和同步對象等。
        [DllImport("kernel32.dll")]
        private static extern void CloseHandle(IntPtr hObject);
 
        #endregion
 
        #region 使用方法
 
        //根據進程名獲取PID
        public static int GetPidByProcessName(string processName)
        {
            Process[] arrayProcess = Process.GetProcessesByName(processName);
            foreach (Process p in arrayProcess)
            {
                return p.Id;
            }
            return 0;
        }
 
        //讀取記憶體中的值
        public static int ReadMemoryValue(int baseAddress, string processName)
        {
            try
            {
                byte[] buffer = new byte[4];
                //獲取緩衝區地址
                IntPtr byteAddress = Marshal.UnsafeAddrOfPinnedArrayElement(buffer, 0);
                //打開一個已存在的進程對象  0x1F0FFF 最高許可權
                IntPtr hProcess = OpenProcess(0x1F0FFF, false, GetPidByProcessName(processName));
                //將制定記憶體中的值讀入緩衝區
                ReadProcessMemory(hProcess, (IntPtr)baseAddress, byteAddress, 4, IntPtr.Zero); 
                //關閉操作
                CloseHandle(hProcess);
                //從非托管記憶體中讀取一個 32 位帶符號整數。
                return Marshal.ReadInt32(byteAddress);
            }
            catch
            {
                return 0;
            }
        }
 
        //將值寫入指定記憶體地址中
        public static void WriteMemoryValue(int baseAddress, string processName, int value)
        {
            try
            {
                //打開一個已存在的進程對象  0x1F0FFF 最高許可權
                IntPtr hProcess = OpenProcess(0x1F0FFF, false, GetPidByProcessName(processName));
                //從指定記憶體中寫入位元組集數據
                WriteProcessMemory(hProcess, (IntPtr)baseAddress, new int[] { value }, 4, IntPtr.Zero);
                //關閉操作
                CloseHandle(hProcess);
            }
            catch { }
        }
 
        #endregion
 
        //游戲記憶體基址
        private int baseAddress = 0x0015E944;
        //游戲進程名字
        private string processName = "PlantsVsZombies";
 
        //開啟/關閉 功能 的按鈕
        private void button1_Click(object sender, EventArgs e)
        {
            if (GetPidByProcessName(processName) == 0)
            {
                MessageBox.Show("游戲沒有運行!");
                return;
            }
            if (button1.Text == "開啟")
            {
                button1.Text = "關閉";
                timer1.Enabled = true;
            }
            else
            {
                button1.Text = "開啟";
                timer1.Enabled = false;
            }
        }
 
        //定時器
        private void timer1_Tick(object sender, EventArgs e)
        {
            if (GetPidByProcessName(processName) == 0)
            {
                timer1.Enabled = false;
            }
            //baseAddress : 游戲記憶體基址   processName : 游戲進程名
            //讀取 基址1 中存放的值
            int address = ReadMemoryValue(baseAddress, processName);
            //計算 基址2的地址 = 基址1中的值 + 偏移量1
            address = address + 0x868;
            //讀取 基址2 中存放的值
            address = ReadMemoryValue(address, processName);
            //計算 陽光的地址 = 基址2中的值 + 偏移量2
            address = address + 0x5578;
            //給陽光地址中寫入數值,0x378 : 888
            WriteMemoryValue(address, processName, 0x378);
        } 
    }
}

下麵增加了一個刷新金幣的示例 (註意, 金幣值是界面的金幣輸 除以 10 , 我們刷100 在界面裡面顯示為1000)

private void button2_Click(object sender, EventArgs e)
        {
            if (GetPidByProcessName(processName) == 0)
            {
                MessageBox.Show("游戲沒有運行!");
                return;
            }
 
            //baseAddress : 游戲記憶體基址   processName : 游戲進程名
            //讀取 基址1 中存放的值
            int address = ReadMemoryValue(baseAddress, processName);
            //計算 基址2的地址 = 基址1中的值 + 偏移量1
            address = address + 0x950;
            //讀取 基址2 中存放的值
            address = ReadMemoryValue(address, processName);
            //計算 陽光的地址 = 基址2中的值 + 偏移量2
            address = address + 0x50;
            //給陽光地址中寫入數值,0x378 : 888
            WriteMemoryValue(address, processName, GetInt(textBox2.Text));
        }
 
        private int GetInt(string s)
        {
            int n = 0;
            int.TryParse(s, out n);
            if (n <= 0)
            {
                n = 100;
            }
            return n;
        }

今天又增加了無植物無冷卻時間的功能

int count = 0;
        private void Form1_Load(object sender, EventArgs e)
        {
            if (GetPidByProcessName(processName) != 0)
            {
                int address = ReadMemoryValue(baseAddress, processName);
                address = address + 0x868;
                address = ReadMemoryValue(address, processName);
                address = address + 0x15c;
                address = ReadMemoryValue(address, processName);
                address = address + 0x24;
                address = ReadMemoryValue(address, processName);
                count = address;
                label3.Text = "植物欄個數: " + address.ToString() + "";
            }
        }
 
        private void timer2_Tick(object sender, EventArgs e)
        {
            if (GetPidByProcessName(processName) == 0)
            {
                timer2.Enabled = false;
            }
            if (count > 0)
            {
                for (int i = 0; i < count; i++)
                {
                    int address = ReadMemoryValue(baseAddress, processName);
                    address = address + 0x868;//一級地址
                    address = ReadMemoryValue(address, processName);
                    address = address + 0x15c;//二級地址
                    address = ReadMemoryValue(address, processName);
                    address = address + 0x4c;//第一欄 植物的地址
                    // 每後一個植物 地址 偏移 50 (在十進位里是80)
                    //偏移 0x24 的地址 是標示是否在冷卻中 值 :( 0 :  為冷卻中, 1 為冷卻完成)
                    address = address + 80 * i + 0x24;
                    WriteMemoryValue(address, processName, 1);
                    //如果不偏移 0x24 的地址為冷卻時間地址, 值不確定, 一般最大設為6000  也可以完成此功能
                    //address = address + 80 * i;
                    //WriteMemoryValue(address, processName, 6000);
                }
            }
            else
            {
                int address = ReadMemoryValue(baseAddress, processName);
                address = address + 0x868;
                address = ReadMemoryValue(address, processName);
                address = address + 0x15c;
                address = ReadMemoryValue(address, processName);
                address = address + 0x24;
                address = ReadMemoryValue(address, processName);
                count = address;
                label3.Text = "植物欄個數: " + address.ToString() + "";
            }
        }
 
        private void button3_Click(object sender, EventArgs e)
        {
            if (GetPidByProcessName(processName) == 0)
            {
                MessageBox.Show("游戲沒有運行!");
                return;
            }
            if (button3.Text == "有冷卻")
            {
                button3.Text = "無冷卻";
                timer2.Enabled = true;
            }
            else
            {
                button3.Text = "有冷卻";
                timer2.Enabled = false;
            }
        }

轉自:https://blog.csdn.net/changblade/article/details/82027440


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 第一步:在打開的VS2015中,右擊項目,在彈出的對話框中,點擊“發佈”。 第二步:配置發佈的WEB項目。 1.為要發佈的項目起個名稱,一般和項目名稱相同。 2.選擇以文件系統發佈。 3.設置相關配置。 第三步:發佈預覽,點擊“發佈”按鈕,會在設定的目錄下,生成發佈的打包文件。 ...
  • 概述 ScheduleMaster 是一個開源的分散式任務調度系統,它基於Asp.Net Core平臺構建,支持跨平臺多節點部署運行。 它的項目主頁在這裡: "https://github.com/hey hoho/ScheduleMasterCore" 關於它的簡單介紹可以看這裡: "https: ...
  • 在寫了很多年.NET程式之後,年長的猿類在面對非同步編程時,仍不時會犯下致命錯誤,乃至被拖出去殺了祭天。本篇就async/await中的Exception處理進行討論,為種族的繁衍生息做出貢獻……處理async/await中的Exception,最致命的莫過於想抓的Exception抓不到,程式崩的莫 ...
  • 本次主要採用Asp.net core3.1來實現API服務(優勢:跨平臺),其中此技術比較新穎,其中連接Oracle遇到各種困難(特別是資料較少,各個插件版本比較多,且依賴的版本也有要求),經過各種嘗試,最終可以連接上Oracle資料庫,此處僅僅是demo。 ...
  • 1、傳多個參數 介面定義:(ResponseFormat與RequestFormat分別將相應參數序列化、請求參數反序列化) [OperationContract] [WebInvoke(UriTemplate = "api/fun2", Method = "POST", BodyStyle = W ...
  • 1、.Net Core介紹 .NET Core是.NET Framework的新一代版本, 是微軟開發的第一個具有跨平臺(Windows、Macosx、Linux) 能力的應用程式開發框架,未來也將會支持FreeBSD與Alpine平臺,是微軟在一開始發展時就開源的軟體平臺,它也經常被拿來和現有的開 ...
  • 一、泛型 泛型就是封裝,將重覆的工作簡單化 1.泛型方法 public static void Show<T>(T tParameter) { Console.WriteLine("This is {0}, parameter = {1}, type = {2}", typeof(CommonMet ...
  • 驗證碼(Captcha)基於十道安全柵欄, 為網頁、App、小程式開發者打造立體、全面的人機驗證,最大程度地保護註冊登錄、活動秒殺、點贊發帖、數據保護等各大場景下的業務安全。要做自動化腳本程式,就要能識別驗證碼這一關。 效果如下圖: ​ 演示程式結構在vs2019創建WinForm窗體程式,添加相應 ...
一周排行
    -Advertisement-
    Play Games
  • GoF之工廠模式 @目錄GoF之工廠模式每博一文案1. 簡單說明“23種設計模式”1.2 介紹工廠模式的三種形態1.3 簡單工廠模式(靜態工廠模式)1.3.1 簡單工廠模式的優缺點:1.4 工廠方法模式1.4.1 工廠方法模式的優缺點:1.5 抽象工廠模式1.6 抽象工廠模式的優缺點:2. 總結:3 ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 本章將和大家分享ES的數據同步方案和ES集群相關知識。廢話不多說,下麵我們直接進入主題。 一、ES數據同步 1、數據同步問題 Elasticsearch中的酒店數據來自於mysql資料庫,因此mysql數據發生改變時,Elasticsearch也必須跟著改變,這個就是Elasticsearch與my ...
  • 引言 在我們之前的文章中介紹過使用Bogus生成模擬測試數據,今天來講解一下功能更加強大自動生成測試數據的工具的庫"AutoFixture"。 什麼是AutoFixture? AutoFixture 是一個針對 .NET 的開源庫,旨在最大程度地減少單元測試中的“安排(Arrange)”階段,以提高 ...
  • 經過前面幾個部分學習,相信學過的同學已經能夠掌握 .NET Emit 這種中間語言,並能使得它來編寫一些應用,以提高程式的性能。隨著 IL 指令篇的結束,本系列也已經接近尾聲,在這接近結束的最後,會提供幾個可供直接使用的示例,以供大伙分析或使用在項目中。 ...
  • 當從不同來源導入Excel數據時,可能存在重覆的記錄。為了確保數據的準確性,通常需要刪除這些重覆的行。手動查找並刪除可能會非常耗費時間,而通過編程腳本則可以實現在短時間內處理大量數據。本文將提供一個使用C# 快速查找並刪除Excel重覆項的免費解決方案。 以下是實現步驟: 1. 首先安裝免費.NET ...
  • C++ 異常處理 C++ 異常處理機制允許程式在運行時處理錯誤或意外情況。它提供了捕獲和處理錯誤的一種結構化方式,使程式更加健壯和可靠。 異常處理的基本概念: 異常: 程式在運行時發生的錯誤或意外情況。 拋出異常: 使用 throw 關鍵字將異常傳遞給調用堆棧。 捕獲異常: 使用 try-catch ...
  • 優秀且經驗豐富的Java開發人員的特征之一是對API的廣泛瞭解,包括JDK和第三方庫。 我花了很多時間來學習API,尤其是在閱讀了Effective Java 3rd Edition之後 ,Joshua Bloch建議在Java 3rd Edition中使用現有的API進行開發,而不是為常見的東西編 ...
  • 框架 · 使用laravel框架,原因:tp的框架路由和orm沒有laravel好用 · 使用強制路由,方便介面多時,分多版本,分文件夾等操作 介面 · 介面開發註意欄位類型,欄位是int,查詢成功失敗都要返回int(對接java等強類型語言方便) · 查詢介面用GET、其他用POST 代碼 · 所 ...
  • 正文 下午找企業的人去鎮上做貸後。 車上聽同事跟那個司機對罵,火星子都快出來了。司機跟那同事更熟一些,連我在內一共就三個人,同事那一手指桑罵槐給我都聽愣了。司機也是老社會人了,馬上聽出來了,為那個無辜的企業經辦人辯護,實際上是為自己辯護。 “這個事情你不能怪企業。”“但他們總不能讓銀行的人全權負責, ...