用UDP套接字实现客户端和服务端通信

IP地址和port端口号

IP地址

数据有IP(公网)标识一台唯一的主机。

port端口号

为了更好的标识一台主机上服务进程的唯一性,我们采用端口号port,标识服务器进程,客户端进程的唯一性!

ip+端口号

IP地址(主机全网唯一性) + 该主机上的端口号,标识该服务器上进程的唯一性
ipA + portA, ipB + portB

网络通信的本质

概述

相当于一个服务端进程 与一个客户端进程 通过网络资源来进行通信

本质

1. 需要让不同的进程,先看到同一份资源 -- 网络

2. 通信不就是在做IO吗? -- 所以,我们所有的上网行为,无外乎两种: 
   a. 我要把我的数据发出去。
   b. 我要收到别人给我发的数据。

为什么不用PID做端口号?

a. 系统是系统,网络是网络,单独设置 -- 系统与网络解耦
b. 需要客户端每次都能找到服务器进程 -- 服务器的唯一性不能做任何改变 -- IP+port 不能随便改变 <-- 不能使用轻易会改变的值(pid每次进程启动会改变)
c. 不是所有的进程都要提供网络服务请求,但是所有的进程都需要pid

认识TCP协议

1. 传输层协议
2. 有连接
3. 可靠传输
4. 面向字节流

认识UDP协议

1. 传输层协议
2. 无连接
3. 不可靠传输
4. 面向数据报

TCP与UDP的可靠与不可靠

概述

可靠与不可靠是中性词。

可靠是有成本的。往往是比较复杂的,维护成本和编码成本比较高。

不可靠比较简单,维护成本和编码成本比较低。

应对应不同的场景

TCP:银行系统、搜索引擎等;
UDP:推送广告等;

网络字节序中的大小端问题

概念

由于在同一台主机上的数据一般不是大端就是小端,我们不需要特别的注意。但是当两台主机通过网络发送和接受数据的时候,这个问题就要考虑起来,因此我们规定网络中的数据都是大端!当前发送主机是小端,就需要先将数据转成大端,否则就忽略,直接发送即可。

大小端转换的接口

主机序列转网络序列:

uint32_t htonl(unit32_t hostlong);
uint16_t htons(uint16_t hostshort);

网络序列转主机序列:

uint32_t ntohl(uint32_t netlong);
uint16_t ntohs(uint16_t netshort);

认识socket套接字

概念

ip+port就是套接字

分类

1.  网络套接字
2.  原始套接字
3.  unix域间套接字 (只能本地通信)

本地通信和网络通信的判断

通过 struct sockaddr{} 
这一套结构体可以分别表示网络通信和本地通信。

根据第一个参数传入的是AF_INET还是AF_UNIX来内部判断

UDP的通信

socket接口

#include <sys/types.h>
#include <sys/socket.h>

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

 参数

domain

AF_UNIX: 域间通信(本地通信)
AF_INET: 网络通信 (也可以是PF_INET)


type:套接字提供能力类型

SOCK_DGRAM: 数据报套接字 -- (对应的UDP)
SOCK_STREAM: 流式套接字 -- (对应的TCP)

protocol:

未来我们想使用什么协议(指明是TCP协议还是UDP协议)
可以缺省为0 (前两个参数确定之后就可以确定是使用TCP还是UDP了不需要再传入第三个参数了)

返回值

文件描述符 

bind接口

#include <sys/types.h>
#include <sys/socket.h>

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

意义

 让OS知道我们的IP地址和端口号

参数

 sockfd:

创建套接字的返回值

const struct sockaddr *addr:

含有 IP地址和端口号属性的结构体,我们通过该结构体将这两个属性传入进去

addrlen:

传入结构体的长度

返回值

绑定成功返回0
绑定失败返回-1

结构体 sockaddr_in

port

uint16_t sin_port;

ip

struct in_addr sin_addr;

IP地址的表达形式

“124.223.97.182” -- > 点分十进制风格的IP,字符串,可读性好

uint32_t ip:整数风格的IP - 网络通信使用

uint32_t ip=12345;

struct _ip
{
   unsigned char p1;
   unsigned char p2;
   unsigned char p3;
   unsigned char p4;
};

str_ip=
to_string(((struct _ip)&ip)->p1)+"."
to_string(((struct _ip)&ip)->p2)+"."
to_string(((struct _ip)&ip)->p3)+"."
to_string(((struct _ip)&ip)->p4);


