如何調(diào)用C語言寫的庫,如a.lib等,有對(duì)應(yīng)的庫頭文件a.h。假設(shè)a.h中定義了函數(shù):
成都創(chuàng)新互聯(lián)公司成立于2013年,先為牟定等服務(wù)建站,牟定等地企業(yè),進(jìn)行企業(yè)商務(wù)咨詢服務(wù)。為牟定企業(yè)網(wǎng)站制作PC+手機(jī)+微官網(wǎng)三網(wǎng)同步一站式服務(wù)解決您的所有建站問題。
int
WhyCoding(int
a,
float
b);
做法是,
/*
cpp_a.h
*/
extern
"C"
{
#include
"a.h"
}
或
/*
cpp_a.h
*/
extern
"C"
{
int
WhyCoding(int
a,
float
b);
/*
重定義所有的C函數(shù)
*/
}
從上面可以看出,extern
"C"
是用在C和C++之間的橋梁。之所以需要這個(gè)橋梁是因?yàn)镃編譯器編譯函數(shù)時(shí)不帶
函數(shù)的類型信息,只包含函數(shù)符號(hào)名字,如C編譯器把函數(shù)int
a(float
x)編譯成類似_a這樣的符號(hào),C連接器只要
找到了調(diào)用函數(shù)的符號(hào),就可以連接成功,它假設(shè)參數(shù)類型信息是正確的,這是C編譯連接器的缺點(diǎn)。而C++
編譯器為了實(shí)現(xiàn)函數(shù)重載,編譯時(shí)會(huì)帶上函數(shù)的類型信息,如他把上面的a函數(shù)可能編譯成_a_float這樣的
符號(hào)為了實(shí)現(xiàn)重載,注意它還是沒有帶返回值得信息,這也是為什么C++不支持采用函數(shù)返回值來區(qū)別函數(shù)
重載的原因之一,當(dāng)然,函數(shù)的使用者對(duì)函數(shù)返回值的處理方式(如忽略)也是重要原因。
基于以上,C調(diào)用C++,首先需要用封裝函數(shù)把對(duì)C++的類等的調(diào)用封裝成C函數(shù)以便C調(diào)用,于是extern
"C"
的
作用是:讓編譯器知道這件事,然后以C語言的方式編譯和連接封裝函數(shù).(通常是把封裝函數(shù)用C++編譯器按C++
方式編譯,用了extern
"C"
后,編譯器便依C的方式編譯封裝接口,當(dāng)然接口函數(shù)里面的C++語法還是按C++方式
編譯;對(duì)于C語言部分--調(diào)用者,還是按C語言編譯;分別對(duì)C++接口部分和C部分編譯后,再連接就可以實(shí)現(xiàn)C
調(diào)用C++了).
相反,C++調(diào)用C函數(shù),extern
"C"
的作用是:讓C++連接器找調(diào)用函數(shù)的符號(hào)時(shí)采用C的方式,即使用_a而不是
_a_float來找調(diào)用函數(shù)。
C語言中,函數(shù)調(diào)用的一般形式為:
函數(shù)名(實(shí)際參數(shù)表)
對(duì)無參函數(shù)調(diào)用時(shí)則無實(shí)際參數(shù)表。實(shí)際參數(shù)表中的參數(shù)可以是常數(shù)、變量或其它構(gòu)造類型數(shù)據(jù)及表達(dá)式。各實(shí)參之間用逗號(hào)分隔。
#includestdio.h
int?fun(int?x,?int?y);?//?函數(shù)聲明,如果函數(shù)寫在被調(diào)用處之前,可以不用聲明
void?main()
{
int?a=1,?b=2,?c;
c?=?fun(a,?b);?//?函數(shù)的調(diào)用,調(diào)用自定義函數(shù)fun,其中a,b為實(shí)際參數(shù),傳遞給被調(diào)用函數(shù)的輸入值
}
//?自定義函數(shù)fun
int?fun(int?x,?int?y)??//?函數(shù)首部
{??//?{}中的語言為函數(shù)體
return?xy???x?:?y;??//?返回x和y中較大的一個(gè)數(shù)
}
擴(kuò)展資料
C語言中不允許作嵌套的函數(shù)定義。因此各函數(shù)之間是平行的,不存在上一級(jí)函數(shù)和下一級(jí)函數(shù)的問題。但是C語言允許在一個(gè)函數(shù)的定義中出現(xiàn)對(duì)另一個(gè)函數(shù)的調(diào)用。
這樣就出現(xiàn)了函數(shù)的嵌套調(diào)用。即在被調(diào)函數(shù)中又調(diào)用其它函數(shù)。這與其它語言的子程序嵌套的情形是類似的。其關(guān)系可表示如圖。
圖表示了兩層嵌套的情形。其執(zhí)行過程是:執(zhí)行main函數(shù)中調(diào)用a函數(shù)的語句時(shí),即轉(zhuǎn)去執(zhí)行a函數(shù),在a函數(shù)中調(diào)用b 函數(shù)時(shí),又轉(zhuǎn)去執(zhí)行b函數(shù),b函數(shù)執(zhí)行完畢返回a函數(shù)的斷點(diǎn)繼續(xù)執(zhí)行,a函數(shù)執(zhí)行完畢返回main函數(shù)的斷點(diǎn)繼續(xù)執(zhí)行。
參考資料:函數(shù)調(diào)用_百度百科
#define _CRT_SECURE_NO_WARNINGS
#include stdio.h
//聲明一個(gè)求和的函數(shù)
int sum(int a, int b);
int main(int argc, char* argv[])
{
int x = 5, y = 7;
int s = sum(x, y); //調(diào)用函數(shù)
printf("x + y = %d \n", s);
system("pause");
return 0;
}
//定義求和函數(shù)
int sum(int a, int b)
{
return a + b;
}
更多C語言教程
函數(shù)分為庫函數(shù)和自定義函數(shù),在調(diào)用庫函數(shù)時(shí)只要標(biāo)明頭文件(所謂頭文件就是一個(gè)東西,里面包括了一些函數(shù)各聲明之類的,當(dāng)你要調(diào)用它中的函數(shù)時(shí)就就得先向程序說明你要調(diào)用這個(gè)文件里的函數(shù),否則有有侵權(quán)行為哦)可以在主調(diào)函數(shù)中調(diào)用這個(gè)頭文件里的所有函數(shù)了,例如頭文件為#includestdio.h你就可以在主調(diào)函數(shù)中調(diào)用這個(gè)頭文件里的函數(shù):
main
{ printf("haha");/*這時(shí)main函數(shù)為主調(diào)函數(shù),printf為被調(diào)函數(shù)*/
}
調(diào)用自定義函數(shù)時(shí)其實(shí)和調(diào)用庫函數(shù)一樣的,只不過這個(gè)函數(shù)是你自己定義的,再如:
main
{ void f();/*聲明 f()函數(shù)*/
f(); /*調(diào)用f()函數(shù)*/
}
f()
{
printf("haha");/*f()函數(shù)調(diào)用庫函數(shù)*/
}
工具/材料
Ubuntu16.04
gcc+vim
01
打開Ubuntu,并在目標(biāo)路徑下開啟一個(gè)終端。
02
選定一個(gè)路徑,使用touch命令創(chuàng)建三個(gè)文件,function.h,function.c,test.c,分別用來做函數(shù)頭文件、函數(shù)源文件以及測(cè)試文件
03
這里以四則運(yùn)算函數(shù)為例說明函數(shù)的編寫流程與調(diào)用流程。首先用vim命令與vsplit依次打開三個(gè)文件。
04
在頭文件中添加重復(fù)包含的宏,并添加四則運(yùn)算的函數(shù)聲明。
05
在function.c中將function.h包含進(jìn)來,并具體實(shí)現(xiàn)四個(gè)方法。注意出發(fā)要對(duì)除數(shù)是否為0進(jìn)行判斷。
06
然后編寫測(cè)試程序進(jìn)行測(cè)試??偟某绦虼a如下:
07
保存所有的程序并退出,使用gcc進(jìn)行編譯,并運(yùn)行得到的結(jié)果如下。這就是函數(shù)的聲明、定義以及調(diào)用。
每一個(gè)函數(shù)調(diào)用都會(huì)分配函數(shù)棧,在棧內(nèi)進(jìn)行函數(shù)執(zhí)行過程。調(diào)用前,先把返回地址壓棧,然后把當(dāng)前函數(shù)的esp指針壓棧。