從程序完成的功能來看,函數庫提供的函數通常是不需要操作系統的服務. 函數是在用戶空間內執行的,除非函數涉及到I/O操作等,一般是不會切到核心態的。系統調用是要求操作系統為用戶提供進程,提供某種服務,通常是涉及系統的硬件資源和一些敏感的軟件資源等。
函數庫的函數,尤其與輸入輸出相關的函數,大多必須通過Linux的系統調用來完成。因此我們可以將函數庫的函數當成應用程序設計人員與系統調用程序之間的一個中間層,通過這個中間層,我們可以用一致的接口來安全的調用系統調用。這樣程序員可以只要寫一次代碼就能夠在不同版本的linux系統間使用積壓種具體實現完全不同的系統調用。至於如何實現對不同的系統調用的兼容性問題,那是函數庫開發者所關心的問題。
從程序執行效率來看,系統調用的執行效率大多要比函數高,尤其是處理輸入輸出的函數。當處理的數據量比較小時,函數庫的函數執行效率可能比較好,因為函數庫的作法是將要處理的數據先存入緩沖區內,等到緩沖區裝滿了,再將數據一次寫入或者讀出。這種方式處理小量數據時效率比較高,但是在進行系統調用時,因為用戶進程從用戶模式進入系統核心模式,中間涉及了許多額外的任務的切換工作,這些操作稱為上下文切換,此類的額外工作會影響系統的執行效率。但是當要處理的數據量比較大時,例如當輸入輸出的數據量超過文件系統定義的盡寸時,利用系統調用可獲得較高的效率。
從程序的可移植性的角度來看,相對於系統調用,C語言的標准備函數庫(ANSI C) 具備較高的可移植性,在不同的系統環境下,只要做很少的修改,通常情況是不需要修改的。
Linux C中庫函數和系統調用的區別 www.2cto.com
摘自:The Linux Kernel Module Programming Guide
庫函數是高層的,完全運行在用戶空間, 為程序員提供調用真正的在幕後完成實際事務的系統調用的更方便的接口。系統調用在內核態運行並且由內核自己提供。標准C庫函數printf()可以被看做是一個通用的輸出語句,但它實際做的是將數據轉化為符合格式的字符串並且調用系統調用write()輸出這些字符串。
是否想看一看printf()究竟使用了哪些系統調用? 這很容易,編譯下面的代碼。
#include <stdio.h>
int main(void)
{
printf("hello");
return 0;
}
使用命令
gcc -Wall -o hello hello.c
編譯。用命令strace hello 跟蹤該可執行文件。是否很驚訝? 每一行都和一個系統調用相對應。strace是一個非常有用的程序,它可以告訴你程序使用了哪些系統調用和這些系統調用的參數,返回值。 這是一個極有價值的查看程序在干什麼的工具。在輸出的末尾,你應該看到這樣類似的一行write(1, "hello", 5hello)。這就是我們要找的。藏在面具printf() 的真實面目。既然絕大多數人使用庫函數來對文件I/O進行操作(像fopen, fputs, fclose)。 你可以查看man說明的第二部分使用命令man 2 write 。man說明的第二部分專門介紹系統調用(像kill()和read())。man說明的第三部分則專門介紹你可能更熟悉的庫函數(像cosh()和random())。
你甚至可以編寫代碼去覆蓋系統調用,正如我們不久要做的。駭客常這樣做來為系統安裝後門或木馬。 但你可以用它來完成一些更有益的事,像讓內核在每次某人刪除文件時輸出 “Tee hee, that tickles!” 的信息。