系统接口可以直接对点分十进制和uint32_t类型的IP地址进行互相转换。

bzero接口

#include<cstrings>

void bzero(void *s, size_t n);

作用:初始化结构体类型的对象。


inet_addr接口 (用于ip地址)

#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

in_addr_t inet_addr(const char *cp);

内部完成两件事情

1. string -> uint32_t
2. htonl();

服务器ip的设置

一般情况下我们的服务器不指明一个特定的IP,一般会设置成INADDR_ANY。

任意地址bind,这样的话就不会错过其他不同的IP但是端口号与我们的服务器匹配的网络。 

struct sockaddr_in local; // 定义一个sockaddr_in的结构体
local.sin_addr.s_addr = INADDR_ANY; // 不指名特定的一个IP

 recvfrom接口

#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);

参数

flags

设置为0  为阻塞式读取

struct sockaddr *src_addr 

所传的结构体

socklen_t *addrlen

所传的结构体长度

返回值

读取成功返回读取到的字符数。
读取失败返回-1。

inet_ntoa 接口

#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

char *inet_ntoa(struct in_addr in);

作用

1. 将网络序列转换为主机序列 
2. 并且将int转为点分十进制

为什么客户端不用显示的bind?

因为写服务器的是一家公司,但是写客户端的是无数家公司。如果我们在客户端上绑定了特定的端口号,如果刚好这个端口号被别人占用了,那么就无法与服务器正常通信了。因此服务端是一定要我们手动去bind特定的端口号的,为了服务端端口号的唯一。客户端的端口号我们让OS自动形成就可以了。

sendto接口

#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);

参数

const void *buf

要发送的缓冲区

size_t len

缓冲区长度

UDP通信的代码

简单的通信

客户端发数据,服务端接收数据。

服务端代码

创建套接字

// 1. 套接字的创建
_sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if (_sockfd == -1)
{
    cerr << "socket error :" << errno << " : " << strerror(errno) << endl;
    exit(SOCKET_ERR);
}
cout << "socket success : " << _sockfd << endl;

调用bind

// 2. bind
struct sockaddr_in local;
bzero(&local, sizeof(local)); // 初始化local

local.sin_family = AF_INET;

// 1. 主机转网络 点分十进制转int
local.sin_addr.s_addr = inet_addr(_serverip.c_str());
// local.sin_addr.s_addr = INADDR_ANY;
local.sin_port = htons(_serverport);
int n = bind(_sockfd, (struct sockaddr *)&local, sizeof(local));

if (n == -1)
{
    cerr << "bind error: " << errno << " : " << strerror(errno) << endl;
    exit(BIND_ERR);
}

调用recvfrom接受客户端的数据

char buffer[NUM];
for (;;)
{
    struct sockaddr_in peer;
    socklen_t len = sizeof(peer);
    //
    ssize_t s = recvfrom(_sockfd, buffer, sizeof(buffer) - 1, 0, (struct sockaddr *)&peer, &len);

    if (s > 0)
    {
        // 读取数据
        buffer[s] = 0;
        uint16_t clientport = ntohs(peer.sin_port);
        string clientip = inet_ntoa(peer.sin_addr);
        string message = buffer;

        cout << clientip << " [" << clientport << "]# " << message << endl;
    }
}

客户端代码

创建套接字


// 1. 创建socket
_sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if (_sockfd == -1)
{
    cerr << "socket error: " << errno << " : " << strerror(errno) << endl;
    exit(2);
}
cout << "socket success: " << _sockfd << endl;

调用sendto接口让信息发给服务端

struct sockaddr_in server;
memset(&server, 0, sizeof(server));
server.sin_family = AF_INET;
server.sin_addr.s_addr = inet_addr(_serverip.c_str());
server.sin_port = htons(_serverport);

string message;
while (!_quit)
{
    cout << "Please Enter# ";
    cin >> message;

    sendto(_sockfd, message.c_str(), message.size(), 0, (struct sockaddr *)&server, sizeof(server));
}

通信界面

完整代码  

makefile

cc=g++

.PHONY:all
all:udpClient udpServer

udpClient:udpClient.cc
	$(cc) -o $@ $^ -lpthread -std=c++11
udpServer:udpServer.cc
	$(cc) -o $@ $^ -std=c++11

.PHONY:clean
clean:
	rm -f udpClient udpServer

udpClient.hpp

#pragma once

