[linux]--使用dup和dup2重定向和還原
阿新 • • 發佈:2019-02-14
源於開發一個應用將資料寫的裝置節點中,但是裝置節點具有可變性,所以不在寫死,而是先確定好,開啟fd再將內容寫進去,最終寫入fd中的是一個開源程式,沒有辦法傳遞fd,不過它會把它要輸出的東西輸出到stdout中,開啟這個開源程式之前把stdout重定向到fd上就可以完成了對接。鋪墊完了,下面就開始正題。
其實這個APUE上有講這個兩個函式,但是說的太標準了的書面語,也沒有給例子,就沒有辦法快速理解和使用了。看到一個10年前的好貼《使用dup2重定向了標準輸出後,使用什麼方法恢復對終端的輸出??》解決了問題。完整程式碼如下:
[cpp] view plain copy- #include <stdio.h>;
- #include <unistd.h>;
- #include <stdlib.h>;
- #include <fcntl.h>;
- #include <sys/types.h>;
- #include <sys/stat.h>;
- #include <string.h>;
- #include <strings.h>;
- int main()
- {
- int sfd = dup(STDOUT_FILENO), testfd;
- printf("sfd = [%d]\n"
- testfd = open("./temp",O_CREAT | O_RDWR | O_APPEND);
- if (-1 == testfd)
- {
- printf("open file error.\n");
- exit(1);
- }
- /* 重定向 */
- if (-1 == dup2(testfd,STDOUT_FILENO) ) {
- printf("can't redirect fd error\n"
- exit(1);
- }
- /* 此時向stdout寫入應該輸出到檔案 */
- write(STDOUT_FILENO,"file\n",5);
- /* 恢復stdout */
- if (-1 != dup2(sfd,STDOUT_FILENO) ) {
- printf("recover fd ok \n");
- /* 恢復後,寫入stdout應該向螢幕輸出 */
- write(STDOUT_FILENO,"stdout\n",7);
- }
- printf("gogogogogogo!\n");
- close(testfd);
- }
#include <stdio.h>;
#include <unistd.h>;
#include <stdlib.h>;
#include <fcntl.h>;
#include <sys/types.h>;
#include <sys/stat.h>;
#include <string.h>;
#include <strings.h>;
int main()
{
int sfd = dup(STDOUT_FILENO), testfd;
printf("sfd = [%d]\n", sfd);
testfd = open("./temp",O_CREAT | O_RDWR | O_APPEND);
if (-1 == testfd)
{
printf("open file error.\n");
exit(1);
}
/* 重定向 */
if (-1 == dup2(testfd,STDOUT_FILENO) ) {
printf("can't redirect fd error\n");
exit(1);
}
/* 此時向stdout寫入應該輸出到檔案 */
write(STDOUT_FILENO,"file\n",5);
/* 恢復stdout */
if (-1 != dup2(sfd,STDOUT_FILENO) ) {
printf("recover fd ok \n");
/* 恢復後,寫入stdout應該向螢幕輸出 */
write(STDOUT_FILENO,"stdout\n",7);
}
printf("gogogogogogo!\n");
close(testfd);
}
總結:重定向好似抗戰片中的將鐵軌移到別的路上,或者走向深谷/dev/null,或者走向想讓其走的地方fd。再說下,一開始以為我還以為dup2(STDOUT_FILENO, STDOUT_FILENO);就恢復了呢,結果關閉fd時候,顯示關閉成功,但是裝置卻再也打不開了,因為我並沒有真正還原。
另外:這裡給出一些乾貨以明確理解。