如何在在動(dòng)態(tài)鏈接庫(kù)中調(diào)用外部函數(shù)?
大家好,我是一個(gè)動(dòng)態(tài)鏈接庫(kù)!
這個(gè)名字,相信你一定早就如雷貫耳了。
在計(jì)算機(jī)早期時(shí)代,由于內(nèi)存資源緊張,我可是發(fā)揮了重大的作用!
不論是在 Windows 系統(tǒng)中,還是在 Unix 系列平臺(tái)上,到處都能見(jiàn)到我的身影,因?yàn)槲夷転榇蠹夜?jié)省很多資源啊,資源就是人民幣!
愉快的玩耍
比如:我的主人編寫(xiě)了這么一段簡(jiǎn)單的代碼:
# 文件:lib.c
#include <stdio.h>
int func_in_lib(int k)
{
printf("func_in_lib is called ");
return k + 1;
}
只要用如下命令來(lái)編譯,我就誕生出來(lái)了 lib.so,也就是一個(gè)動(dòng)態(tài)鏈接庫(kù):
$ gcc -m32 -fPIC --shared -o lib.so lib.c
這個(gè)時(shí)候,主人隨便把我丟給誰(shuí),我都可以為他服務(wù),只要他調(diào)用我肚子里的這個(gè)函數(shù) func_in_lib 就可以了。
雖然目前你看到我提供的這個(gè)函數(shù)很簡(jiǎn)單,但是道理都是一樣的,后面如果有機(jī)會(huì),我就在這個(gè)函數(shù)里來(lái)計(jì)算機(jī)器人的運(yùn)動(dòng)軌跡,給你瞧一瞧!
例如:張三今天寫(xiě)了一段代碼,需要調(diào)用我的這個(gè)函數(shù)。
張三這個(gè)人比較喜歡騷操作,明明他在編譯可執(zhí)行程序的時(shí)候,把我動(dòng)態(tài)鏈接一下就可以了,就像下面這樣:
$ gcc -m32 -o main main.c ./lib.so
但是張三偏偏不這么做,為了炫技,他選擇使用 dlopen 動(dòng)態(tài)加載的方式,來(lái)把我從硬盤上加載到進(jìn)程中。
咱們來(lái)一起圍觀一下張三寫(xiě)的可執(zhí)行程序代碼:
# 文件:main.c
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <dlfcn.h>
typedef int (*pfunc)(int);
int main(int argc, char *agv[])
{
int a = 1;
int b;
// 打開(kāi)動(dòng)態(tài)庫(kù)
void *handle = dlopen("./lib.so", RTLD_NOW);
if (handle)
{
// 查找動(dòng)態(tài)庫(kù)中的函數(shù)
pfunc func = (pfunc) dlsym(handle, "func_in_lib");
if (func)
{
b = func(a);
printf("b = %d ", b);
}
else
{
printf("dlsym failed! ");
}
dlclose(handle);
}
else
{
printf("dlopen failed! ");
}
return 0;
}
從代碼中可以看到,張三預(yù)先知道我肚子里的這個(gè)函數(shù)名稱是 func_in_lib,所以他使用了系統(tǒng)函數(shù) dlsym(handle, "func_in_lib"); 來(lái)找到這個(gè)函數(shù)在內(nèi)存中的加載地址,然后就可以直接調(diào)用這個(gè)函數(shù)了。
張三編譯得到可執(zhí)行文件 main 之后,執(zhí)行結(jié)果完全正確,很開(kāi)心!
悲從中來(lái)
可是有一天,我遇到一件煩人的事情,我的主人說(shuō):你這個(gè)服務(wù)函數(shù)的計(jì)算過(guò)程太單調(diào)了,給你找點(diǎn)樂(lè)子,你在執(zhí)行的時(shí)候啊,到其他一個(gè)外部模塊里調(diào)用一個(gè)函數(shù)。
話剛說(shuō)完,就丟給我一個(gè)函數(shù)名:void func_in_main(void);。
也就是說(shuō),我需要在我的服務(wù)函數(shù)中,去調(diào)用其他模塊里的函數(shù),就像下面這樣:
#include <stdio.h>
// 外部函數(shù)聲明
void func_in_main(void);
int func_in_lib(int k)
{
printf("func_in_lib is called ");
// 調(diào)用外部函數(shù)
func_in_main();
return k + 1;
}
那么這個(gè)函數(shù)在哪里呢?天哪,我怎么知道這個(gè)函數(shù)是什么鬼?怎么才能找到它藏在內(nèi)存的那個(gè)角落(地址)里?
不管怎么樣,主人修改了代碼之后,還是很順利的把我編譯了出來(lái):
$ gcc -m32 -fPIC --shared -o lib.so lib.c
編譯指令完全沒(méi)有變化。
因?yàn)槲覂H僅是一個(gè)動(dòng)態(tài)鏈接庫(kù),這個(gè)時(shí)候即使我不知道 func_in_main 函數(shù)的地址,也是可以編譯成功的。
只不過(guò)我要把這個(gè)家伙標(biāo)記一下:誰(shuí)要是想使用我,就必須告訴我這個(gè)家伙的地址在哪里!,否則就別怪我耍賴。