#include <iostream>
#include <string>
#include <cerrno>
#include <cstring>
#include <cstdlib>
#include <strings.h>

#include <netinet/in.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>

namespace Client
{
    using namespace std;

    class udpClient
    {
    public:
        udpClient(const string &serverip, const uint16_t &serverport)
            : _serverip(serverip), _serverport(serverport), _quit(false), _sockfd(-1)
        {
        }
        void initClient()
        {
            // 1. 创建socket
            _sockfd = socket(AF_INET, SOCK_DGRAM, 0);
            if (_sockfd == -1)
            {
                cerr << "socket error: " << errno << " : " << strerror(errno) << endl;
                exit(2);
            }
            cout << "socket success: " << _sockfd << endl;

            // 2. client要不要bind[不需要的]  , client要不要显示的bind,需不需要程序员自己bind? 不需要
            // 写服务器的一家公司,写客户端是无数家公司 -- 因此让OS自动形成端口进行bind! -- OS在什么时候,如何bind
        }
        void run()
        {
            struct sockaddr_in server;
            memset(&server, 0, sizeof(server));
            server.sin_family = AF_INET;
            server.sin_addr.s_addr = inet_addr(_serverip.c_str());
            server.sin_port = htons(_serverport);

            string message;
            while (!_quit)
            {
                cout << "Please Enter# ";
                cin >> message;

                sendto(_sockfd, message.c_str(), message.size(), 0, (struct sockaddr *)&server, sizeof(server));
            }
        }
        ~udpClient()
        {
        }

    private:
        int _sockfd;
        string _serverip;
        uint16_t _serverport;
        bool _quit;
    };
} // namespace Client

udpClient.cc

#include "udpClient.hpp"
#include <memory>

using namespace Client;

static void Usage(string proc)
{
    cout << "\nUsage:\n\t" << proc << " local_ip local_port\n\n";
}

// ./udpClient server_ip server_port
int main(int argc, char *argv[])
{
    if (argc != 3)
    {
        Usage(argv[0]);
        exit(1);
    }

    string serverip = argv[1];
    uint16_t serverport = atoi(argv[2]); // 字符串转整数

    unique_ptr<udpClient> ucli(new udpClient(serverip, serverport));

    ucli->initClient();
    ucli->run();

    return 0;
}

udpServer.hpp

#include <iostream>
#include <string>
#include <cerrno>
#include <cstring>
#include <cstdlib>
#include <strings.h>
#include <functional>

#include <netinet/in.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>

using namespace std;

namespace Server
{
    enum
    {
        SOCKET_ERR = 1,
        BIND_ERR,
        OPEN_ERR
    };

    const int NUM = 1024;
    static const string defaultIp = "0.0.0.0";

    class udpServer
    {

    public:
        udpServer(const uint16_t &port, const string ip = defaultIp)
            : _serverport(port), _serverip(defaultIp), _sockfd(-1)
        {
            // cout << "拷贝构造" << endl;
        }

        void initServer()
        {
            // 1. 套接字的创建
            _sockfd = socket(AF_INET, SOCK_DGRAM, 0);
            if (_sockfd == -1)
            {
                cerr << "socket error :" << errno << " : " << strerror(errno) << endl;
                exit(SOCKET_ERR);
            }
            cout << "socket success : " << _sockfd << endl;

            // 2. bind
            struct sockaddr_in local;
            bzero(&local, sizeof(local)); // 初始化local

            local.sin_family = AF_INET;

            // 1. 主机转网络 点分十进制转int
            local.sin_addr.s_addr = inet_addr(_serverip.c_str());
            // local.sin_addr.s_addr = INADDR_ANY;
            local.sin_port = htons(_serverport);
            int n = bind(_sockfd, (struct sockaddr *)&local, sizeof(local));

            if (n == -1)
            {
                cerr << "bind error: " << errno << " : " << strerror(errno) << endl;
                exit(BIND_ERR);
            }
        }
        void start()
        {
            char buffer[NUM];
            for (;;)
            {
                struct sockaddr_in peer;
                socklen_t len = sizeof(peer);
                //
                ssize_t s = recvfrom(_sockfd, buffer, sizeof(buffer) - 1, 0, (struct sockaddr *)&peer, &len);

                if (s > 0)
                {
                    // 读取数据
                    buffer[s] = 0;
                    uint16_t clientport = ntohs(peer.sin_port);
                    string clientip = inet_ntoa(peer.sin_addr);
                    string message = buffer;

                    cout << clientip << " [" << clientport << "]# " << message << endl;
                }
            }
        }

