成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

應用程序設計:在動態庫中如何調用外部函數?

開發 前端
不論是在 Windows 系統中,還是在 Unix 系列平臺上,到處都能見到我的身影,因為我能為大家節省很多資源啊,資源就是人民幣!

[[400812]]

大家好,我是一個動態鏈接庫!

這個名字,相信你一定早就如雷貫耳了。

[[400813]]

在計算機早期時代,由于內存資源緊張,我可是發揮了重大的作用!

不論是在 Windows 系統中,還是在 Unix 系列平臺上,到處都能見到我的身影,因為我能為大家節省很多資源啊,資源就是人民幣!

愉快的玩耍

比如:我的主人編寫了這么一段簡單的代碼:

  1. # 文件:lib.c 
  2.  
  3. #include <stdio.h> 
  4.  
  5. int func_in_lib(int k) 
  6.     printf("func_in_lib is called \n"); 
  7.     return k + 1; 

只要用如下命令來編譯,我就誕生出來了 lib.so,也就是一個動態鏈接庫:

  1. $ gcc -m32 -fPIC --shared -o lib.so lib.c 

這個時候,主人隨便把我丟給誰,我都可以為他服務,只要他調用我肚子里的這個函數 func_in_lib 就可以了。

雖然目前你看到我提供的這個函數很簡單,但是道理都是一樣的,后面如果有機會,我就在這個函數里來計算機器人的運動軌跡,給你瞧一瞧!

[[400814]]

例如:張三今天寫了一段代碼,需要調用我的這個函數。

張三這個人比較喜歡騷操作,明明他在編譯可執行程序的時候,把我動態鏈接一下就可以了,就像下面這樣:

  1. $ gcc -m32 -o main main.c ./lib.so 

但是張三偏偏不這么做,為了炫技,他選擇使用 dlopen 動態加載的方式,來把我從硬盤上加載到進程中。

咱們來一起圍觀一下張三寫的可執行程序代碼:

[[400815]]

  1. # 文件:main.c 
  2.  
  3. #include <unistd.h> 
  4. #include <stdio.h> 
  5. #include <stdlib.h> 
  6. #include <dlfcn.h> 
  7.  
  8. typedef int (*pfunc)(int); 
  9.  
  10. int main(int argc, char *agv[]) 
  11.     int a = 1; 
  12.     int b; 
  13.  
  14.     // 打開動態庫 
  15.     void *handle = dlopen("./lib.so", RTLD_NOW); 
  16.     if (handle) 
  17.     { 
  18.         // 查找動態庫中的函數 
  19.         pfunc func = (pfunc) dlsym(handle, "func_in_lib"); 
  20.         if (func) 
  21.         { 
  22.             b = func(a); 
  23.             printf("b = %d \n", b); 
  24.         } 
  25.         else 
  26.         { 
  27.             printf("dlsym failed! \n"); 
  28.         } 
  29.         dlclose(handle); 
  30.     } 
  31.     else 
  32.     { 
  33.         printf("dlopen failed! \n"); 
  34.     } 
  35.      
  36.     return 0; 

從代碼中可以看到,張三預先知道我肚子里的這個函數名稱是 func_in_lib,所以他使用了系統函數 dlsym(handle, "func_in_lib"); 來找到這個函數在內存中的加載地址,然后就可以直接調用這個函數了。

張三編譯得到可執行文件 main 之后,執行結果完全正確,很開心!

[[400816]]

悲從中來

可是有一天,我遇到一件煩人的事情,我的主人說:你這個服務函數的計算過程太單調了,給你找點樂子,你在執行的時候啊,到其他一個外部模塊里調用一個函數。

話剛說完,就丟給我一個函數名:void func_in_main(void);。

[[400817]]

