【win編程0003】- 動態庫

来源:https://www.cnblogs.com/awmking/archive/2023/01/02/17019311.html
-Advertisement-
Play Games

動態鏈接庫(dynamic link library)介紹 代碼放到exe中,肯定會造成磁碟冗餘; 電腦ABCD四個軟體,lib加入到代碼中不是在編譯期進入的,而是在運行期 (A進程啟動,把dll加入到A進程中……),編譯的時候不需要這份代碼, 尾碼是.dll 如果要更新軟體,把dll換掉就可以了, ...


 

動態鏈接庫(dynamic link library)介紹

代碼放到exe中,肯定會造成磁碟冗餘; 電腦ABCD四個軟體,lib加入到代碼中不是在編譯期進入的,而是在運行期 (A進程啟動,把dll加入到A進程中……),編譯的時候不需要這份代碼, 尾碼是.dll

如果要更新軟體,把dll換掉就可以了,所有軟體都會跟著更新

 

分析DLL的工具

.dll中凡是能給外面用的函數叫 導出函數;

這個工具能看導出函數;

調win api的時候跟不到源碼中,因為放到了dll中了

 

 

動態庫的創建

 

文件導入並去掉巨集,巨集在靜態庫和動態庫表現不一致

 

直接編譯,生成dll文件,dll文件用工具打開, 然而並沒有導出函數顯示

 

導出函數

編譯之後,用工具打開發現 預設的沒有導出函數, 需要手動指定後,才導出;

就像類一樣,預設時私有的;

用編譯器命令導出

 

註意細節1:

兩個cpp文件中把頭文件註釋掉了,編譯之後,在工具中發現沒有導出函數,為什麼?

.h在預處理時期,預編譯時把頭文件中東西拷貝到.cpp文件中,沒有包含頭文件就沒有導出函數了

 

註意細節2:

再把頭文件加上,有了導出函數之後,編譯之後生成了lib; 這裡和靜態的lib有區別,這裡文件大小2kb;

這裡只是一個描述性的lib,裡面沒有代碼,這裡描述dll中有哪些導出函數,是給開發者使用的

 

 

使用dll

使用dll, 頭文件的這裡改成dllimport, 告訴編譯器這裡用的是dll中的函數;

編譯有誤,找不到函數是在哪個dll文件中;

 

 

問題: 一個.cpp 兩個.h聲明,通過這些能判斷這兩個函數要載入哪個dll嗎?

此時就用到了描述性的lib文件,拷到目錄下(類似Makefile)

和靜態的lib用法相同

編譯的時候不需要dll,運行的時候需要;這裡是運行狀態; dll拿出來放到Debug文件里

 

1-導出全局變數

全局變數可以導出

VS的bug: dll中的全局變數,在VS的監視視窗中看不到值

 

 

2-導出類

類可以導出

但這樣寫不能導出,報錯內容如下:

這樣寫就可以了

工具輸出的內容,上面兩個是構造還是有重載,下麵那個是自己寫的類

 

這個時候斷點跟源碼,能跟到自己寫的庫里,因為調試信息都放到.pdb文件里了;把這個刪了就跟不進去了;

 

 

導出和導出函數的巨集處理

這個東西,每次要用都要改,文件多了就很麻煩,用巨集給它安排下

源碼中 : 把巨集的開關放到這裡,全局的巨集放置的位置

 

 

 

 

工具里導出的都是C++的符號,如果C的呢?

 

 

 

隱式/顯示載入方式

隱式載入方式:

lib + .h (函數 全局變數 類 都可以用)

 

顯示載入方式:

Ollydebug中強大的是插件,很多的dll,使用某個功能的時候,就會去載入dll

Ollydebug發佈的時候,有沒有用人家寫的插件dll的lib和.h?沒有啊, 所以它的載入方式和這裡隱式載入方式不同,Ollydebug是顯示的載入方式;

並不知道別人寫的插件叫啥名字,是怎麼載入別人寫的dll呢?

做法: 當前路徑下的某個文件夾,遍歷文件夾下尾碼為.dll的文件,載入到進程; 這時候的載入,只要有個文件名和文件路徑就可以載入,顯示載入 (函數 全局變數 類(很麻煩))

 

 

使用dll,調用導出函數的方法(Ollydebug載入dll的做法)

載入dll(載入到進程記憶體中) -> 獲取導出函數的地址 -> 然後調用(通過函數地址的方式調用)-> 從進程中卸掉dll

01載入dll的api

參數: 的module指的是 dll / exe(主模塊) 的地址 ;