        ~udpServer()
        {
        }

    private:
        int _sockfd;
        uint16_t _serverport;
        string _serverip;
    };
} // udpServer

udpServer.cc

#include "udpServer.hpp"
#include <memory>

using namespace Server;
using namespace std;

static void Usage(string proc)
{
    cout << "\nUsage:\n\t" << proc << " local_port\n\n";
}

// ./udpClient server_ip server_port
int main(int argc, char *argv[])
{
    if (argc != 2)
    {
        Usage(argv[0]);
        exit(1);
    }

    uint16_t port = atoi(argv[1]); // 字符串转整数

    unique_ptr<udpServer> usvr(new udpServer(port));

    usvr->initServer();

    usvr->start();

    return 0;
}








本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/32852.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Redis7【⑦ Redis哨兵(sentinel)】

Redis哨兵 Redis Sentinel&#xff08;哨兵&#xff09;是 Redis 的高可用性解决方案之一&#xff0c;它可以用于监控和管理 Redis 主从复制集群&#xff0c;并在主节点发生故障时自动将从节点升级为新的主节点&#xff0c;从而保证系统的高可用性和可靠性。 Redis Sentinel …

Flex写法系列-Flex布局之基本语法

以前的传统布局&#xff0c;依赖盒装模型。即 display position float 属性。但是对于比较特殊的布局就不太容易实现&#xff0c;例如&#xff1a;垂直居中。下面主要介绍flex的基本语法。 一、什么是Flex布局&#xff1f; Flex布局个人理解为弹性盒子&#xff0c;为盒装模型…

【机器学习】——神经网络与深度学习

目录 引入 一、神经网络及其主要算法 1、前馈神经网络 2、感知器 3、三层前馈网络&#xff08;多层感知器MLP&#xff09; 4、反向传播算法 二、深度学习 1、自编码算法AutorEncoder 2、自组织编码深度网络 ①栈式AutorEncoder自动编码器 ②Sparse Coding稀疏编码 …

opencv编译

文章目录 一、编译前工作二、编译安装1、Windows2、Linux 一、编译前工作 进入下载页面https://github.com/opencv/opencv&#xff0c;下载指定.tar.gz源码包&#xff0c;例如&#xff1a;opencv-4.7.0.tar.gz。解压到指定目录。 二、编译安装 opencv构建时&#xff0c;需要…

chatgpt赋能python:使用Python连接网络摄像头

使用Python连接网络摄像头 网络摄像头是现代生活中不可或缺的设备之一&#xff0c;其允许用户在远程位置查看实时视频流。Python语言提供了强大的工具来连接和控制网络摄像头。本文将向您展示如何使用Python连接网络摄像头以及如何将视频流数据流式传输到本地计算机。 环境设…

OpenGL 鼠标拾取模型

1.简介 在我们的场景中&#xff0c;使用鼠标光标点击或“挑选”一个3d对象是很有用的。一种方法是从鼠标投射3d光线&#xff0c;通过相机&#xff0c;进入场景&#xff0c;然后检查光线是否与任何物体相交。这通常被称为光线投射。 我们不是从局部空间中的网格开始&#xff0c…

vscode 出现 No such file or directory 的解决办法(python tkinter)

问题 主要解决的问题是python在linux下包没办法安装的问题 Traceback (most recent call last): File “e:\Github\Python-GUI\PyQt-Fluent-Widgets\examples\navigation\demo.py”, line 202, in w Window() File “e:\Github\Python-GUI\PyQt-Fluent-Widgets\examples\na…

Python学习—装饰器的力量

Python学习—装饰器的力量 作为许多语言都存在的高级语法之一&#xff0c;装饰器是你必须掌握的知识点。 Python的装饰器&#xff08;Decorator&#xff09;允许你扩展和修改可调用对象&#xff08;函数、方法和类&#xff09;的行为&#xff0c;而无需永久修改可调用的对象本身…

leedcode-只出现一次的数字-异或

题目 题目 代码 class Solution { public:int singleNumber(vector<int>& nums) {int ansnums[0];for(int i1;i<nums.size();i){ansans^nums[i];}return ans;} };

C++ - 哈希的应用

