一个可通过UDP和TCP套接字连接的回显服务器

服务器端代码:

#include <stdio.h>
#include <stdlib.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <errno.h>
#include <string.h>
#include <unistd.h>
#include <signal.h>
#include <sys/wait.h>

#define SERV_PORT 9877
#define MAXLINE 4096
#define LISTENQ 1024
void sig_child(int signo)
{
    int status;
    while (waitpid(-1,&status,WNOHANG) < 0)
    {
    printf("waitpid %s\n",strerror(errno));
    exit(1);
    }
}

void client(int tcpfd,int udpfd)
{
    fd_set fs;
    FD_ZERO(&fs);
    FD_SET(tcpfd,&fs);
    FD_SET(udpfd,&fs);
    pid_t pid;
    int maxfd = tcpfd > udpfd ? tcpfd: udpfd;
    int confd;
    for (; ;)
    {
    FD_SET(tcpfd,&fs);
    FD_SET(udpfd,&fs);

    if (select(maxfd + 1,&fs,NULL,NULL,NULL) < 0)
    {
        if (errno == EINTR)
        continue;
        printf("select %s\n",strerror(errno));
        exit(1);
    }

    if (FD_ISSET(tcpfd,&fs))
    {
        if ((confd = accept(tcpfd,NULL,NULL)) < 0)
        {
        printf("accept: %s\n",strerror(errno));
        exit(1);
        }

        if ((pid = fork()) < 0)
        {
        printf("fork: %s\n",strerror(errno));
        exit(1);
        }
        else if (pid == 0)
        {

        char buf[MAXLINE];
        int n;
            for (; ;)
        {

            if ((n = read(confd,buf,MAXLINE)) < 0)
            {
            printf("read: %s\n",strerror(errno));
            exit(1);
            }
            printf("read over\n");  
            if (write(confd,buf,n) != n)
            {
            printf("write: %s\n",strerror(errno));
            exit(1);
            }
        }
        exit(0);
        }
    }

    if (FD_ISSET(udpfd,&fs))
    {
        char buf[MAXLINE];
        int n;
        struct sockaddr_in clientaddr;
        socklen_t socklen = sizeof(clientaddr);
        if ((n = recvfrom(udpfd,buf,MAXLINE,0,(struct sockaddr *)&clientaddr,&socklen)) < 0)
        {
        printf("recvfrom %s\n",strerror(errno));
        exit(1);
        }

        if (sendto(udpfd,buf,n,0,(struct sockaddr*)&clientaddr,socklen) != n)
        {
        printf("sendto: %s\n",strerror(errno));
        exit(1);
        }
    }
    }
}

int main()
{
    int udpfd,tcpfd;

    if ((udpfd = socket(AF_INET,SOCK_DGRAM,0)) < 0)
    {
    printf("socket: %s\n",strerror(errno));
    exit(1);
    }

    if ((tcpfd = socket(AF_INET,SOCK_STREAM,0)) < 0)
    {
    printf("socket: %s\n",strerror(errno));
    exit(1);
    }

    struct sockaddr_in servaddr;

    bzero(&servaddr,sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(SERV_PORT);
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);

    if (bind(udpfd,(struct sockaddr*)&servaddr,sizeof(servaddr)) < 0)
    {
    printf("udp bind: %s\n",strerror(errno));
    exit(1);
    }

    bzero(&servaddr,sizeof(servaddr));

    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(SERV_PORT);
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);

    if (bind(tcpfd,(struct sockaddr*)&servaddr,sizeof(servaddr)) < 0)
    {
    printf("tcp bind: %s\n",strerror(errno));
    exit(1);
    }

    if (listen(tcpfd,LISTENQ) < 0)
    {
    printf("listen %s\n",strerror(errno));
    exit(1);
    }

    if (signal(SIGCHLD,sig_child) == SIG_ERR)
    {
    printf("signal: %s\n",strerror(errno));
    exit(1);
    }
    client(tcpfd,udpfd);
}
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值