也就是說,我需要在我的服務函數中,去調用其他模塊里的函數,就像下面這樣:

  1. #include <stdio.h> 
  2.  
  3. // 外部函數聲明 
  4. void func_in_main(void); 
  5.  
  6. int func_in_lib(int k) 
  7.     printf("func_in_lib is called \n"); 
  8.  
  9.     // 調用外部函數 
  10.     func_in_main(); 
  11.      
  12.     return k + 1; 

那么這個函數在哪里呢?天哪,我怎么知道這個函數是什么鬼?怎么才能找到它藏在內存的那個角落(地址)里?

不管怎么樣,主人修改了代碼之后,還是很順利的把我編譯了出來:

  1. $ gcc -m32 -fPIC --shared -o lib.so lib.c 

編譯指令完全沒有變化。

因為我僅僅是一個動態鏈接庫,這個時候即使我不知道 func_in_main 函數的地址,也是可以編譯成功的。

只不過我要把這個家伙標記一下:誰要是想使用我,就必須告訴我這個家伙的地址在哪里!,否則就別怪我耍賴。

無辜的張三

我的主人對張三說:兄弟,我的這個動態鏈接庫升級了,功能更強大哦,想不想試一下?

張三心想:我是使用 dlopen 的方式來動態加載動態庫文件的,不需要對可執行程序重新編譯或者鏈接,直接運行就完事了!

于是他二話不說,直接就把我拿過去,丟在他的可執行程序目錄下,然后執行 main 程序。

可是這一次,他看到的結果卻是:

  1. dlopen failed! 

為什么會加載失敗呢?上次明明是正常執行的!張三一臉懵!

[[400818]]

其實,這壓根就不能怪我!以為我剛才就說了:誰要是想使用我,就必須告訴我 func_in_main 這個函數的地址在哪里!

可是在張三的這個進程里,我到處都找不到這個函數的地址。既然你沒法滿足我,那我就沒法滿足你!

錦囊1: 導出符號表

張三這下也沒轍了,只要找我的主人算賬:我的應用程序代碼一絲一毫都沒有動,怎么換了你給的新動態鏈接庫就不行了呢?

主人慢條斯理的回答:疏忽了,疏忽了,忘記跟你說一件事情了:這個動態庫啊,它需要你多做一件事情:在你的程序中提供一個名為 func_in_main 的函數,這樣就可以了。

[[400819]]

張三一想:這個好辦,加一個函數就是了。

因為這個可執行程序只有一個 main.c 文件,于是他在其中新加了一個函數:

  1. void func_in_main(void) 
  2.     printf("func_in_main \n"); 

然后就開始編譯、執行,一頓操作猛如虎:

  1. # gcc -m32 -o main main.c -ldl 
  2. # ./main 
  3. dlopen failed! 

咦?怎么還是失敗?!已經按照要求加了 func_in_main 這個函數了啊?!

[[400820]]

這個傻X張三,對,你確實是在 main.c 中加了這個函數,但是你僅僅是加在你的可執行程序中的,但是我卻壓根就看不到這個函數啊!

不信的話,你檢查一下編譯出來的可執行程序中,是否把 func_in_main 這個符號導出來了?如果不導出來,我怎么能看到?

  1. # 查看導出的符號表 
  2. $ objdump -e main -T | grep func_in_main 
  3. # 這里輸出為空 

既然輸出為空,就說明沒有導出來!這個就不用我教你了吧?

茴香豆的“茴”字,一共有四種寫法。。。

[[400821]]

哦,不,導出符號,一共有兩種方式:

方式1:導出所有的符號

  1. $ gcc -m32 -rdynamic -o main main.c -ldl 

當然,下面這個指令也可以:

  1. gcc -m32 -Wl,--export-dynamic -o main main.c -ldl 

方式2:導出指定的符號

先定義一個文件,把需要導出的符號全部羅列出來:

文件:exported.txt

  1.     extern "C" 
  2.     { 
  3.         func_in_main; 
  4.     }; 
  5. }; 

