我們都知道,一個進程在存在期間,會有一些文件被打開,從而會返回一些文件描述符,從shell中運行一個進程,默認(rèn)會有3個文件描述符存在,即0,1,2.0與進程的標(biāo)準(zhǔn)輸入相關(guān)聯(lián),1與進程的標(biāo)準(zhǔn)輸出相關(guān)聯(lián),2與進程的標(biāo)準(zhǔn)輸出錯誤相關(guān)聯(lián)。
創(chuàng)新互聯(lián)服務(wù)項目包括梓潼網(wǎng)站建設(shè)、梓潼網(wǎng)站制作、梓潼網(wǎng)頁制作以及梓潼網(wǎng)絡(luò)營銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢、行業(yè)經(jīng)驗、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,梓潼網(wǎng)站推廣取得了明顯的社會效益與經(jīng)濟效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到梓潼省份的部分城市,未來相信會繼續(xù)擴大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!
dup和dup2函數(shù)用來復(fù)制一個文件的描述符,用于文件描述符的重定向,重定向一個進程的0,1,2文件描述符。這兩個函數(shù)的原型如下:
int dup(int oldfd);
int dup2(int oldfd,int newfd);
利用dup函數(shù),我們可以復(fù)制一個文件描述符,傳給該函數(shù)一個既有的描述符,它就會返回一個新的描述符,這個新的描述符是傳給它的舊描述符的拷貝,這意味著這兩個文件描述符共享同一個數(shù)據(jù)結(jié)構(gòu),即所有的鎖定,讀寫指針,各項權(quán)限和標(biāo)志位等等。但復(fù)制成功時返回最小的尚未被使用的文件描述符,這個文件描述符指向oldfd所擁有的文件表項,例如,假如oldfd的值為1,當(dāng)前文件文件描述符的最小值為3,那么新描述符3指向描述符1所擁有的文件表項。出錯返回-1,錯誤代碼在errno中。
dup2函數(shù)可以用參數(shù)newfd指定新文件描述符的數(shù)值,若newfd已經(jīng)被程序使用,系統(tǒng)就會將其關(guān)閉以釋放該文件描述符。若newfd與oldfd相等,dup2將返回newfd,而不關(guān)閉它。dup2調(diào)用成功返回新的文件描述符,出錯返回-1.dup2函數(shù)可以讓用戶自己指定文件描述符,它的作用也是復(fù)制文件描述符,將newfd描述符所對應(yīng)的文件表改成oldfd所對應(yīng)的文件表項。之后,newfd與oldfd指向同一個文件表,這樣就將newfd重定向到oldfd。
dup和dup2的區(qū)別就是可以用newfd參數(shù)指定新描述符的數(shù)值,如果newfd已經(jīng)打開,則先將其關(guān)閉,如果newfd等于oldfd,則dup2返回newfd而不關(guān)閉它。
下面我們來看看一段有關(guān)dup的代碼:
#include#include #include #include #include #include #include #define _FILE_ "./log" int main() { umask(0); int fd=open(_FILE_,O_CREAT |O_WRONLY,0644); printf("%d\n",fd); if(fd<0) { perror("open"); return 1; } close(1); int new_fd=dup(fd); if(new_fd<0) { perror("dup"); return 2; } close(fd); printf("new_fd:%d\n",new_fd); int count=0; while(count++<100) { printf("hello world\n"); fflush(stdout); } close(new_fd); return 0; }
運行結(jié)果如下:
我們可以看到,當(dāng)沒有close(1)的時候,新open的文件描述符是3,當(dāng)我們close(1)后并調(diào)用dup函數(shù)后,new_fd變成了1,本來將要輸出到終端的內(nèi)容被重定向到了log文件里。
dup2函數(shù)示例:
#include#include #include #include #include #include #include #define _FILE_ "./log" int main() { umask(0); int fd=open(_FILE_,O_CREAT |O_WRONLY,0644); if(fd<0) { perror("open"); exit(0); } close(1); int ret=dup2(fd,1);//復(fù)制oldfd到文件描述符1(標(biāo)準(zhǔn)輸出) if(ret<0) { perror("dup2"); exit(1); } char buf[1024]; int done=0; while(!done) { memset(buf,'\0',sizeof(buf)); ssize_t _s=read(0,buf,sizeof(buf)-1); if(_s<=0) { perror("read"); exit(2); } else if(_s>0) { buf[_s]='\0'; if(strncmp(buf,"quit",4)==0) { done=1; continue; } } printf("%s\n",buf); fflush(stdout); } return 0; }
運行結(jié)果如下:
從結(jié)果可以看到,我們從標(biāo)準(zhǔn)輸入輸入的數(shù)據(jù)全都被重定向到了log文件里面了。