前面的文章中我们讲解了如何进行哈希表的构建以及使用实现的哈希表来模拟实现unordered_map&#xff0c;在本文中我们将继续来讲解一下哈希的应用。 位图 问题引入 首先我们来引入一个问题&#xff1a;给40亿个不重复的无符号整数&#xff0c;没排过序。给一个无符号整数&am…

介绍 9 个研发质量度量指标

研发质量管理中的 MTTR、MTBF、MTTF、MTTD 都是什么&#xff1f;今天我们从生产事件的全生命周期出发&#xff0c;认识研发质量管理的 9 个度量指标——「MT 家族」。 01 Mean Time To ALL 「MT」是 Mean Time 的缩写&#xff0c;意为平均时间&#xff0c;「MT 家族」则是 Li…

【AcWing算法基础课】第一章 基础算法(部分待更)

文章目录 前言课前温习一、快速排序核心模板1.1题目描述1.2思路分析1.3代码实现 二、归并排序核心模板2.1题目描述2.2思路分析2.3代码实现 三、二分查找整数二分题目一3.1题目描述3.2思路分析3.3代码实现 浮点数二分题目二3.1题目描述3.2思路分析3.3代码实现 四、高精度加法核心…

记录--巧用 overflow-scroll 实现丝滑轮播图

这里给大家分享我在网上总结出来的一些知识&#xff0c;希望对大家有所帮助 前言: 近期我在项目中就接到了一个完成轮播图组件的需求。最开始我也像大家一样&#xff0c;直接选择使用了知名的开源项目 "Swiper"&#xff0c;但是后来发现它在移动端项目中某些测试环境…

函数调用的机器级表示

文章目录 1.Call和ret指令2. 如何访问栈帧里面的数据为什么栈底放在上面&#xff0c;栈顶放在下面X86中的寄存器EBP、ESP寄存器push 、pop 指令mov 指令总结如何访问栈帧 3. 如何切换栈帧函数调用时函数返回时 4. 完整的函数调用过程1. 一个函数的栈帧内包含哪些内容2. 汇编代码…

Jenkins 发送文件到远程服务器:Publish Over SSH 插件

Jenkins 发送文件到远程服务器&#xff1a;Publish Over SSH 插件 文章目录 Jenkins 发送文件到远程服务器&#xff1a;Publish Over SSH 插件一、Publish Over SSH 插件1、概述2、主要功能和特点3、插件主页4、安装 Publish Over SSH 插件5、配置远程主机 二、发送文件到远程主…

windows安装python开发工具pycharm

下载地址 PyCharm: the Python IDE for Professional Developers by JetBrains 点击下载 安装 双击exe安装等待安装完成即可 设置python环境 添加本地python环境 选择python.exe 所在路径即可&#xff0c;2.x版本和3.x版本都可&#xff0c;根据需要进行调整

【Spring】——Spring生命周期

前言 ❤️❤️❤️Spring专栏更新中&#xff0c;各位大佬觉得写得不错&#xff0c;支持一下&#xff0c;感谢了&#xff01;❤️❤️❤️ Spring_冷兮雪的博客-CSDN博客 前面我们讲完了Spring中有关Bean的读和取&#xff0c;我们还没有好好去了解了解Bean对象&#xff0c;这篇 …

基于appnium+python+夜神模拟器的自动化

目录 1、安装夜神模拟器 2、定位元素 3、开始编码 首先搭好appnium环境&#xff01;参考https://www.cnblogs.com/testlearn/p/11419797.html 1、安装夜神模拟器 下载安装夜神模拟器后&#xff0c;在cmd命令输入adb connect 127.0.0.1:62001&#xff0c;显示出设备则表示…

redis协议与异步方式学习笔记

目录 1 交互方式 pipline2 广播机制2.1 概念演示2.2 使用场景 3 redis事物3.1 概念3.2 使用场景3.3 解决的问题3.3.1 背景&#xff1a;多线程竞争出现问题3.3.2 事务3.3.3 安全性事务 3.4两种类型的“事务”3.4.1 watch ... multi exec3.4.2 lua 脚本实现“原子”执行&#xff…

再以汇编代码分析c++的右值引用

汇编分析c语言的执行结果最为准确。 可见&#xff0c;右值引用其实还是引用&#xff0c; bb 和 cc 都是对 aa 的引用&#xff0c;其内存里存储了 aa 的地址。 而且还有一个很奇特的现象&#xff0c;bb无法给cc赋值&#xff0c;右值引用无法给右值赋值。 同样是调用std:: move…
最新文章