文件描述符的重定向: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;
}

这里写图片描述

这里写图片描述
结果可对比观察其异同。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值