C/S架构学习之UDP服务器

  • UDP服务器的实现流程:
  • 一、创建用户数据报套接字(socket函数):
  • 通信域选择IPV4网络协议、套接字类型选择数据报式
c 复制代码
	int sockfd = socket(AF_INET,SOCK_DGRAM,0); 
  • 二、填充服务器的网络信息结构体:
  • 1.定义网络信息结构体变量;
  • 2.求出结构体变量的内存空间大小;
  • 3.结构体清零;
  • 4.使用IPV4网络协议;
  • 5.预留给在终端输入的网络字节序的端口号;
  • 6.预留给在终端输入的IP地址;
c 复制代码
	struct sockaddr_in serveraddr; //定义网络信息结构体变量
    socklen_t serveraddrlen = sizeof(serveraddr);//求出结构体变量的内存空间大小

    memset(&serveraddr,0,serveraddrlen); //结构体清零

    serveraddr.sin_family = AF_INET;  //使用IPV4网络协议
    serveraddr.sin_addr.s_addr = inet_addr(argv[1]); //网络字节序的端口号
    serveraddr.sin_port = htons(atoi(argv[2])); //IP地址
  • 三、套接字和服务器的网络信息结构体进行绑定(bind函数):
c 复制代码
	int ret = bind(sockfd,(struct sockaddr *)&serveraddr,serveraddrlen);
  • 四、接收来自客户端的数据(recvfrom函数)和给客户端发送应答消息(sendto函数):
c 复制代码
	
	int ret1 = recvfrom(sockfd,buf,sizeof(buf),0,(struct sockaddr *)&clientaddr,&clientaddrlen);
	
	strcat(buf,"------------k");
	int ret2 =sendto(sockfd,buf,sizeof(buf),0,(struct sockaddr *)&clientaddr,clientaddrlen);
  • 五、关闭套接字(close函数):
c 复制代码
	close(sockfd);
  • 综合应用实例代码如下所示:
c 复制代码
//UDP服务器

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

#define ERRLOG(msg) do{\
                        printf("%s:%s:%d\n", __FILE__, __func__, __LINE__);\
                        perror(msg);\
                        exit(-1);\
                }while(0)

int main(int argc, const char *argv[])
{
        //入参合理性检查
        if(3 != argc){
                printf("Usage : %s <IP> <port>\n", argv[0]);
                exit(-1);
        }

        //创建用户数据报套接字
        int sockfd = socket(AF_INET,SOCK_DGRAM,0);
        if(-1 == sockfd)
        {
            ERRLOG("socket error");
        }
        //填充服务器网络信息结构体
        struct sockaddr_in serveraddr;
        socklen_t serveraddrlen = sizeof(serveraddr);

        memset(&serveraddr,0,serveraddrlen);
        
        serveraddr.sin_family = AF_INET;

        serveraddr.sin_addr.s_addr = inet_addr(argv[1]);
        serveraddr.sin_port = htons(atoi(argv[2]));


        //将套接字与服务器网络信息结构体绑定
        if(-1 == bind(sockfd,(struct sockaddr *)&serveraddr,serveraddrlen))
        {
            ERRLOG("bind error");
        }

        //定义新的结构体,保存来自客户端信息
       
        struct sockaddr_in clientaddr;
        socklen_t clientaddrlen = sizeof(clientaddr);
        memset(&clientaddr,0,clientaddrlen);
        



        //收发数据
        char buf[128] = {0};
       
        while(1)
        {
            
            memset(buf,0,sizeof(buf));
            if(-1 == recvfrom(sockfd,buf,sizeof(buf),0,(struct sockaddr *)&clientaddr,&clientaddrlen))
            {
                ERRLOG("recvfrom error");
            }
            printf("客户端[%s:%d]发来数据[%s]\n",inet_ntoa(clientaddr.sin_addr),ntohs(clientaddr.sin_port),buf);

            //组装应答消息
            strcat(buf,"------------k");
            if(-1 == sendto(sockfd,buf,sizeof(buf),0,(struct sockaddr *)&clientaddr,clientaddrlen))
            {
                ERRLOG("sendto error");
            }
        }
        //关闭套接字
        close(sockfd);

        return 0;
}
  • 本示例代码,仅供参考;
相关推荐
FOLLOW ME3111 分钟前
MySQL集群高可用架构
数据库·mysql·架构
TG_yunshuguoji1 分钟前
阿里云国际代理:稳定、高效、便捷的数据库服务-云数据库RDS
运维·服务器·数据库·阿里云·云计算
梁小憨憨12 分钟前
PyCharm 连接 AutoDL 远程服务器
服务器·ide·pycharm
一个帅气昵称啊18 分钟前
C#,RabbitMQ从入门到精通,.NET8.0(路由/分布式/主题/消费重复问题 /延迟队列和死信队列/消息持久化 )/RabbitMQ集群模式
分布式·微服务·架构·rabbitmq·.net
歪歪10037 分钟前
Redux和MobX在React Native状态管理中的优缺点对比
前端·javascript·react native·react.js·架构·前端框架
Insist75337 分钟前
OpenEuler安装gitlab,部署gitlab-runner
linux·运维·服务器
月夕·花晨1 小时前
Gateway -网关
java·服务器·分布式·后端·spring cloud·微服务·gateway
失散131 小时前
分布式专题——6 Redis缓存设计与性能优化
java·redis·分布式·缓存·架构
虫无涯1 小时前
LangChain中的Prompt模板如何使用?
服务器·langchain·prompt
草莓熊Lotso3 小时前
《算法闯关指南:优选算法-双指针》--01移动零,02复写零
c语言·c++·经验分享·算法·leetcode