發(fā)表評(píng)論
請(qǐng)輸入評(píng)論內(nèi)容...
請(qǐng)輸入評(píng)論/評(píng)論長(zhǎng)度6~500個(gè)字
圖片新聞
-
機(jī)器人奧運(yùn)會(huì)戰(zhàn)報(bào):宇樹(shù)機(jī)器人摘下首金,天工Ultra搶走首位“百米飛人”
-
存儲(chǔ)圈掐架!江波龍起訴佰維,索賠121萬(wàn)
-
長(zhǎng)安汽車母公司突然更名:從“中國(guó)長(zhǎng)安”到“辰致科技”
-
豆包前負(fù)責(zé)人喬木出軌BP后續(xù):均被辭退
-
字節(jié)AI Lab負(fù)責(zé)人李航卸任后返聘,Seed進(jìn)入調(diào)整期
-
員工持股爆雷?廣汽埃安緊急回應(yīng)
-
中國(guó)“智造”背后的「關(guān)鍵力量」
-
小米汽車研發(fā)中心重磅落地,寶馬家門口“搶人”
最新活動(dòng)更多
-
即日-9.1立即下載>> 【限時(shí)下載】ADI中國(guó)三十周年感恩回饋助力企業(yè)升級(jí)!
-
即日-9.16點(diǎn)擊進(jìn)入 >> 【限時(shí)福利】TE 2025國(guó)際物聯(lián)網(wǎng)展·深圳站
-
10月23日立即報(bào)名>> Works With 開(kāi)發(fā)者大會(huì)深圳站
-
10月24日立即參評(píng)>> 【評(píng)選】維科杯·OFweek 2025(第十屆)物聯(lián)網(wǎng)行業(yè)年度評(píng)選
-
11月27日立即報(bào)名>> 【工程師系列】汽車電子技術(shù)在線大會(huì)
-
12月18日立即報(bào)名>> 【線下會(huì)議】OFweek 2025(第十屆)物聯(lián)網(wǎng)產(chǎn)業(yè)大會(huì)
推薦專題
- 1 阿里首位程序員,“掃地僧”多隆已離職
- 2 先進(jìn)算力新選擇 | 2025華為算力場(chǎng)景發(fā)布會(huì)暨北京xPN伙伴大會(huì)成功舉辦
- 3 宇樹(shù)機(jī)器人撞人事件的深度剖析:六維力傳感器如何成為人機(jī)安全的關(guān)鍵屏障
- 4 清華跑出具身智能獨(dú)角獸:給機(jī)器人安上眼睛和大腦,融資近20億
- 5 特朗普要求英特爾首位華人 CEO 辭職
- 6 踢館大廠和微軟,剖析WPS靈犀的AI實(shí)用主義
- 7 騰訊 Q2 財(cái)報(bào)亮眼:AI 已成第二增長(zhǎng)曲線
- 8 谷歌吹響AI沖鋒號(hào),AI還有哪些機(jī)會(huì)
- 9 蘋(píng)果把身家押在Siri上:一場(chǎng)輸不起的自我革命
- 10 共探合作新機(jī)遇!江門市新會(huì)區(qū)(深圳)“AI + 機(jī)器人” 產(chǎn)業(yè)對(duì)接會(huì)成功舉辦