dup/dup2的使用請參考其他資料,個人只是想了解dup后文件描述符,進程表項,文件表的關系。
進程要對文件進行操作,一般使用open調用打開一個文件進行訪問,每個進程都有一個文件描述符表,該表中存放打開的文件描述符。用戶使用open等調用得到的文件描述符其實是文件描述符在該表中的索引號,該表項的內容是一個指向文件表的指針。應用程序只要使用該描述符就可以對指定文件進行操作。
為了了解dup與賦值語句用于文件描述符的區別,請看如下程序。
程序描述:
打開一個文件描述符,分別適用dup和賦值語句進行復制,復制之后,打印原始和被復制的文件描述符id,看看是否具有相同的值,然后關閉文件,測試關閉是否成功。
程序示例:
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <unistd.h>
int sys_err(char *str)
{
puts(str);
exit(0);
}
int main(void)
{
int p,q;
if((p=open("c_fid.c", O_RDONLY)) == -1)
sys_err("open error");
q = dup(p);
puts("dup:");
printf("file p,q fd is:%d %d\n", q, p);
printf("close file p ok?: %d\n", close(p));
printf("close file q ok?: %d\n", close(q));
if((p=open("c_fid.c", O_RDONLY)) == -1)
sys_err("open error");
q = p;
puts("=:");
printf("file p,q fd is:%d %d\n", q, p);
printf("close file p ok?: %d\n", close(p));
printf("close file q ok?: %d\n", close(q));
return 0;
}
程序運行結果:
dup:
file p,q fd is:4 3 //文件p,q使用不同的文件描述符
close file p ok?: 0
close file q ok?: 0 //文件關閉成功
=:
file p,q fd is:3 3 //簡單復制
close file p ok?: 0
close file q ok?: -1//關閉失敗,原因是此描述符已經被關閉了
由此證明,dup是產生一個新的文件描述符id和指針在進程表項中,但是他們共用文件表,這時,關閉一個文件描述符,另外一個仍舊可用,文件表并不會被釋放。而賦值語句不同,它只是簡單的在另外一個變量中記錄原始文件指針等,2個變量的文件描述符相同,進程表項中并不產生新的項目。
關于socket的文件描述符
socket接口增加了網絡通信操作的抽象定義,與文件操作一樣,每個打開的socket都對應一個整數,我們稱它為socket描述符,該整數也是socket描述符在文件描述符表中的索引值。但socket描述符在描述符表中的表項并不指向文件表,而是指向一個與該socket有關的數據結構。BSD UNIX中新增加了一個socket調用,應用程序可以調用它來新建一個socket描述符,注意進程用open只能產生文件描述符,而不能產生socket描述符。socket調用只能完成建立通信的部分工作,一旦建立了一個socket,應用程序可以使用其他特定的調用來為它添加其他詳細信息,以完成建立通信的過程。