新聞中心
Linux 是一個(gè)開放源代碼操作系統(tǒng),作為一名開發(fā)者,學(xué)會(huì)開發(fā) Linux 系統(tǒng)是極為重要的。在 Linux 開發(fā)中,函數(shù)調(diào)用是最基礎(chǔ)、最重要的一環(huán)。通過函數(shù)調(diào)用,我們可以實(shí)現(xiàn)各種各樣的功能,從而完善 Linux 系統(tǒng)。本文將為大家詳解 Linux 函數(shù)調(diào)用,包括調(diào)用過程、參數(shù)傳遞、實(shí)例演示等內(nèi)容。

一、函數(shù)調(diào)用過程
在 Linux 開發(fā)中,函數(shù)調(diào)用遵循一定的規(guī)則,大體過程如下:
1. 將調(diào)用參數(shù)傳遞給被調(diào)用函數(shù)。傳遞參數(shù)時(shí),如果參數(shù)是基本類型,則直接傳遞值;但如果是數(shù)組或結(jié)構(gòu)體等復(fù)雜類型,需要傳遞地址。
2. 被調(diào)用函數(shù)使用傳遞的參數(shù)來執(zhí)行相應(yīng)的任務(wù)。
3. 可能會(huì)返回一個(gè)值給調(diào)用函數(shù)。
4. 調(diào)用函數(shù)使用返回值來完成自己的任務(wù)。
5. 調(diào)用函數(shù)通過堆棧(stack)來實(shí)現(xiàn)上述過程。在調(diào)用函數(shù)前,先將返回地址值(即調(diào)用函數(shù)時(shí)下一條指令的地址)壓入堆棧;在函數(shù)執(zhí)行完后,返回值從特定寄存器或內(nèi)存中取出,并跳轉(zhuǎn)到返回地址。
二、參數(shù)傳遞細(xì)節(jié)
1. 基本類型的參數(shù)傳遞:整形參數(shù)會(huì)被直接復(fù)制到被調(diào)用函數(shù)的棧幀中,而不是函數(shù)的參數(shù)列表中。同時(shí),一個(gè)值的傳遞不會(huì)影響原來變量的值。例如:
“`
int a = 100;
int b = 200;
swap(a, b);// 不會(huì)改變a和b的值
“`
2. 數(shù)組和結(jié)構(gòu)體的參數(shù)傳遞:與傳遞整型變量不同,數(shù)組和結(jié)構(gòu)體類型的參數(shù)在傳遞時(shí),實(shí)際上傳遞的是地址。例如:
“`
typedef struct{
int a;
int b;
}Point;
void setPoint(Point* p, int a, int b){
p->a = a;
p->b = b;
}
“`
在調(diào)用 setPoint 函數(shù)時(shí),可以使用如下方式:
“`
Point p;
setPoint(&p, 1, 2);
“`
3. 可變參數(shù)函數(shù)(如 printf 函數(shù))的參數(shù)傳遞:可變參數(shù)函數(shù)處理時(shí),需要使用不定量參數(shù)列表(va_list)和相關(guān)宏或函數(shù)來獲得參數(shù)的值。例如:
“`
#include
int sum(int num, …){
va_list valist;
int sum = 0;
va_start(valist, num);
for(int i = 0; i
sum += va_arg(valist, int);
}
va_end(valist);
return sum;
}
“`
三、實(shí)例演示
1. 獲取用戶輸入并輸出
“`
#include
int mn(){
char* buffer;
printf(“Input something: “);
buffer = get_input();
printf(“Your input: %s”, buffer);
return 0;
}
char* get_input(){
char* buffer = (char*)malloc(100);
scanf(“%s”, buffer);
return buffer;
}
“`
2. 簡(jiǎn)單排序
“`
#include
void bubble_sort(int arr[], int len);
void print_arr(int arr[], int len);
int mn(){
int len = 5;
int arr[] = {5, 4, 3, 2, 1};
printf(“before sort: “);
print_arr(arr, len);
bubble_sort(arr, len);
printf(“\nafter sort: “);
print_arr(arr, len);
return 0;
}
void bubble_sort(int arr[], int len){
int temp;
for(int i = 0; i
for(int j = 0; j
if(arr[j] > arr[j + 1]){
temp = arr[j];
arr[j] = arr[j + 1];
arr[j + 1] = temp;
}
}
}
}
void print_arr(int arr[], int len){
for(int i = 0; i
printf(“%d “, arr[i]);
}
}
“`
相關(guān)問題拓展閱讀:
- 如何調(diào)用Linux內(nèi)核函數(shù)
- linux 怎樣調(diào)用daemon庫函數(shù)
如何調(diào)用Linux內(nèi)核函數(shù)
注意看這個(gè)文件
sysdeps/unix/sysv/linux/syscalls.list
里面記錄著系統(tǒng)調(diào)用的名字和一些屬性,具體我也沒有研究過,不懂。
再看select的實(shí)現(xiàn),很讓人驚訝,一旦使用,結(jié)果就是“報(bào)錯(cuò)“。
int
__select (nfds, readfds, writefds, exceptfds, timeout)
int nfds;
fd_set *readfds;
fd_set *writefds;
fd_set *exceptfds;
struct timeval *timeout;
{
__set_errno (ENOSYS);
return -1;
}
libc_hidden_def (__select)
stub_warning (select)
weak_alias (__select, select)
這是因?yàn)間libc并沒有實(shí)現(xiàn)系統(tǒng)調(diào)用,而是調(diào)用系統(tǒng)調(diào)用,
更進(jìn)一步,連調(diào)用系統(tǒng)調(diào)用都沒有一個(gè)鋒螞個(gè)實(shí)現(xiàn),而是使用了通用的辦法,
理由很簡(jiǎn)單,所有的系統(tǒng)調(diào)用在linux內(nèi)核頭文件里都能找到,
所有的系統(tǒng)調(diào)用參數(shù)類型就那么幾種,參數(shù)個(gè)數(shù)也是有限的,
因此沒有必要針對(duì)所有的系統(tǒng)調(diào)用一一封裝,
于是就有了這個(gè)list文件,自動(dòng)生成調(diào)用系統(tǒng)調(diào)用的函數(shù),
如果生成失敗,也就是你看到的“報(bào)錯(cuò)”。
符號(hào)是有強(qiáng)弱的,當(dāng)自掘租動(dòng)生成成功的時(shí)候,“報(bào)錯(cuò)”的弱符號(hào)就被忽略了。
當(dāng)你在glibc中找到判基兆一個(gè)系統(tǒng)調(diào)用的封裝源碼,是以下原因,
1. 編譯的目標(biāo)系統(tǒng)不支持這個(gè)系統(tǒng)調(diào)用,所以自己用另一種方式實(shí)現(xiàn)了。
2. 這個(gè)系統(tǒng)調(diào)用無法使用通用的自動(dòng)生成方式生成,用特化的方式覆蓋。
3. 針對(duì)這個(gè)系統(tǒng)調(diào)用做了特別的優(yōu)化。
4. 其它可能的原因。
具體可以留意
SYSCALL, PSEUDO, DO_CALL, INLINE_CALL 等名字
這兩個(gè)文件是重點(diǎn)所在
sysdeps/unix/i386/sysdep.h
sysdeps/unix/i386/sysdep.S
要搞清楚具體的自動(dòng)生成過程,恐怕得研究glibc自身的編譯過程了
linux 怎樣調(diào)用daemon庫函數(shù)
守護(hù)進(jìn)程(Daemon)是運(yùn)行在后臺(tái)的一種特殊進(jìn)程。它獨(dú)立于控制終端并且周期性地執(zhí)行某種任務(wù)或等待處理某些發(fā)生的事件。守護(hù)進(jìn)程是一種很有用的進(jìn) 程。Linux的核瞎模大多數(shù)服務(wù)器就是用守護(hù)進(jìn)程實(shí)現(xiàn)的。比如,Internet服務(wù)器inetd,Web服務(wù)器httpd等。同時(shí),守護(hù)進(jìn)程完成許多系統(tǒng)任 務(wù)。比如,作業(yè)規(guī)劃進(jìn)程crond,打印進(jìn)程lpd等。
守護(hù)進(jìn)程的編程本身并不復(fù)雜,復(fù)雜的是各種版本的Unix的實(shí)現(xiàn)機(jī)制不盡相同,造成不同Unix環(huán)境下守護(hù)進(jìn)程的編程規(guī)則并不一致。這需要讀者注意,照搬 某些書上的規(guī)則(特別是BSD4.3和低版本的System V)到Linux會(huì)出現(xiàn)錯(cuò)誤的。下面將全面介紹Linux下守護(hù)進(jìn)程的編程要點(diǎn)并給出詳細(xì)實(shí)例。
一. 守護(hù)進(jìn)程及其特性
守護(hù)進(jìn)程最重要的特性是后臺(tái)運(yùn)行。在這一點(diǎn)上DOS下的常駐內(nèi)存程序TSR與之相似。其次,守護(hù)進(jìn)程必須與其運(yùn)行前的環(huán)境隔離開來。這些環(huán)境包括未關(guān)閉的 文件描述符,控制終端,會(huì)話和進(jìn)程組,工作目錄以及文件創(chuàng)建掩模等。這些環(huán)境通常是守護(hù)進(jìn)程從執(zhí)行它的父進(jìn)程(特別是shell)中繼承下來的。最后,守 護(hù)進(jìn)程的啟動(dòng)方式有其特殊之處。它可以在Linux系統(tǒng)啟動(dòng)時(shí)從啟動(dòng)腳本/etc/rc.d中啟動(dòng),可以由作業(yè)規(guī)劃進(jìn)程crond啟動(dòng),還可以由用戶終端 (通常是shell)執(zhí)行。
總之,除開這些特殊性以外,守護(hù)進(jìn)程與普通進(jìn)程基本上沒有什么區(qū)別。因此,編寫守護(hù)進(jìn)程實(shí)際上是把一個(gè)普通進(jìn)改緩程按照上述的守護(hù)進(jìn)程的特性改造成為守護(hù)進(jìn)程。如果讀者對(duì)進(jìn)程有比較深入的認(rèn)識(shí)就更容易理解和編程了。
二. 守護(hù)進(jìn)程的編程要點(diǎn)
前面講過,不同Unix環(huán)境下守護(hù)進(jìn)程的編程規(guī)則并不一致。所幸的是守護(hù)進(jìn)程的編程原則其實(shí)都一樣,區(qū)別在于具體的實(shí)現(xiàn)細(xì)節(jié)不同。這個(gè)原則就是要滿足守護(hù) 進(jìn)程的特性。同時(shí),Linux是基于Syetem V的SVR4并遵循Posix標(biāo)準(zhǔn),實(shí)現(xiàn)起來與BSD4相比更方便。編程要點(diǎn)如下;
1. 在后臺(tái)運(yùn)行。
為避免掛起控制終端將Daemon放入后臺(tái)執(zhí)行。方法是在進(jìn)程中調(diào)用fork使父進(jìn)程終止,讓Daemon在子進(jìn)程中后臺(tái)執(zhí)行。
if(pid=fork())
exit(0);//是父進(jìn)程,結(jié)束父進(jìn)程,子進(jìn)程繼續(xù)
2. 脫離控制終端,登錄會(huì)話和進(jìn)程組
有必要先介紹一下Linux中的進(jìn)程與控制終端,登錄會(huì)話和進(jìn)程組之間的關(guān)系:進(jìn)程屬于一個(gè)進(jìn)程組,進(jìn)程組號(hào)(GID)就是進(jìn)程組長(zhǎng)的進(jìn)程號(hào)(PID)。登錄會(huì)話可以包含多個(gè)進(jìn)程組。這些進(jìn)程組共享一個(gè)控制終端。這個(gè)控制終端通常是創(chuàng)建進(jìn)程的登錄終端。
控制終端,登錄會(huì)話和進(jìn)程組通常是從父進(jìn)程繼承下來的。我們的目的就是要擺脫它們,使之不受它們的影響神鄭。方法是在第1點(diǎn)的基礎(chǔ)上,調(diào)用setsid()使進(jìn)程成為會(huì)話組長(zhǎng):
setsid();
說明:當(dāng)進(jìn)程是會(huì)話組長(zhǎng)時(shí)setsid()調(diào)用失敗。但之一點(diǎn)已經(jīng)保證進(jìn)程不是會(huì)話組長(zhǎng)。setsid()調(diào)用成功后,進(jìn)程成為新的會(huì)話組長(zhǎng)和新的進(jìn)程組長(zhǎng),并與原來的登錄會(huì)話和進(jìn)程組脫離。由于會(huì)話過程對(duì)控制終端的獨(dú)占性,進(jìn)程同時(shí)與控制終端脫離。
3. 禁止進(jìn)程重新打開控制終端
現(xiàn)在,進(jìn)程已經(jīng)成為無終端的會(huì)話組長(zhǎng)。但它可以重新申請(qǐng)打開一個(gè)控制終端??梢酝ㄟ^使進(jìn)程不再成為會(huì)話組長(zhǎng)來禁止進(jìn)程重新打開控制終端:
if(pid=fork())
exit(0);//結(jié)束之一子進(jìn)程,第二子進(jìn)程繼續(xù)(第二子進(jìn)程不再是會(huì)話組長(zhǎng))
4. 關(guān)閉打開的文件描述符
進(jìn)程從創(chuàng)建它的父進(jìn)程那里繼承了打開的文件描述符。如不關(guān)閉,將會(huì)浪費(fèi)系統(tǒng)資源,造成進(jìn)程所在的文件系統(tǒng)無法卸下以及引起無法預(yù)料的錯(cuò)誤。按如下方法關(guān)閉它們:
for(i=0;i 關(guān)閉打開的文件描述符close(i);>
for(i=0;i
#include
#include
#include
#include
void init_daemon(void)
{
int pid;
int i;
if(pid=fork())
exit(0);//是父進(jìn)程,結(jié)束父進(jìn)程
else if(pid
#include
void init_daemon(void);//守護(hù)進(jìn)程初始化函數(shù)
main()
{
FILE *fp;
time_t t;
init_daemon();//初始化為Daemon
while(1)//每隔一分鐘向test.log報(bào)告運(yùn)行狀態(tài)
{
sleep(60);//睡眠一分鐘
if((fp=fopen(“test.log”,”a”)) >=0)
{
t=time(0);
fprintf(fp,”I’m here at %sn”,asctime(localtime(&t)) );
fclose(fp);
}
}
}
以上程序在RedHat Linux6.0下編譯通過。步驟如下:
編譯:gcc –g –o test init.c test.c
執(zhí)行:./test
查看進(jìn)程:ps –ef
從輸出可以發(fā)現(xiàn)test守護(hù)進(jìn)程的各種特性滿足上面的要求。
linux 調(diào)用函數(shù)怎么寫的介紹就聊到這里吧,感謝你花時(shí)間閱讀本站內(nèi)容,更多關(guān)于linux 調(diào)用函數(shù)怎么寫,Linux 開發(fā)學(xué)習(xí)指南:函數(shù)調(diào)用詳解與實(shí)戰(zhàn),如何調(diào)用Linux內(nèi)核函數(shù),linux 怎樣調(diào)用daemon庫函數(shù)的信息別忘了在本站進(jìn)行查找喔。
創(chuàng)新互聯(lián)成都網(wǎng)站建設(shè)公司提供專業(yè)的建站服務(wù),為您量身定制,歡迎來電(028-86922220)為您打造專屬于企業(yè)本身的網(wǎng)絡(luò)品牌形象。
成都創(chuàng)新互聯(lián)品牌官網(wǎng)提供專業(yè)的網(wǎng)站建設(shè)、設(shè)計(jì)、制作等服務(wù),是一家以網(wǎng)站建設(shè)為主要業(yè)務(wù)的公司,在網(wǎng)站建設(shè)、設(shè)計(jì)和制作領(lǐng)域具有豐富的經(jīng)驗(yàn)。
標(biāo)題名稱:Linux開發(fā)學(xué)習(xí)指南:函數(shù)調(diào)用詳解與實(shí)戰(zhàn)(linux調(diào)用函數(shù)怎么寫)
網(wǎng)站鏈接:http://m.fisionsoft.com.cn/article/djogogo.html


咨詢
建站咨詢
