動態鏈接

来源:https://www.cnblogs.com/linhaostudy/archive/2018/09/08/9610947.html
-Advertisement-
Play Games

動態鏈接 要解決空間浪費和更新困難這兩個問題最簡單的辦法就是把程式的模塊相互分割開來,形成獨立的文件,而不再將它們靜態地鏈接在一起。簡單地講,就是不對那些組成程式的目標文件進行鏈接,等到程式要運行時才進行鏈接。也就是說,把鏈接這個過程推遲到了運行時再進行,這就是動態鏈接( Dynamic Linki ...


動態鏈接

要解決空間浪費和更新困難這兩個問題最簡單的辦法就是把程式的模塊相互分割開來,形成獨立的文件,而不再將它們靜態地鏈接在一起。簡單地講,就是不對那些組成程式的目標文件進行鏈接,等到程式要運行時才進行鏈接。也就是說,把鏈接這個過程推遲到了運行時再進行,這就是動態鏈接( Dynamic Linking)的基本思想。

動態庫的基本實現

動態鏈接的基本思想是把程式按照模塊拆分成各個相對獨立部分,在程式運行時才將它們鏈接在一起形成一個完整的程式,而不是像靜態鏈接一樣把所有的程式模塊都鏈接成一個個單獨的可執行文件。那麼我們能不能按照前面例子中所描述的那樣,直接使用目標文件進行動態鏈接呢?這個問題的答案是:理論上是可行的,但實際上動態鏈接的實現方案與直接使用目標文件稍有差別。我們將在後面分析目標文件和動態鏈接文件的區別。

動態鏈接涉及運行時的鏈接及多個文件的裝載,必需要有操作系統的支持,因為動態鏈接的情況下,進程的虛擬地址空間的分佈會比靜態鏈接情況下更為複雜,還有一些存儲管理、記憶體共用、進程線程等機制在動態鏈接下也會有一些微妙的變化。目前主流的操作系統幾乎都支持動態鏈接這種方式,在 Linux系統中,ELF動態鏈接文件被稱為動態共用對象(DSO, Dynamic Shared objects),簡稱共用對象,它們一般都是以“so”為擴展名的一些文件; 而在 Windows系統中,動態鏈接文件被稱為動態鏈接庫( Dynamical Linking Library),它們通常就是我們平時很常見的以“dll”為擴展名的文件。

從本質上講,普通可執行程式和動態鏈接庫中都包含指令和數據,這一點沒有區別。在使用動態鏈接庫的情況下,程式本身被分為了程式主要模塊( Progran1)和動態鏈接庫( Lib. so),但實際上它們都可以看作是整個程式的一個模塊,所以當我們提到程式模塊時可以指程式主模塊也可以指動態鏈接庫。

在 Linux中,常用的C語言庫的運行庫glib,它的動態鏈接形式的版本保存在“/lib”目錄下,文件名叫做“ libc.so”。整個系統只保留一份C語言庫的動態鏈接文件“libc.so”,而所有的C語言編寫的、動態鏈接的程式都可以在運行時使用它。當程式被裝載的時候,系統的動態鏈接器會將程式所需要的所有動態鏈接庫(最基本的就是libc.so)裝載到進程的地址空間,並且將程式中所有未決議的符號綁定到相應的動態鏈接庫中,併進行重定位工作。

程式與libc.so之間真正的鏈接工作是由動態鏈接器完成的,而不是由我們前面看到過的靜態鏈接器ld完成的。也就是說,動態鏈接是把鏈接這個過程從本來的程式裝載前被推遲到了裝載的時候。可能有人會問,這樣的做法的確很靈活,但是程式每次被裝載時都要進行重新進行鏈接,是不是很慢?的確,動態鏈接會導致程式在性能的一些損失,但是對動態鏈接的鏈接過程可以進行優化,比如我們後面要介紹的延遲綁定( Lazy Binding)等方法,可以使得動態鏈接的性能損失儘可能地減小。據估算,動態鏈接與靜態鏈接相比,性能損失大約在5%以下。當然經過實踐的證明,這點性能損失用來換取程式在空間上的節省和程式構建和升級時的靈活性,是相當值得的。

動態鏈接程式運行時地址空間分佈

對於靜態鏈接的可執行文件來說,整個進程只有一個文件要被映射,那就是可執行文件本身,我們在前面的章節己經介紹了靜態鏈接下的進程虛擬地址空間的分佈。但是對於動態鏈接來說,除了可執行文件本身之外,還有它所依賴的共用目標文件。那麼這種情況下,進程的地址空間分佈又會怎樣呢?

我們還是以上面的 Program1為例,但是當我們試圖運行 Program1並且查看它的進程空間分佈時,程式一運行就結束了。所以我們得對程式做適當的修改,在Libc中的 foobar.c 函數裡面加入sleep函數

然後就可以查看進程的虛擬地址空間分佈:

我們看到,整個進程虛擬地址空間中,多出了幾個文件的映射。Lib.so 與 Program1 一樣,它們都是被操作系統用同樣的方法映射至進程的虛擬地址空間,只是它們占據的虛擬地址和長度不同。 ProgramI除了使用Lb.so以外,它還用到了動態鏈接形式的C語言運行庫libc-2.61so。另外還有一個很值得關註的共用對象就是ld-2.6so,它實際上是 Linux下的動態鏈接器。動態鏈接器與普通共用對象一樣被映射到了進程的地址空間,在系統開始運行Program1之前,首先會把控制權交給動態鏈接器,由它完成所有的動態鏈接工作以後再把控制權交給 Program1,然後開始執行。

我們通過 readelf工具來查看 Lib. so的裝載屬性,就如我們在前面查看普通程式一樣:

除了文件的類型與普通程式不同以外,其他幾乎與普通程式一樣。還有有一點比較不同的是,動態鏈接模塊的裝載地址是從地址0x00000始的。我們知道這個地址是無效地址,並且從上面的進程虛擬空間分佈看到, Lib. so的最終裝載地址並不是0x000000是0xb7efc000。從這點我們可以推斷,共用對象的最終裝載地址在編譯時是不確定的,而是在裝載時,裝載器根據當前地址空間的空閑情況,動態分配一塊足夠大小的虛擬地址空間給相應的共用對象。

當然,這僅僅是一個推斷,至於為什麼要這樣做,為什麼不將每個共用對象在進程中的地址固定,或者在真正的系統中是怎麼運作的,我們將在下一節進行解釋。


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

-Advertisement-
Play Games
更多相關文章
  • Java當中的類集框架 類集框架 ,那麼什麼是類集框架,集合的種類有哪些,類集框架的基礎結構。 類集框架是一組類和介面的集合,位於 包當中,是用來用戶存儲和管理對象的,在這個類集合框架中,我們主要學習的為三大類, 分別是集合,列表和映射。 集合,列表,映射 Set為集合 ,在集合中的對象是不按照順序 ...
  • 1、什麼是RabbitMQ MQ(Message Queue):消息隊列,是服務端設計的一個可以存儲大量消息的隊列,並提供客戶端操作隊列的方法:生產隊列(向隊列中添加數據)、消費隊列(從隊列中取數據)。RabbitMQ就是基於消息隊列的一個典型應用。RabbitMQ除了普通的生產消費功能,還有一些高 ...
  • 本文首發於我的個人博客: "尾尾部落" 1. Iterator介面 Iterator介面,這是一個用於遍歷集合中元素的介面,主要包含hashNext(),next(),remove()三種方法。它的一個子介面LinkedIterator在它的基礎上又添加了三種方法,分別是add(),previous ...
  • Web框架本質 我們可以這樣理解:所有的Web應用本質上就是一個socket服務端,而用戶的瀏覽器就是一個socket客戶端。 這樣我們就可以自己實現Web框架了。 半成品自定義web框架 可以說Web服務本質上都是在這十幾行代碼基礎上擴展出來的。這段代碼就是它們的祖宗。 用戶的瀏覽器一輸入網址,會 ...
  • 好久沒寫博客了,小伙伴們最近在幹嘛呢? 最近在搞AI開放平臺,就類似騰訊優圖,百度人工智慧平臺~~. 說得是很高大上啦,核心技術的演算法並不是我寫的。我負責搞API介面,寫前端。 前端的Vue和Bootstrap,兩門技術是目前前端比較流利的。最近也在搞Vue,感覺挺吃力的~_~ 正文 pycharn ...
  • 在寫爬蟲的時候總是遇到一些以圖片的形式展示的信息,因此要怎麼解析圖片上的信息呢?在Google上查了一下,需要安裝pytesseract和pillow(我用的python3.7)和Tesseract-OCR 1. 安裝pytesseract pip insatll pytesseract 2. 安裝 ...
  • 以下代碼是單片機程式,51單片機,編譯器為HT-IDE3000, 簡單來說 頭文件中只能申明, 變數在頭文件中申明時,要加上extern 這個關鍵字用來告訴編譯器,變數在其它的文件中定義,為什麼要在頭文件中申明變數? >因為想在其它文件里的代碼中使用這些變數, 如在a.c中使用b.c里定義的變數, ...
  • 一.函數的定義 return語句不寫或後邊不加任何對象即為return None 二.函數的參數 無參數 一個參數 多個參數 必須參數 必須按照正確順序和數量傳入參數 關鍵字參數 預設參數 預設參數必須放在必須參數的後面 不定長參數 參數帶一個星號* 參數帶兩個星號** 定義函數的參數時請以必須參數 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...