Linux进程间通信(四):套接字之UDP

目录

UDP特点

UDP适用情况

UDP通信流程

函数接口

socket

bind

recvfrom 

sendto     

UDP流程

服务器端

流程

代码演示

客户端

流程

代码演示
​​​​​​​


UDP特点

无连接不可靠的传输协议

  • UDP(User Datagram Protocol)用户数据报协议,是不可靠的无连接的协议。在数据发送前,因为不需要进行连接,所以可以进行高效率的数据传输。

UDP适用情况

  • 发送小尺寸数据(如对DNS服务器进行IP地址查询时)
  • 在接收到数据,给出应答较困难的网络中使用UDP。
  • 适合于广播/组播式通信中。
  • MSN/QQ/Skype等即时通讯软件的点对点文本通讯以及音视频通讯通常采用UDP协议
  • 流媒体、VOD、VoIP、IPTV等网络多媒体服务中通常采用UDP方式进行实时数据传输

UDP通信流程

函数接口

socket


      NAME:socket - create an endpoint for communication

        SYNOPSIS://头文件

       #include <sys/types.h>          /* See NOTES */

       #include <sys/socket.h>

       int socket(int domain, int type, int protocol);

           功能: 创建套接字文件

           参数:

                        domain:协议族

                                 AF_UNIX, AF_LOCAL   用于本地通信

                                 AF_INET             IPv4 Internet protocols         

                                 AF_INET6            IPv6 Internet protocols         

                        type:协议类型

                                SOCK_STREAM      TCP

                                SOCK_DGRAM     UDP  

                SOCK_RAW    原始套接字      

                        protocol:

                                一般情况下写0

                                系统默认自动帮助匹配对应协议

                                传输层:IPPROTO_TCP、IPPROTO_UDP、IPPROTO_ICMP

                                网络层:htons(ETH_P_IP|ETH_P_ARP|ETH_P_ALL)     

                返回值:

                        成功: 返回一个特殊文件描述符;

                        失败: -1 更新errno

bind


        int bind(int sockfd, const struct sockaddr *addr,socklen_t addrlen);

        功能: 绑定,将socket()返回值和IP/端口号进行绑定;

(以什么样的形式去绑定?就是填充第二个结构体,把端口号和IP填充到这个结构体中)

        参数:

                sockfd:  是socket()函数的返回值;

                const struct sockaddr *addr:

                        struct sockaddr是结构体类型,是一个通用结构体;

                            struct sockaddr {
                                                 sa_family_t sa_family;    // 2个字节typedef unsigned short int  sa_family_t;  //

                                        char        sa_data[14];  // 14字节

                                }

                                整个结构体大小为16个字节


(程序员每次填充的时候填充自己的结构体,将自己的结构体强转成通用的结构体,原因:每个协议都对应一个结构体,如果每个协议都调用一次这个函数,就会调用很多函数接口。为了做到统一,你可以定义自己的结构体。

用IPv4通信时传值需传对应结构体struct sockaddr_in是Internet的结构体,本地通信还会有本地通信所要填充的结构体sockaddr_un,每种协议都有自己需要填充的一个结构体,如果每种协议都有自己的函数接口的话,函数接口太多,没办法记忆,为了做到统一性,填充的填充自己的结构体,传值的时候传struct sockaddr,那么就需要把自己填充的sockaddr_in强制转换成struct sockaddr形式) 


struct sockaddr_in {
                                      unsigned short sin_family;  //协议IPv4,2个字节

                                      unsigned short sin_port;    //端口号 ,2个字节                                                                          struct in_addr    sin_addr;  

                                      struct in_addr {
                                                                      __be32 s_addr;//IP地址                       };

                                        /* Pad to size of `struct sockaddr'. */

                                      unsigned char     __pad[__SOCK_SIZE__ - sizeof(short int) -

                                                sizeof(unsigned short int) - sizeof(struct in_addr)];

                                                        //8个字节

                                };

                addrlen:

                        结构体的大小;

                                sizeof(serveraddr);

                返回值:0

