Chinaunix首页 | 论坛 | 博客
  • 博客访问: 221935
  • 博文数量: 38
  • 博客积分: 2060
  • 博客等级: 大尉
  • 技术积分: 388
  • 用 户 组: 普通用户
  • 注册时间: 2009-03-17 10:10
文章分类

全部博文(38)

文章存档

2011年(1)

2009年(37)

我的朋友

分类: LINUX

2009-07-17 19:45:30

#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <time.h>
#include <sys/time.h>
#include <errno.h>
#include <stdlib.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string.h>
#include <sys/types.h>
#include <sys/select.h>

#define SERV_ADDR "127.0.0.1"
#define SERV_PORT 5358
#define BUF_LEN 1024

#define MAX(a, b) (a)>(b)?(a):(b)

ssize_t writen(int fd, const void *vptr, size_t n)
{
    size_t nleft;
    ssize_t nwriten;
    const char *ptr;

    ptr = vptr;
    nleft = n;
    while(nleft>0) {
        if((nwriten = write(fd, ptr, nleft)) <= 0) {
            if(nwriten < 0 && errno == EINTR) {
                nwriten = 0;        /*interrupt by signal*/
            } else {
                return -1;
            }
        }

        nleft -= nwriten;
        ptr += nwriten;
    }
    return n;
}

char *gf_time()
{
    struct timeval tv;
    static char str[30];
    char *ptr;

    if(gettimeofday(&tv, NULL) < 0)
        printf("gettimeofday error.\n");

    ptr = ctime(&tv.tv_sec);
    strcpy(str, &ptr[11]);
    snprintf(str+8, sizeof(str)-8, ".%06ld", tv.tv_usec);

    return str;
}

void str_cli(FILE *fp, int sockfd)
{
    int maxfdp1, val, stdineof;
    ssize_t n, nwritten;
    fd_set rset, wset;
    char to[BUF_LEN] = {0}, fr[BUF_LEN] = {0};
    char *toiptr, *tooptr, *froptr, *friptr;

    val = fcntl(sockfd, F_GETFL, 0);
    fcntl(sockfd, F_SETFL, val | O_NONBLOCK);

    val = fcntl(STDIN_FILENO, F_GETFL, 0);
    fcntl(STDIN_FILENO, F_SETFL, val | O_NONBLOCK);

    val = fcntl(STDOUT_FILENO, F_GETFL, 0);
    fcntl(STDOUT_FILENO, F_SETFL, val | O_NONBLOCK);

    toiptr = tooptr = to;
    friptr = froptr = fr;
    stdineof = 0;

    maxfdp1 = MAX(MAX(STDIN_FILENO, STDOUT_FILENO), sockfd) + 1;

    while(1) {
        FD_ZERO(&rset);
        FD_ZERO(&wset);
        if(stdineof == 0 && toiptr < &to[BUF_LEN])
            FD_SET(STDIN_FILENO, &rset);
        if(friptr < &fr[BUF_LEN])
            FD_SET(sockfd, &rset);
        if(tooptr != toiptr)
            FD_SET(sockfd, &wset);
        if(froptr != friptr)
            FD_SET(STDOUT_FILENO, &wset);
        select(maxfdp1, &rset, &wset, NULL, NULL);

        if(FD_ISSET(STDIN_FILENO, &rset)) {
            if((n = read(STDIN_FILENO, toiptr, &to[BUF_LEN] - toiptr)) < 0) {
                if(errno != EAGAIN) {
                    printf("read error\n");
                    return;
                }
            } else if(n == 0) {
                fprintf(stderr, "%s: EOF on stdin\n", gf_time());
                stdineof = 1;
                if(tooptr == toiptr)
                    shutdown(sockfd, SHUT_WR);
            } else {
                fprintf(stderr, "%s: read %d bytes from stdin\n", gf_time(), n);
                toiptr += n;
                FD_SET(sockfd, &wset);
            }
        }
        if(FD_ISSET(sockfd, &rset)) {
            if((n = read(sockfd, friptr, &fr[BUF_LEN] - friptr)) < 0) {
                if(errno != EAGAIN) {
                    printf("read error\n");
                    return;
                }
            } else if(n == 0) {
                fprintf(stderr, "%s: EOF on socket\n", gf_time());
                if(stdineof)
                    return;
                else {
                    printf("strcli: server terminated prematurely!\n");
                    return;
                }
            } else {
                fprintf(stderr, "%s: read %d bytes from socket\n", gf_time(), n);
                friptr += n;
                FD_SET(STDOUT_FILENO, &wset);
            }
        }
        if(FD_ISSET(STDOUT_FILENO, &wset) && ((n = friptr - froptr) > 0)) {
            if((nwritten = write(STDOUT_FILENO, froptr, n)) < 0) {
                if(errno != EWOULDBLOCK)
                    printf("write error to stdout.\n");
            } else {
                fprintf(stderr, "%s: wrote %d bytes to stdout\n", gf_time(), nwritten);
                froptr += nwritten;
                if(froptr == friptr)
                    froptr = friptr = fr;
            }
        }
        if(FD_ISSET(sockfd, &wset) && ((n = toiptr - tooptr) > 0 )) {
            if((nwritten = write(sockfd, tooptr, n)) < 0) {
                if(errno != EWOULDBLOCK)
                    printf("write error to socket.\n");
            } else {
                fprintf(stderr, "%s: wrote %d bytes to socket\n", gf_time(), nwritten);
                tooptr += nwritten;
                if(tooptr == toiptr) {
                    tooptr = toiptr = to;
                    if(stdineof)
                        shutdown(sockfd, SHUT_WR);
                }
            }
        }
    }
    return;
}

int main(int argc, char **argv)
{
    int fd;
    struct sockaddr_in servaddr;

    fd = socket(AF_INET, SOCK_STREAM, 0);

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

    if (connect(fd, (struct sockaddr *)&servaddr, sizeof(servaddr)) == -1) {
        printf("connect error: %s\n", strerror(errno));
        return 0;
    }

    str_cli(stdin, fd);
    return 0;
}

 

优点:

是所有版本中速度最快的。

不足之处:

代码太多太复杂,而且还要维护一个自己的buffer,相当不划算.

阅读(1201) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~