linux网络编程--第一弹UDP基本通信

来源:互联网 发布:淘宝售后维修时间 编辑:程序博客网 时间:2024/05/13 04:34

 在linux下用网络通信个人觉得是件比较好玩的事情。在用socket编程时,发现可以先不了解UDP,TCP传输层及传输层以下的事情,是件轻松的事情,当然后续会把tcp ip详解卷给补上。同时也发现了一本很不错的书籍 《unix网络编程卷一》,来学习linux C/C++网络编程。

                                                                                                               UDP程序所用套接字

      

        
        先介绍UDP中用到的socket的几个函数:
        /************************************************/
        #include <sys/types.h>    
        #include <sys/socket.h>
       int socket(int domain, int type, int protocol);
       /************************************************/
     为了执行网络I/O,一个进程必须要做的第一件事情就是调用socket函数。
     arg1 domain -- 协议族类,此处一致设置为AF_INET,为IPV4协议族。
     arg2 type -- 套接字类型,SOCK_STREAM--用于TCP,字节流套接字,
                                                   SOCK_DGRAM--用于UDP, 数据报套接字。
     arg3 protocol --协议类型常值,设为0,根据domain 和type可以默认设定系统协议。
     retn(返回值) 套接字描述符--sockfd,成功返回非负描述符,出错为-1。
     
     /***************************************************************************/
     #include <sys/types.h>        
     #include <sys/socket.h>
     int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
     /***************************************************************************/
     bind函数把一个本地协议地址赋予一个套接字,对于网际协议,协议地址是32位IPv4地址与16位的TCP或UDP端口号的组合。
     arg2 addr -- 指向特定于协议的地址结构指针,结构体成员sin_addr和sin_port,ip地址和端口号。
     arg3 addrlen -- 该地址结构的长度。
     retn(返回值) 成功则为0, 出错则为-1。
     [b]调用bind函数可以指定一个端口号,一个IP地址或者两者都指定,也可以两者都不指定。
                   sin_addr.s_addr选择为统配地址时常值INADDR_ANY, 有内核决定IPv4地址即本机IPv4地址。
                   sin_port选择为0时, 有内核决定临时端口号。[/b]


     /**********************************************************************/
     #include <sys/types.h>
     #include <sys/socket.h>
     ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags, struct sockaddr *src_addr, socklen_t *addrlen);
     /***********************************************************************/
    recvfrom函数使用数据报协议(UDP)接收的典型用途。
    argv2 -- buf,读出缓冲区指针
    argv3 -- len,读出缓冲区大小,
    argv4 -- flags,总设置为0
    argv5 -- src_addr,,数据报发送者的协议地址套接字地址结构
    argv6 -- addrlen,套接字结构地址中字节数指针。
    retn -- 成功为写字节数,出错为-1。     


    /*************************************************************************/
     #include <sys/types.h>
     #include <sys/socket.h>
     ssize_t sendto(int sockfd, const void *buf, size_t len, int flags,const struct sockaddr *dest_addr, socklen_t addrlen);
     /*************************************************************************/
     sendto函数使用数据报协议(UDP)发送的典型用途。
     argv2 -- buf, 写入缓冲区指针。
     argv3 -- len, 写入缓冲区大小。
     argv4 -- flags,总设置为0。
     argv5 -- dest_addr, 数据报目的协议地址套接字地址结构
     argv6 -- addrlen,套接字地址结构字节数。
     retn -- 成功为写字节数,出错为-1。     


如下代码为UDP服务器:
[code=c]
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h> 
#include <netinet/in.h> 
#include <sys/socket.h>
#include <arpa/inet.h>


#define BUFFERSIZE  1500
char recvbuf[BUFFERSIZE];


int main (int argc, char *argv[])
{
    int sockfd;
    uint32_t recvNum;
    uint32_t addrlen;
    struct sockaddr_in addr_local;
    struct sockaddr    addr_remort;
    char sendBuffer[] = "Linux Received.";




    sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd < 0)
    {
        printf ("Socket Descriptor Error.\r\n");


        return 1;
    }


    bzero(&addr_local, sizeof(addr_local));
    addr_local.sin_family = AF_INET;
    addr_local.sin_port = htons(8080);
    addr_local.sin_addr.s_addr = htonl(INADDR_ANY);     //local IP addr
    if (bind(sockfd, (struct sockaddr *)&addr_local, sizeof(addr_local)) < 0)
    {
        printf ("Socket Binding Error.\r\n");
        return 1;
    }


    for (;;)
    {
        recvNum = recvfrom(sockfd, recvbuf, sizeof(recvbuf), 0, (struct sockaddr*)&addr_remort, &addrlen);
        if (recvNum > 0)
        {
            sendto(sockfd, sendBuffer, sizeof(sendBuffer), 0, (struct sockaddr*)&addr_remort, addrlen);
            printf ("Recv Data: %s \r\n", recvbuf);
        }
    }


    return 0;
}
[/code]
      
0 0
原创粉丝点击