然后,在編譯選項中指定這個導出文件:

  1. gcc -m32 -Wl,-dynamic-list=./exported.txt -o main main.c -ldl 

使用以上兩種方式的任意一種即可,編譯之后,再使用 objdump 指令看一下導出符號:

  1. $ objdump -e main -T | grep func_in_main 
  2. 080485bb g    DF .text  00000019  Base        func_in_main 

嗯,很好很好!張三趕緊按照這樣的方式操作了一下,果真成功執行了函數!

  1. $ ./main  
  2. func_in_lib is called  
  3. func_in_main  
  4. b = 2 

也就是說,在我的動態庫文件中,正確的找到了外部其他模塊中的函數地址,并且愉快的執行成功了!

[[400822]]

錦囊2: 動態注冊

雖然執行成功了,張三的心里隱隱約約的仍然有一絲不爽的感覺,每次編譯都要導出符號,真麻煩,能不能優化一下?

于是他找到我的主人,表達了自己的不滿。

主人一瞧,有個性!既然你不想提供,那我就滿足你:

  1. 首先,在動態庫中提供一個默認的函數實現(func_in_main_def);
  2. 然后,再提供一個專門的注冊函數(register_func),如果外部模塊想提供 func_in_main 這個函數,就調用注冊函數注冊進來;

此時,lib.c 最新的代碼就變成這個樣子了:

  1. #include <stdio.h> 
  2.  
  3. // 默認試下 
  4. void func_in_main_def(void) 
  5.     printf("the main is lazy, do NOT register me! \n"); 
  6.  
  7. // 定義外部函數指針 
  8. void (*func_in_main)() = func_in_main_def; 
  9.  
  10. void register_func(void (*pf)()) 
  11.     func_in_main = pf; 
  12.  
  13. int func_in_lib(int k) 
  14.     printf("func_in_lib is called \n"); 
  15.  
  16.     if (func_in_main) 
  17.         func_in_main(); 
  18.  
  19.     return k + 1; 

然后編譯,全新的我再一次誕生了 lib.so:

  1. gcc -m32 -fPIC --shared -o lib.so lib.c 

主人把我丟給張三的時候說:好了,滿足你的需求,這一次你不用提供 func_in_main 這個函數了,當然也就不用再導出符號了。

不過,如果如果有一天,你改變了注意,又想提供這個函數了,那么你就要通過動態庫中的 register_func 函數,把你的函數注冊進來。

Have you got it?趕緊再去試一下!

[[400823]]

這個時候,張三再次使用我的時候,就不需要導出他的 main.c 里的那個函數 func_in_main了,實際上他可以把這個函數從代碼中刪掉!

編譯、執行,張三再一次猛如虎的操作:

  1. $ gcc -m32 -o main main.c -ldl 
  2. $ ./main 
  3. func_in_lib is called  
  4. the main is lazy, do NOT register me!  
  5. b = 2 

嗯,結果看起來是正確的。

咦?怎么多了一行字:the main is lazy, do NOT register me!

[[400824]]

難道是在質疑我的技術能力嗎?好吧,既然如此,我也滿足你,不就是注冊一個函數嘛,簡單:

  1. // 文件: main.c 
  2.  
  3. #include <unistd.h> 
  4. #include <stdio.h> 
  5. #include <stdlib.h> 
  6. #include <dlfcn.h> 
  7.  
  8. typedef int (*pfunc)(int); 
  9. typedef int (*pregister)(void (*)()); 
  10.  
  11. // 控制注冊函數的宏定義 
  12. #define REG_FUNC 
  13.  
  14. #ifdef REG_FUNC 
  15. void func_in_main(void) 
  16.     printf("func_in_main \n"); 
  17. #endif 
  18.  
  19. int main(int argc, char *agv[]) 
  20.     int a = 1; 
  21.     int b; 
  22.  
  23.     // 打開動態庫 
  24.     void *handle = dlopen("./lib.so", RTLD_NOW); 
  25.     if (handle) 
  26.     { 
  27. #ifdef REG_FUNC 
  28.         // 查找動態庫中的注冊函數 
  29.         pregister register_func = (pregister) dlsym(handle, "register_func"); 
  30.         if (register_func) 
  31.         { 
  32.  
  33.             register_func(func_in_main); 
  34.         } 
  35. #endif 
  36.  
  37.         // 查找動態庫中的函數 
  38.         pfunc func = (pfunc) dlsym(handle, "func_in_lib"); 
  39.         if (func) 
  40.         { 
  41.             b = func(a); 
  42.             printf("b = %d \n", b); 
  43.         } 
  44.         else 
  45.         { 
  46.             printf("dlsym failed! \n"); 
  47.         } 
  48.         dlclose(handle); 
  49.     } 
  50.     else 
  51.     { 
  52.         printf("dlopen failed! \n"); 
  53.     } 
  54.      
  55.     return 0; 

