【网络】UDP协议的简单使用

目录

服务器

客户端

测试

code for Udp_echo_serve

Udp_dict_serve


UDP是基于socket(基于IP和port进行通信就叫做socket通信)进行网络通信的,那我们这篇博客就来介绍一下基于UDP通信的基本流程,先让服务端和客户端进行简单的跨网络通信。

服务器

首先我们需要创建UDP套接字,用到的接口是

man socket

如果要使用UDP通信,第一个参数就传

它用来表示基于网络通信,第二个参数就传

它用来表示是面向数据报,第三个参数我们就默认传0就可以了,因为前两个参数就可以确定它就是UDP协议。它的返回值我们可以认为是一个文件描述符,未来我们就可以向这个文件中去写入或读取东西

接下来是bind socket和网络信息

man 2 bind

第一个参数就是前面接口的返回值,第二个参数是输入性参数,是一个结构体的指针,所以我们要创建这样一个结构体并且填充内容,第三个参数就是第二个参数的大小

所以在绑定之前,我们需要创建struct sockaddr的对象并且填充内容:

因为socket编程是有不同种类的,而操作系统又是拿C语言写的,为了统一接口,我们就用struct sockaddr这个类型,具体到网络通信是struct sockaddr_in这个类型,所以我们创建的是struct sockaddr_in这个类型的变量,到时候进行强制类型转换成struct sockaddr类型

local的类型是一个结构体,结构体中有这些类型:

我们可以看到这里面有port,就是我们传进来的端口号,这个端口号当然也要进行网络传输,因为放到网络中的数据必须是大端,所以我们有接口用来主机序列转网络序列

有了这些接口,我们就可以任意的16位,32位,主机序列转网络序列,网络序列转主机序列。

传完了端口号,就要传ip地址了,就是sin_addr的内容,同样这个也需要主机序列转网络序列,但同时我们知道从命令行参数传进来的IP地址是字符串风格的点分十进制的格式(比如193.168.111.222,这样占15个字节),这样占的空间就大,我们知道每一位都是0~255,这用一个字节来表示整数足够了(这样只占四个字节),所以我们要将字符串风格的点分十进制转换成4字节IP。上面两个工作可以通过下面的接口全部完成

man inet_addr

填充完了我们就可以进行bind了,到此初始化的工作就完成了

下面就是不断的从网络中获取信息,我们可以设置成死循环的形式,我们先实现这样的场景,谁给我们发的,我们再发回去

UDP是面向数据报的,从网络中获取信息的接口是:

man recvfrom

第二个参数是要把从网络中获得的信息存到哪,后两个参数是输出型参数,里面就存放着这个消息是谁发来的一些信息

我们再把消息发回去的接口是:

man sendto

后两个参数就存放着要把消息发送给谁

这样,服务器大致就写好了,还有一些细节需要处理

我们运行主函数的代码时命令行参数就要写上ip和端口号

客户端

客户端要运行主函数肯定是要加上你想访问的服务器的ip地址和端口号,因为这样可以在互联网中唯一确定一个进程,客户端和服务器一样,也是要创建socket,但是不用显示bind,就是我们不要bind,客户端第一次向服务器发消息的时候操作系统会自动bind,因为如果客户端显示的bind,那就意味着不同的客户端可能用一个端口号(比如手机中的抖音和淘宝用一个端口号),这样就导致两个客户端无法同时启动,就会出现端口号冲突的问题。那么就意味着客户端的代码要比服务器要好写

测试

我们像下面这样运行服务器

它是可以成功的,127.0.0.1这个IP我们称为本地环回,它不将服务发送到网络中,可以用于本地通信,常用于代码测试,于是此时我们再起一个SSH渠道就可以实现本地的服务器和客户端通信了

这是本地环回,毕竟不是跨网络的,我们把IP换成服务器的公网IP试一试

可以看到它会报错,因为我们的服务器,无法直接bind公网IP(云服务器不允许),我们也严重不推荐绑定任意一个确定的IP(虚拟机可以)

因为一台设备可能有多个IP地址,我向每个IP地址发,进程都应该收到,如果bind特定的IP地址,那么只有这个IP地址可以收到,所以我们一般传0.0.0.0,简写成0,表示任意IP地址bind,所以之前的代码中还是不要传IP了,因为确定是0

这样,客户端给本地环回和公网IP就都可以发消息了

网络命令

还有一些网络命令我们补充一下:

ping 用来检测网络连通性,加上-c3(count=3)表示检测三次

netstat -puan 用来查udp网络服务

p代表process进程,u代表udp,a代表all,n代表number,就是能显示成数字的都显示成数字

同样,我们要查tcp网络服务是netstat -tlnp

t就表示tcp,l表示状态是listen的

watch -n 1 ls表示每个一秒执行一次ls命令,我们就可以用它来查网络服务

我们之前要查一个进程的pid要这样:

ps ajx | head -1 && ps ajx | grep server非常繁琐,于是我们可以用

pidof server就可以直接查到进程的pid

我们如果想通过pid杀掉进程就可以

pidof server | xargs kill -9

这里的xargs就是把管道中的东西放到-9后面

code for Udp_echo_serve

//UdpServer.hpp

#include <iostream>
#include <string>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <cerrno>
#include<cstdlib>
#include <cstring>
#include "Log.hpp"