                        -1  失败,更新errno

recvfrom 


      ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags,struct sockaddr *src_addr, socklen_t *addrlen);

        功能: 接收数据

        参数:

                sockfd: acceptfd ;

                buf  存放位置

                len  大小

                flags  一般填0,相当于read()函数

                        MSG_DONTWAIT  非阻塞

                src_addr 指向发送数据的客户端地址信息的结构体(sockaddr_in需类型转换)

                addrlen 指向from结构体长度值

        返回值:

                < 0  失败出错

                >0   成功接收的字节个数
 

sendto     

  

        ssize_t sendto(int sockfd, const void *buf, size_t len, int flags, const struct sockaddr *dest_addr, socklen_t addrlen);

        功能:发送数据

        参数:

                sockfd:socket函数的返回值

                buf:发送内容存放的地址

                len:发送内存的长度

                flags :    如果填0,相当于write()

                dest_addr:  指向接收数据的主机地址信息的结构体(sockaddr_in需类型转换)

                addrlen: to所指结构体的长度        

        返回值:

                < 0  失败出错

                >0   成功发送的字节个数

UDP流程

服务器端

流程

  1. socket(),返回一个文件描述符,用于通信
  2. bind(); 绑定
  3. recvfrom(), 接收数据
  4. sendto(),  发送数据
  5. close(sockfd);  

代码演示

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>

int main(int argc, char const *argv[])
{
    //1.创建一个数据报套接字
    int sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd < 0)
    {
        perror("socket error.");
        return -1;
    } //填充结构体
    struct sockaddr_in serveraddr, clientaddr;
    serveraddr.sin_family = AF_INET;
    serveraddr.sin_port = htons(atoi(argv[1]));
    serveraddr.sin_addr.s_addr = inet_addr("0.0.0.0");

    socklen_t len = sizeof(clientaddr);

    //2.绑定服务器端ip地址和端口
    if (bind(sockfd, (struct sockaddr *)&serveraddr, sizeof(serveraddr)) < 0)
    {
        perror("bind err.");
        return -1;
    }
    ssize_t recvbyte;
    char buf[128];
    while (1)
    {
        recvbyte = recvfrom(sockfd, buf, sizeof(buf), 0, (struct sockaddr *)&clientaddr, &len);
        if (recvbyte < 0)
        {
            perror("recvfrom err.");
            return -1;
        }
        printf("ip=%s,port=%d->client:%s\n", inet_ntoa(clientaddr.sin_addr),
               ntohs(clientaddr.sin_port), buf);
		strcat(buf,"--------"); 
		sendto(sockfd,buf,sizeof(buf),0,(struct sockaddr *)&clientaddr,sizeof(clientaddr));
    }
    close(sockfd);
    return 0;
}

客户端

流程

  1. socket(),返回一个文件描述符,用于通信
  2. bind(); 绑定
  3. recvfrom(), 接收数据
  4. sendto(),  发送数据
  5. close(sockfd);  

代码演示

#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdlib.h>

int main(int argc, char const *argv[])
{
    int sockfd=socket(AF_INET,SOCK_DGRAM,0);
    if(sockfd < 0)
    {
        perror("socket err.");
        return -1;
    }
    //填充服务器的ip和端口
    struct sockaddr_in serveraddr,tempaddr;
    serveraddr.sin_family=AF_INET;
    serveraddr.sin_port=htons(atoi(argv[2]));
    serveraddr.sin_addr.s_addr=inet_addr(argv[1]);

	socklen_t len=sizeof(tempaddr);

    char buf[128];
    ssize_t recvbyte;
    while(1)
    {
        fgets(buf,sizeof(buf),stdin);
        if(buf[strlen(buf)-1]=='\n')
          buf[strlen(buf)-1]='\0';
        sendto(sockfd,buf,sizeof(buf),0,(struct sockaddr *)&serveraddr,sizeof(serveraddr));

        recvbyte = recvfrom(sockfd, buf, sizeof(buf), 0, (struct sockaddr *)&tempaddr, &len);
        if (recvbyte < 0)
        {
            perror("recvfrom err.");
            return -1;
        }
        printf("ip=%s,port=%d->server:%s\n", inet_ntoa(tempaddr.sin_addr),
               ntohs(tempaddr.sin_port), buf);
		
    }
    close(sockfd);
    return 0;
}

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值