然后編譯、執行:

  1. $ gcc -m32 -o main main.c -ldl 
  2. $ ./main  
  3. func_in_lib is called  
  4. func_in_main  
  5. b = 2 

[[400825]]

完美收官!

PS:很多平臺級的代碼,例如一些工控領域的運行時(Runtime)軟件,大部分都是通過注冊的方式,來把平臺代碼、用戶代碼進行連接、綁定的。

 

責任編輯:姜華 來源: IOT物聯網小鎮
相關推薦

2012-03-30 15:47:50

ibmdw

2010-03-04 10:11:17

Android手機系統

2022-05-04 23:08:36

標準Go應用程序

2012-02-15 14:39:55

GNOME 3

2017-10-27 13:30:59

大數據MongoDBeBay

2010-08-04 09:34:51

Flex設計

2020-12-28 14:40:47

云計算云應用SaaS

2010-06-12 16:41:10

BlackBerry開

2010-08-12 15:59:23

Flex應用程序

2009-09-03 08:46:55

UML類圖Java

2011-05-18 10:42:48

2009-02-25 14:51:05

應用程序設計ASP.NET.NET

2018-01-24 20:42:06

數據庫NoSQL驅動力

2023-12-29 22:39:25

Golang應用程序數據庫

2012-04-16 13:47:37

JavaMatlab

2009-07-17 10:42:06

Swing應用程序處理函數

2009-06-18 15:41:36

動態分配CPUJava

2021-05-06 05:37:40

JavascriptSTT機器學習

2010-07-13 10:33:49

Perl用戶函數

2012-06-14 09:32:13

微軟Windows 8
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久av一区二区三区 | 欧美精品福利视频 | 极品国产视频 | 亚洲一区二区在线视频 | 欧美中国少妇xxx性高请视频 | 亚洲成人高清 | 日本视频中文字幕 | 精品国产一区二区国模嫣然 | 成人免费视频网站在线看 | 久草热8精品视频在线观看 午夜伦4480yy私人影院 | 午夜精品视频 | 一区二区三区中文字幕 | 精品三级在线观看 | 国产精品美女视频 | 综合二区 | 色视频欧美 | 久久久久久久国产精品影院 | 999久久精品| 给我免费的视频在线观看 | 欧美在线观看一区 | 狠狠操在线 | 看av片网站 | 亚洲欧美日韩电影 | 国产高清免费 | 国产不卡一区 | 97国产精品视频人人做人人爱 | 91干b| 91偷拍精品一区二区三区 | 欧美一区二区在线观看视频 | 精品国产乱码久久久久久蜜柚 | 色在线免费视频 | 亚洲三区在线观看 | 午夜免费在线电影 | 91在线中文字幕 | 红色av社区 | 一区二区三区亚洲精品国 | 涩涩视频网站在线观看 | 国产成人综合亚洲欧美94在线 | 在线观看视频你懂得 | 国产高清精品一区二区三区 | 小川阿佐美pgd-606在线 |