enum error
{
    SOCKET_ERR = 1,
    BIND_ERR,
    USAGE_ERR,

};
class UdpServer
{
public:
    UdpServer(uint16_t port) : _port(port), _socketfd(-1)
    {
    }
    void InitServer()
    {
        // 1.创建UDP socket套接字
        _socketfd = socket(AF_INET, SOCK_DGRAM, 0);
        if (_socketfd < 0)
        {
            LOG(FATAL, "socket error,%d,%s", errno, strerror(errno));
            exit(SOCKET_ERR);
        }
        LOG(INFO, "create socket success,socket:%d", _socketfd);

        // 填充struct sockaddr_in结构
        struct sockaddr_in local;
        bzero(&local, sizeof(local)); // 对空间进行清0
        local.sin_family = AF_INET;
        local.sin_port = htons(_port);
        local.sin_addr.s_addr=INADDR_ANY;//这个数就是0,因为是0,大小端也就没意义了
        // local.sin_addr.s_addr = inet_addr(_ip.c_str());

        int n = bind(_socketfd, (struct sockaddr *)&local, sizeof(local));
        if (n < 0)
        {
            LOG(FATAL, "bind error,%d,%s", errno, strerror(errno));
            exit(BIND_ERR);
        }
        LOG(INFO, "bind success");
    }
    void start()
    {
        while (true)
        {
            char buffer[1024];
            struct sockaddr_in peer;
            socklen_t len=sizeof(peer);
            ssize_t n=recvfrom(_socketfd,buffer,sizeof(buffer)-1,0,(struct sockaddr*)&peer,&len);
            if(n>0)
            {
                buffer[n]=0;
                LOG(DEBUG,"get message from[%s,%d]:%s",inet_ntoa(peer.sin_addr),ntohs(peer.sin_port),buffer);//这样解析就知道是谁发过来的
                sendto(_socketfd,buffer,strlen(buffer),0,(struct sockaddr*)&peer,len);
            }
        }
    }

private:
    int _socketfd;
    // std::string _ip;
    uint16_t _port;
};
//Main.cc

#include<iostream>
#include"UdpServer.hpp"
void Usage(char* arg)
{
    std::cout<<"Usage:  please enter:"<<std::endl;
    std::cout<<arg<<' '<<"server port"<<std::endl;
}
int main(int argc,char*argv[])
{
    if(argc!=2)
    {
        Usage(argv[0]);
        exit(USAGE_ERR);
    }
    Enablescreen();
    uint16_t port=std::stoi(argv[1]);
    UdpServer udps(port);
    udps.InitServer();
    udps.start();


    return 0;
}
//UdpClient.cc

#include <iostream>
#include <string>
#include <cstring>
#include <cstdlib>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
void Usage(char *arg)
{
    std::cout << "Usage:  please enter:" << std::endl;
    std::cout << arg << ' ' << "server ip " << "server port" << std::endl;
}
int main(int argc, char *argv[])
{
    if (argc != 3)
    {
        Usage(argv[0]);
        exit(1);
    }
    std::string serverip = argv[1];
    uint16_t serverport = std::stoi(argv[2]);
    //创建socket套接字
    int socketfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (socketfd < 0)
    {
        std::cerr << "socket error" << std::endl;
    }
    struct sockaddr_in server;
    //填充服务器信息,因为要给服务器发消息
    memset(&server, 0, sizeof(server));
    server.sin_family=AF_INET;
    server.sin_port=htons(serverport);
    server.sin_addr.s_addr=inet_addr(serverip.c_str());

    std::string message;
    while(true)
    {
        std::cout<<"Please Enter# ";
        std::getline(std::cin,message);
        sendto(socketfd,message.c_str(),message.size(),0,(struct sockaddr*)&server,sizeof(server));//给服务器发消息
        struct sockaddr_in peer;
        socklen_t len=sizeof(peer);
        char buffer[1024];
        ssize_t n=recvfrom(socketfd,buffer,sizeof(buffer)-1,0,(struct sockaddr*)&peer,&len);//接收来自服务器的消息
        if(n>0)
        {
            buffer[n]=0;
            std::cout<<"server echo# "<<buffer<<std::endl;
        }
    }

    return 0;
}

Udp_dict_serve

上面我们已经会了实现服务器和客户端之间的一个基本跨网络通信,下面我们可以给服务器部署一些任务,就比如查字典的服务:客户端通过发送一些英文单词给服务器,然后从服务器中获得汉语翻译,于是我们就在上面的代码上简单写一个Dict.hpp来实现这个功能

#include <iostream>
#include <string>
#include <unordered_map>
#include "Log.hpp"

const std::string defaultconfpath = "./Dict.txt";
const std::string seq = ": ";

class Dict_serve
{
private:
    bool Load()
    {
        std::ifstream in(_conf_filepath);
        if (!in.is_open())
        {
            LOG(FATAL, "open %s error", _conf_filepath.c_str());
            return false;
        }
        std::string line;
        while (std::getline(in, line))
        {
            if (line.empty())
                continue;
            auto pos = line.find(seq);
            if (pos == std::string::npos)
                continue;
            std::string word = line.substr(0, pos);
            if (word.empty())
                continue;
            std::string han = line.substr(pos + seq.size());
            if (han.empty())
                continue;
            LOG(DEBUG, "load %s %s success", word.c_str(), han.c_str());
            _dict.insert({word, han});
        }
        in.close();
        LOG(DEBUG, "load %s success", _conf_filepath.c_str());
        return true;
    }

public:
    Dict_serve(const std::string &path = defaultconfpath) : _conf_filepath(path)
    {
        Load();
    }
    std::string translate(const std::string &word)
    {
        auto iter = _dict.find(word);
        if (iter == _dict.end())
            return "nofind(未找到)";
        // return _dict[word];
        return iter->second;
    }

private:
    std::unordered_map<std::string, std::string> _dict;
    std::string _conf_filepath;
};

所有的服务器本质还是解决的是输入输出的问题,我们不想让网络通信模块和翻译(业务)模块强耦合,所以可以这样做:

把一个通用的任务当作udpserver的成员变量,以后就通过这个变量做任务

所以我们在构建服务器对象时这样去构建:

  • 13
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值