返回值: 模塊句柄和HINSTANCE兩個是等價的;

HINSTANCE是可執行文件exe在記憶體中的首地址

HMODULE是dll在記憶體中的首地址

作用: 把dll載入到進程記憶體中

code:

註意細節:

調用多個 loadLibrary, 不會調用多個dll,而是句柄引用計數的值會增加;

調用兩個loadLibrary,也要有兩個FreeLibrary才能卸載掉;

02獲取函數地址

模塊句柄還有函數名填進去,返回的就是函數地址;

code: 用的函數指針接收

註意全局變數:

如果要拿全局變數怎麼訪問? 這裡拿到的是地址

全局變數這樣訪問,指針p保存的是地址 *p訪問值

 

 

調用: 普通調用就行

一步一步調試看現象

03從進程中卸掉dll

code:

 

code

#include <iostream>
#include <windows.h>

using PFN_ADD = int(*)(int, int);

int main()
{
    HMODULE hDll = LoadLibrary(R"(C:\Users\Yuna\Desktop\CR2\windows\win01\UseDll\Debug\dll.dll)");
    if (NULL == hDll)
    { 
        std::cout << "載入失敗" << std::endl;
        return 0; 
    }

    PFN_ADD pfnAdd = (PFN_ADD)GetProcAddress(hDll,"Add");
    if (NULL == pfnAdd)
    {
        std::cout << "獲取函數地址失敗" << std::endl;
        return 0;
    }

    int nRet = pfnAdd(1,4);
    int i = 0;  //為了調試上一句
    FreeLibrary(hDll);
} 

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

-Advertisement-
Play Games
更多相關文章
  • 家居網購項目實現011 以下皆為部分代碼,詳見 https://github.com/liyuelian/furniture_mall.git 27.功能25-事務管理 27.1下訂單問題思考 在生成訂單的功能中,系統會去同時修改資料庫中的order,order_item,furn三張表,如果有任意 ...
  • 前言 在系統運行過程中,可能由於一些配置項的簡單變動需要重新打包啟停項目,這對於在運行中的項目會造成數據丟失,客戶操作無響應等情況發生,針對這類情況對開發框架進行升級提供yml文件實時修改更新功能 項目依賴 項目基於的是2.0.0.RELEASE版本,所以snakeyaml需要單獨引入,高版本已包含 ...
  • 一、多態 多態指的是一類事物有多種形態,一個類有很多個子類,因而多態的概念是基於繼承的 序列數據類型有多種形態:字元串,列表,元組 動物有多種形態:人,狗,豬 1.動物的多種形態 # 動物有多種形態:人類、豬、狗 class Animal: def run(self): # 子類約定俗稱的必須實現這 ...
  • 企業發放的獎金根據利潤提成。 利潤(I)低於或等於10萬元時,獎金可提10%; 利潤高於10萬元,低於20萬元時,低於10萬元的部分按10%提成,高於10萬元的部分,可提成7.5%; 20萬到40萬之間時,高於20萬元的部分,可提成5%; 40萬到60萬之間時高於40萬元的部分,可提成3%; 60萬 ...
  • JZ75 字元流中第一個不重覆的字元 題目 請實現一個函數用來找出字元流中第一個只出現一次的字元。例如,當從字元流中只讀出前兩個字元 "go" 時,第一個只出現一次的字元是 "g"。 當從該字元流中讀出前六個字元 “google" 時,第一個只出現一次的字元是"l"。 方法1 使用LinkedHas ...
  • 訓練模型過程中,經常需要追蹤一些性能指標的變化情況,以便瞭解模型的實時動態,例如:回歸任務中的MSE、分類任務中的Accuracy、生成對抗網路中的圖片、網路模型結構可視化…… 除了追蹤外,我們還希望能夠將這些指標以動態圖表的形式可視化顯示出來。 TensorFlow的附加工具Tensorboar... ...
  • 本文介紹 C++11 標準中新添加的 long long 超長整型和 nullptr 初始化空指針。 1. C++11:long long 超長整型 C++ 11 標準中,基於整數大小的考慮,共提供瞭如下表所示的這些數據類型。與此同時,標準中還明確限定了各個數據類型最少占用的位數。 | 整數類型 | ...
  • 今天在寫一個通訊錄實現程式的時候,遇到個讓我突然卡殼的問題,不知道怎麼進行兩個結構體之間的成員互換......結構體成員有“姓名”,“性別”,“年齡”,“地址”,“電話”,目的就是實現一個通過年齡進行sort排序的功能,作為一個努力學習的編程小白來說,有太多的東西需要學習了..........代碼如 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...