zoukankan      html  css  js  c++  java
  • The Socket API, Part 4: Datagrams

    转:http://www.linuxforu.com/2011/11/socket-api-part-4-datagrams/

    UDP time

    Let’s try to develop server clients using UDP, the protocol behind some important services like DNS, NFS, etc.

    UDP, the User Datagram Protocol, is a connectionless protocol. This means you don’t establish a connection before sending data (like the three-way handshake for TCP); you just send the data in the form of a datagram, to the target address. That’s why we also call it a unreliable protocol, as we do not know or care if the datagram reached the recipient or not.

    The code

    Now, as we usually do, let’s jump to the program. The code in this article is a modified version of the echo server/clients programs in the UNIX Network Programming by Stevens et al. This file isudpserver.c:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    #include <stdio.h>
    #include <sys/socket.h>
    #include <strings.h>
    #include <netinet/in.h>
     
    int main()
    {
        int sfd, n;
        socklen_t len;
        char line[128];
        struct sockaddr_in saddr, caddr;
     
        sfd = socket(AF_INET, SOCK_DGRAM, 0);  
     
        bzero(&saddr, sizeof(saddr));
        saddr.sin_family = AF_INET;
        saddr.sin_addr.s_addr = htonl(INADDR_ANY);
        saddr.sin_port = htons(2910);
     
        bind(sfd, (struct sockaddr *)&saddr, sizeof(saddr));
     
        printf("Server running ");
        for(;;) {
            len=sizeof(caddr);
            n=recvfrom(sfd, line, 128, 0, (struct sockaddr *)&caddr, &len);
            sendto(sfd, line, n, 0, (struct sockaddr *)&caddr, len);
        }
     
        return 0;
    }

    This is almost like the previous code; let, let’s discuss the differences. First, the call to socket(): the SOCK_STREAM that was for TCP/SCTP is replaced with SOCK_DGRAM. Then we did a bind() to the address and port, and the server was ready to receive data from the client (which, in this case, is simple text that fits in a single packet).

    Next, you see there is no listen() call. Instead you are waiting to receive data at recvform() and when received, echo it back to the client using sendto().

    Before looking into these functions, observe that your server does not fork() to serve each request, but iteratively serves requests at each iteration in the infinite loop. That’s because in TCP, you have to maintain a connection with each client, and so need a separate thread, but in UDP you just reply to a single client at a time, and it’s free to receive data from the next client in the next iteration.

    Thus, you see that we need a buffer for the server that will hold data from clients in queue, to serve in a first-in-first-out order. Most TCP servers are concurrent, and UDP servers are iterative.

    Sending and receiving

    Let’s cover the new functions used for datagrams; first is recvfrom():

    #include <sys/socket.h>
    ssize_t recvfrom (int sockfd, void *buf, size_t nbytes,
        int flags, struct sockaddr *from, socklen_t *addrlen);

    This function blocks the server until it receives some data at sockfd from the address specified in the *from argument, and writes nbytes to *buf. The flag argument is formed by OR-ing one or more values, but we’re not using it, so it is 0. The last argument, *addrlen, is a pointer to the size of the socket address structure.

    Now, send data using sendto():

    #include <sys/socket.h>
    ssize_t sendto (int sockfd, const void *buf, size_t nbytes,
        int flags, const struct sockaddr *to, socklen_t addrlen);

    This function sends the *buf data of size nbytes through sockfd to the address stored in the *tostructure. Its size is addrlen, and do note that it is not a reference, as it was in recvfrom(). The flags argument is as in recvfrom(). Both functions return the number of bytes read/written.

    And the code for the client, udpclient.c, is as follows:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    /**
     
    #include <sys/socket.h>
     
    ssize_t recvfrom (int sockfd, void *buff, size_t nbytes,
        int flags, struct sockaddr *from, socklen_t *addrlen);
     
    ssize_t sendto (int sockfd, const void *buff, size_t nbytes,
        int flags, const struct sockaddr *to, socklen_t addrlen);
     
    **/
     
     
    #include <stdio.h>
    #include <sys/socket.h>
    #include <string.h>
    #include <netinet/in.h>
    #define MAX 100
     
    int main(int argc, char** argv)
    {
        int sfd, n;
        socklen_t len;
        char sline[MAX], rline[MAX+1];
        struct sockaddr_in saddr;
     
        if(argc!=2) {
            printf("Usage: %s ipaddress ", argv[0]);
            return -1;
        }
     
        sfd = socket(AF_INET, SOCK_DGRAM, 0);  
     
        bzero(&saddr, sizeof(saddr));
        saddr.sin_family = AF_INET;
        inet_pton(AF_INET, argv[1], &saddr.sin_addr);
        saddr.sin_port = htons(2910);
     
        printf("Client Running ");
        while(fgets(sline, MAX, stdin)!=NULL) {
            len=sizeof(saddr);
            sendto(sfd, sline, strlen(sline), 0, (struct sockaddr *)&saddr, len);
            n=recvfrom(sfd, rline, MAX, 0, NULL, NULL);
            rline[n]=0;
            fputs(rline, stdout);
        }
     
        return 0;
    }

    The client program is very simple -– it just sends the line read from the standard input to the server using sendto(), and reads the reply from the server using recvfrom().

    Here, don’t specify the address and port you are receiving from, so any random client/server can send data to you if it knows the port number assigned to your client. If you are interested in replies from a specific server, you can store the structure returned by the recvfrom() and compare it with the structure used in sendto().

    This may be a problem if your server is multi-homed (with multiple IPs), and the server has not bound the address like we usually do, using INADDR_ANY.

    Starting server

    Figure 1: Starting server

     

    Client running

    Figure 2: Client running

    Before ending, let’s look into some problems with UDP.

    First, there is no way to know if your datagram has reached the other side. You cannot know whether the request or the reply is lost. This is okay for a server, but your client may keep on waiting for a reply. In that case, you can specify a time-out in the call to recvfrom, so it doesn’t wait forever.

    And here, I end the article, signing out with my usual, “FOSS ROCKS!”

  • 相关阅读:
    如何系统收集网页/电子书等相关的知识点
    针式PKM软件多个版本兼容功能的应用
    软件开发出路:做精某一卖点!
    深入针式PKM应用系列(1) 虚拟文件夹功能,让文件想放哪就放哪
    PKM软件:提供知识应用的工具支持
    解决长串英文字母显示不能自动换行的问题
    C#中判断字符串A中是否包含字符串B
    小技巧锦集
    解决Notes的"The remote server is not a known tcp/IP host"问题的方法
    将Sql Server自增长字段的目前识别值重调!
  • 原文地址:https://www.cnblogs.com/pengdonglin137/p/3467996.html
Copyright © 2011-2022 走看看