1. 程式人生 > >檔案描述符的重定向:dup/dup2

檔案描述符的重定向:dup/dup2


這裡寫圖片描述
dup/dup2:進行檔案描述符的重定向即建立一個oldfd的副本。
dup:最低編號、未被使用的檔案描述符是oldfd的一份拷貝。
dup2:newfd是oldfd的一份拷貝。
返回值:成功(newfd); 失敗(-1)。
基於TCP的socket程式設計:http://blog.csdn.net/better_jh/article/details/72846264
基於TCP的socket程式設計中的多執行緒伺服器為例,將標準輸出重定向到client端的socket中。
程式碼示例:

//server端
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h> #include <pthread.h> #include <arpa/inet.h> #include <stdlib.h> #include <string.h> #include <unistd.h> static void usage(const char* proc) { printf("Usage:%s[local_ip][local_port]\n",proc); } int startup(const char* _ip,int _port) { int
sock = socket(AF_INET,SOCK_STREAM,0); if(sock<0) { perror("socket"); return 1; } struct sockaddr_in servaddr; servaddr.sin_family = AF_INET; servaddr.sin_port = htons(_port); servaddr.sin_addr.s_addr = inet_addr(_ip); if(bind(sock,(struct sockaddr*)&servaddr,sizeof
(servaddr))<0) { perror("bind"); return 2; } if(listen(sock,10)<0) { perror("listen"); return 3; } return sock; } void *handler(void * _sock) { int sock = (int)_sock; char buf[1024]; char *msg = "HTTP/1.0 200 ok\r\n\r\n<html><h1>HELLO WORLD</h1></html>\r\n"; while(1) { ssize_t s =read(sock,buf,sizeof(buf)-1); if(s > 0) { buf[s] = 0; printf("client say:%s\n",buf); write(sock,msg,strlen(msg)); } else { printf("client is quit\n"); break; } fflush(stdout); } return (void *)0; } int main(int argc,char *argv[]) { if(argc != 3) { usage(argv[0]); exit(1); } int listen_sock = startup(argv[1],atoi(argv[2])); struct sockaddr_in clientaddr; socklen_t len = sizeof(clientaddr); while(1) { int sock = accept(listen_sock,(struct sockaddr*)&clientaddr,&len); if(sock<0) { perror("accept"); continue; } printf("client ip:%s port:%d\n",inet_ntoa(clientaddr.sin_addr),ntohs(clientaddr.sin_port)); pthread_t id ; if(pthread_create(&id,NULL,handler,(void *)sock)<0) { perror("pthread_create"); exit(2); } pthread_detach(id); } return 0; }
//client端
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
static void usage(const char* proc)
{
    printf("Usage:%s[server_ip][server_port]\n",proc);
}
int main(int argc,char *argv[])
{
    if(argc != 3)
    {
        usage(argv[0]);
        return 1;
    }
    int sock = socket(AF_INET,SOCK_STREAM,0);
    if(sock<0)
    {
        perror("socket");
        return 2;
    }
    struct sockaddr_in peer;
    peer.sin_family = AF_INET;
    peer.sin_port = htons(atoi(argv[2]));
    peer.sin_addr.s_addr = inet_addr(argv[1]);
    int ret = connect(sock,(struct sockaddr*)&peer,sizeof(peer));
    if(ret<0)
    {
        perror("connect");
        printf("%s\n",strerror(ret));
        return 3;
    }
    dup2(sock,1);//將標準輸出重定向至sock
    char buf[1024];
    while(1)
    {
        printf("Please enter:");
        fflush(stdout);
        ssize_t s=read(0,buf,sizeof(buf));
        if(s<0)
        {
            perror("read");
            return 4;
        }
        buf[s-1]=0;
        write(sock,buf,strlen(buf));
        read(sock,buf,sizeof(buf));
        printf("server echo: %s\n",buf);        
    }
    close(sock);
    return 0;
}

這裡寫圖片描述

這裡寫圖片描述
結果可對比觀察其異同。