C++实现WebSocket通信(服务端和客户端)

news2024/11/28 2:35:48

天行健,君子以自强不息;地势坤,君子以厚德载物。


每个人都有惰性,但不断学习是好好生活的根本,共勉!


文章均为学习整理笔记,分享记录为主,如有错误请指正,共同学习进步。

文章目录

  • 一、简介
  • 二、前言
  • 三、服务端代码
  • 四、客户端代码
  • 五、通信测试
    • 1. 客户端编译运行
    • 2. 服务端编译运行
    • 3. 发送消息


一、简介

这里单纯是个人总结,如需更官方更准确的websocket介绍可百度

  • websocket是一种即时通讯协议,你可以把他想像成http的变种
  • http是一次请求一次回应,而websocket是一次请求建立之后就可以持续互通
  • 业务需求诸如持续实时的消息收发场景时使用

二、前言

因为并没有C++语言基础,所以只是参考网上的一些文章来整合代码实现自己需要的功能
如果你和我一样是没有语言基础但有需要使用这个C++的websocket通信功能,那就需要安装C++的环境才能跑这个代码
C++环境配置可以参考:C++环境配置(下载安装MinGW)
使用VSCode工具
然后就是将代码编译并运行即可

三、服务端代码

websocket_server.cpp

#include <stdio.h>  
#include <winsock2.h>  
 
#pragma comment(lib,"ws2_32.lib")  
// 解决中文乱码问题
// #pragma execution_character_set("gbk");
#pragma execution_character_set("utf-8");
 
int main(int argc, char* argv[])  
{  
    //一、初始化WSA  
    WORD sockVersion = MAKEWORD(2,2);

    WSADATA wsaData;

    if(WSAStartup(sockVersion, &wsaData)!=0)  
    {  
        //终止程序
        return 0;  
    }  
 
    //二、创建套接字  socket(参数1:协议族,参数2:socket类型,参数3:协议类型)
    //AF_INET指IPv4 Internet协议
    //SOCK_STREAM指TCP连接,提供序列化的可靠的,双向连接的字节流,支持带外数据传输
    //IPPROTO_TCP指TCP协议
    SOCKET sockListen = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);  

    if(sockListen == INVALID_SOCKET)  
    {  
        printf("socket error !");  
        return 0;  
    }  
      
    //三、通信协议地址赋值
    //声明通信协议地址参数
    sockaddr_in sin;  
    //定义地址族 AF_INET指IPv4 Internet协议
    sin.sin_family = AF_INET;  
    //定义端口号 16位TCP/UDP端口号
    sin.sin_port = htons(8888);  
    // sin.sin_endpoint = htons(8888);
    //定义ip地址 32位IP地址
    // sin.sin_addr.S_un.S_addr = INADDR_ANY;   
    sin.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");   
    
    //四、绑定 //绑定IP和端口
    //bing()的三个参数:
    //参数1:创建的socket,声明方式SOCKET socket
    //参数2:通信协议地址,声明方式const struct sockaddr_in * addr
    //参数3:对应协议地址的长度,声明方式socklen_t addrlen
    if(bind(sockListen, (LPSOCKADDR)&sin, sizeof(sin)) == SOCKET_ERROR)  
    {  
        printf("bind error !");  
        return 0;
    }  
 
    //五、开始监听  
    //listen()的两个参数:
    //参数1:监听的socket描述字,声明方式SOCKET socket
    //参数2:相应socket可排队的最大连接数,声明方式int backlog
    if(listen(sockListen, 5) == SOCKET_ERROR)  
    {  
        printf("listen error !");  
        return 0;  
    }  
    

    //定义监听socket  
    SOCKET sockClient;  
    //定义协议链接地址
    sockaddr_in remoteAddr;  
    //地址大小
    int addrlen = sizeof(remoteAddr);  
    //定义接收数据载体大小
    char revData[255];

    printf("\nwait connect ...\n");
    while (true)
    {  
        //六、接收请求(接收来自客户端的请求)
        //accept()的三个参数:
        //参数1:服务器的socket描述字,监听socket描述字,声明方式SOCKET socket
        //参数2:通信协议地址,声明方式struct sockaddr_in * addr
        //参数3:协议地址的长度,声明方式socklen_t * addrlen
        //accept()返回值是一个由内核自动生成的全新socket描述字,代表与返回客户端的TCP连接

        sockClient = accept(sockListen, (SOCKADDR *)&remoteAddr, &addrlen);
        if(sockClient == INVALID_SOCKET)  
        {  
            printf("accept error !");
            // continue;
            return 0;
        }  
        printf("\nstay connetction with IP : %s \r\n", inet_ntoa(remoteAddr.sin_addr));

        // printf("等待连接...\n");  
        

        //七、接收数据  
        //recv()的四个参数:
        //参数1:链接socket描述字,声明方式[in] SOCKET socket
        //参数2:接收的多字节数据缓冲区,声明方式[in] const char * recvbuf
        //参数3:接受的多字节长度,声明方式[in] int buflen
        //参数4:指定进行调用的方式,声明方式[in] int flags
        //注:
            //recvbuf:接收数据之前,必须memset进行清空,接收的数据不一定填满空间
            //返回值:
                //未发生错误,则将返接收到的字符数,recvbuf指向的缓冲区将包含接收的数据
                //如果连接已正常关闭,则返回0
                //否则返回SOCKET_ERROR,通过调用WSAGetLastError来检索特定的错误代码
                //错误代码参考微软官网地址https://learn.microsoft.com/zh-cn/windows/win32/api/winsock2/nf-winsock2-recv
        int ret = recv(sockClient, revData, 255, 0);         
        if(ret > 0)  
        {  
            printf("\naccept info from client, content: \n\t\t\t\t\t");  
            //将数据以十六进制存储
            revData[ret] = 0x00;  
            // printf("accept info from client, content: ",revData,"\n");  
            
            printf(revData);
        }  
 
        //发送数据  
        const char * sendData = "hello tcp client!\n";
        // const char * sendData = recData;

        send(sockClient, sendData, strlen(sendData), 0);  

        //八、关闭socket
        //closesocket()的参数为socket描述字,声明方式为int closesocket([in] SOCKET s);
        //注:返回值:
            //无异常,返回0
            //否则返回一个SOCKET_ERROR的值:错误代码和意义如下:
                //WSANOTINITIALISED 未初始化调用WSAStartup
                //WSAENETDOWN 网格子系统出现故障
                //WSAENOTSOCK 描述符不是套接字
                //WSAEINPROGRESS 阻止Windows套接字1.1调用正在进行中,后者服务提供商仍在处理回调函数
                //WSAEINTR 阻止Windows socket 1.1 调用已通过WSACancelBlockingCall取消
                //WSAEWOULDBLOCK 套接字标记为非阻塞,但延迟结构的l_onoff成员设置为非零,l_linger成员的延迟结构设置为非零超时值
                //注:
                    //close标记TCP socket为已关闭,不可作为读写数据的第一个参数
                    //#include<unistd.h> int close(int socket)
                    //注: close只是使socket描述字的引用计数-1,当引用计数为0才会触发TCP客户端向服务器发送中止连接请求
        closesocket(sockClient);  
    }  
 
    closesocket(sockListen);  

    //九、停止使用WSACleanup
    //使用方式int WSACleanup();
    //返回值:
        //无异常则返回0;
        //否则返回SOCKET_ERROR值,调用WSAGetLastError来检索特定的错误代码,错误代码和含义如下:
            //WSANOTINITIALISED* 未初始化调用WSAStartup
            //WSAENETDOWN 网络子系统出现故障
            //WSAEINPROGRESS 阻止Windows套接字1.1调用正在进行中,后者服务提供商仍在处理回调函数
        
    WSACleanup();  
    // printf("WSACleanup!!!")
    return 0;  
} 

四、客户端代码

websocket_client.cpp

#include<WINSOCK2.H>
#include<STDIO.H>
#include<iostream>
#include<cstring>
using namespace std;
#pragma comment(lib, "ws2_32.lib")

int main()
{
    //一、初始化WSA  
    WORD sockVersion = MAKEWORD(2, 2);
    WSADATA data;
    if(WSAStartup(sockVersion, &data)!=0)
    {
        return 0;
    }
    if(true){
        printf("\nconnect to server successfully");
        
    }
    while(true){

        //二、创建套接字  socket(参数1:协议族,参数2:socket类型,参数3:协议类型)
        //AF_INET指IPv4 Internet协议
        //SOCK_STREAM指TCP连接,提供序列化的可靠的,双向连接的字节流,支持带外数据传输
        //IPPROTO_TCP指TCP协议
        SOCKET sclient = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
        if(sclient == INVALID_SOCKET)
        {
            printf("invalid socket!");
            return 0;
        }

        //三、通信协议地址赋值
        //声明通信协议地址参数
        sockaddr_in serAddr;
        //赋值地址:
            //语法1:inet_pton(AF_INET,"127.0.0.1",&addr.sin_addr);
            //语法2:addr.sin_addr.S_un.S_addr=inet_addr("127.0.0.1");
        serAddr.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");
        //协议赋值:赋值为IPv4 Internet协议
        serAddr.sin_family = AF_INET;
        //赋值端口:语法:addr.sin_port=htons(8888)
        serAddr.sin_port = htons(8888);
        
        //四、connect连接
        //connect()的三个参数:
        //参数1:连接的socket描述字,声明方式SOCKET socket
        //参数2:通信协议地址,声明方式const struct sockaddr_in * addr
        //参数3:socket地址长度,声明方式socket_t addrlen
        if(connect(sclient, (sockaddr *)&serAddr, sizeof(serAddr)) == SOCKET_ERROR)
        {  //连接失败 
            printf("connect error !");
            closesocket(sclient);
            return 0;
        }
        if(true){
            printf("\ninput message content please:\r\n ");
            
        }

 
        string data;
        cin>>data;
        // const char * sendData;
        const char * sendData = "hello tcp server, I am client!\n";
        sendData = data.c_str();   //string转const char* 
        //char * sendData = "你好,TCP服务端,我是客户端\n";
        
        //五、send发送数据(向服务端发送)
        //send()的四个参数:声明方式int WSAAPI send()
        //参数1:连接的socket描述字,声明方式[in]SOCKET socket
        //参数2:发送的多字节数据缓冲区,声明方式[in]const char * sendbuf
        //参数3:发送多字节长度,声明方式[in]int buflen
        //参数4:指定进行调度的方式,声明方式[in]int flags
        //注:
            //buflen:
                //大于发送缓冲区的长度时,返回SOCKET_ERROR
                //小于或等于时,send先检查协议是否在发送socket数据,是则等待发送完
                //不是则进行比较socket发送缓冲区的剩余空间:
                    //大于剩余空间大小则一直等待socket发送完
                    //小于剩余空间大小则sendbuf数据copy到剩余空间中
            //flags:
                //MSG_DONTROUTE指定对应数据进行路由,Windows套接字服务提供商可以选择忽略此标志
                //MSG_OOB发送OOB数据(流式套接字,例如仅SOCK_STREAM)
        send(sclient, sendData, strlen(sendData), 0);
        //send()用来将数据由指定的socket传给对方主机
        //int send(int s, const void * msg, int len, unsigned int flags)
        //s为已建立好连接的socket,msg指向数据内容,len则为数据长度,参数flags一般设0
        //成功则返回实际传送出去的字符数,失败返回-1,错误原因存于error 
 
        char recData[255];
        int ret = recv(sclient, recData, 255, 0);
        if(ret>0){
            printf("\naccept info from server, content: \n\t\t\t\t\t");
            
            recData[ret] = 0x00;
            // printf("接收到来自服务端的消息:"+recData);
            // printf("accept info from server, content: ",recData);
            printf(recData);
        } 
        //六、关闭socket
        //closesocket()的参数为socket描述字,声明方式为int closesocket([in] SOCKET s);
            //注:返回值:
                //无异常,返回0
                //否则返回一个SOCKET_ERROR的值:错误代码和意义如下:
                    //WSANOTINITIALISED 未初始化调用WSAStartup
                    //WSAENETDOWN 网格子系统出现故障
                    //WSAENOTSOCK 描述符不是套接字
                    //WSAEINPROGRESS 阻止Windows套接字1.1调用正在进行中,后者服务提供商仍在处理回调函数
                    //WSAEINTR 阻止Windows socket 1.1 调用已通过WSACancelBlockingCall取消
                    //WSAEWOULDBLOCK 套接字标记为非阻塞,但延迟结构的l_onoff成员设置为非零,l_linger成员的延迟结构设置为非零超时值
                    //注:
                        //close标记TCP socket为已关闭,不可作为读写数据的第一个参数
                        //#include<unistd.h> int close(int socket)
                        //注: close只是使socket描述字的引用计数-1,当引用计数为0才会触发TCP客户端向服务器发送中止连接请求
        closesocket(sclient);
    }
     //七、停止使用WSACleanup
    //使用方式int WSACleanup();
    //返回值:
        //无异常则返回0;
        //否则返回SOCKET_ERROR值,调用WSAGetLastError来检索特定的错误代码,错误代码和含义如下:
            //WSANOTINITIALISED* 未初始化调用WSAStartup
            //WSAENETDOWN 网络子系统出现故障
            //WSAEINPROGRESS 阻止Windows套接字1.1调用正在进行中,后者服务提供商仍在处理回调函数
    WSACleanup();
    return 0;    
} 

五、通信测试

在文件所在文件夹中打开cmd窗口进行编译代码
注1:-o参数是指定文件编译后生成的exe文件名称,运行时也是直接使用编译后的exe文件名运行
注2:-lwsock32参数是以命令的形式引入所需的基本库

1. 客户端编译运行

开启一个窗口用来编译运行服务端代码
编译命令

g++ websocket_server.cpp -o server -lwsock32

运行命令

server

在这里插入图片描述

2. 服务端编译运行

开启另一个窗口用来编译运行客户端代码
编译命令

g++ websocket_client.cpp -o client -lwsock32

运行命令

client

在这里插入图片描述
这里就是连接成功了

3. 发送消息

在客户端的窗口中输入消息,回车发送
客户端窗口内容
在这里插入图片描述
服务端窗口内容
在这里插入图片描述
注:中文好像不行,会报错,暂时没有解决,还有一个问题就是在客户端中无法打印内容,只能在有if条件的内容中打印东西
以上就是C++实现websocket客户端和服务端的通信


感谢阅读,祝君暴富!

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【postgresql】替换 mysql 中的ifnull()

数据库由mysql 迁移到postgresql&#xff0c;程序在执行查询时候报错。 HINT: No function matches the given name and argument types. You might need to add explicit type casts. CONTEXT: referenced column: ifnull 具体SQL: SELECT ifnull(phone,) FROM c_user p…

mysql限制用户登录失败次数,限制时间

mysql用户登录限制设置 mysql 需要进行用户登录次数限制,当使用密码登录超过 3 次认证链接失败之后,登录锁住一段时间,禁止登录这里使用的 mysql: 8.1.0 这种方式不用重启数据库. 配置: 首先进入到 mysql 命令行:然后需要安装两个插件: 在 mysql 命令行中执行: mysql> INS…

软件项目管理【UML-类图】

前言 UML图有很多种&#xff0c;但是并非必须掌握所有的UML图&#xff0c;才能完整系统分析和设计工作。一般说来&#xff0c;在UML图中&#xff0c;只要掌握类图、用例图、时序图的使用&#xff0c;就能完成大部分的工作。也就是说&#xff0c;掌握UML的20%&#xff0c;就能做…

Vivado中增加源文件界面中各选项的解释

文章目录 官方解释结论总结验证增加单个.v文件增加文件夹Copy sources into project 参考文献 本文对Vivado中增加源文件界面Add or Create Design Sources和Add or Create Smulaton sources中的选项Scan and add RTL include files into project、Copy sources into project和…

影响软文效果的三大因素,一定要牢记

在信息技术发展速度越来越快的今天&#xff0c;企业宣传时已经不再局限于传统的硬广&#xff0c;开始利用软文来提升曝光率&#xff0c;软文作为一种全新的宣传手段&#xff0c;具有覆盖面广、成本低且持续时间长&#xff0c;但是有三大因素稍不注意就会影响软文发布的效果&…

八月份跳槽了,历经华为测开岗4轮面试,不出意外,被刷了...

大多数情况下&#xff0c;测试员的个人技能成长速度&#xff0c;远远大于公司规模或业务的成长速度。所以&#xff0c;跳槽成为了这个行业里最常见的一个词汇。 前几天&#xff0c;我看到有朋友留言说&#xff0c;他在面试华为的测试开发工程师的时候&#xff0c;灵魂拷问三小…

buuctf web [极客大挑战 2019]LoveSQL

又是这样的界面&#xff0c;这糟糕的熟悉感&#xff0c;依旧使用上题套路 用户名&#xff1a; admin or 11# 密码&#xff1a; 1 有一串很像flag的字符&#xff0c;但是很可惜&#xff0c;这不是flag 看了一眼源代码&#xff0c;没有可以跳转的页面 要换个思路了&#xff0c…

C++ -- 学习系列 std::array 容器

1. std::array 是什么&#xff1f; array 容器是 C 11 标准中新增的序列容器&#xff0c;简单地理解&#xff0c;它就是在 C 普通数组的基础上&#xff0c;添加了一些成员函数和全局函数。在使用上&#xff0c;它比普通数组更安全&#xff0c;且效率并没有因此变差。 与数组一…

OmniPlan Pro 4 for Mac:引领项目管理的创新与高效

OmniPlan Pro 4是一款强大且高效的项目管理工具&#xff0c;专为Mac用户设计。它提供了一套综合性的解决方案&#xff0c;帮助用户在Mac上便捷地进行项目规划、追踪和管理。凭借其直观的界面&#xff0c;用户可以快速上手&#xff0c;并且能充分利用这款工具的各种功能。 规划…

企业如何获得用户信赖感,媒介盒子告诉你

互联网和新媒体的快速发展使得内容市场空前繁荣&#xff0c;营销方式也越来越多元化&#xff0c;消费者已经很难被夸张吸睛的营销模式打动&#xff0c;他们更倾向于选择自己信赖的产品&#xff0c;对于企业而言&#xff0c;用户信任是决定企业发展与否的关键因素&#xff0c;产…

自动直播软件开发方案:打造智能化、高效化的直播体验

一、引言 随着社交媒体和互联网的快速发展&#xff0c;直播已经成为了人们互动和传播的重要方式。然而&#xff0c;传统直播存在着人力成本高、内容质量不稳定等问题&#xff0c;因此&#xff0c;开发一款自动直软件成为了解决这些问题的关键。 二、市场需求分析 1…

APP稳定性测试工具:Monkey

一、Monkey 简介 Monkey 是一款 app 的自动化测试工具&#xff0c;monkey 是猴子的意思&#xff0c;所以从原理上说&#xff0c;它的自动化测试就类似猴子一样在软件上乱敲按键&#xff0c;猴子什么都不懂&#xff0c;就爱捣乱。Monkey 原理也是类似&#xff0c;通过向系统发送…

JsonUtils

1、工具类 package com.atguigu.utils;import com.fasterxml.jackson.annotation.JsonInclude; import com.fasterxml.jackson.core.JsonProcessingException; import com.fasterxml.jackson.core.type.TypeReference; import com.fasterxml.jackson.databind.Deserialization…

解决hadoop使用put上传报错问题

hadoop使用put上传报错 WARN hdfs.DataStreamer: DataStreamer Exception org.apache.hadoop.ipc.RemoteException(java.io.IOException): File /input/yxqzdata.COPYING could only be replicated to 0 nodes instead of minReplication (1). There are 0 datanode(s) runnin…

MySQL搭建主从复制集群,实现读写分离

目录 一、准备 二、配置 2.1 配置主库 修改配置文件/etc/my.cnf 重启服务 为主库再创建一个账户并授权 查看状态 2.2 配置从库 修改配置文件/etc/my.cnf 重启mysql服务 配置需要同步的主机 启动salve同步 查看是否同步 三、测试主从复制是否生效 四、读写分离案…

【计算机毕业设计】基于SpringBoot+Vue热门网游推荐网站的设计与实现

博主主页&#xff1a;一季春秋博主简介&#xff1a;专注Java技术领域和毕业设计项目实战、Java、微信小程序、安卓等技术开发&#xff0c;远程调试部署、代码讲解、文档指导、ppt制作等技术指导。主要内容&#xff1a;毕业设计(Java项目、小程序等)、简历模板、学习资料、面试题…

C++之va_start、vasprintf、va_end应用总结(二百二十六)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

腾讯mini项目-【指标监控服务重构】2023-07-29

今日已办 根据导师代码评审的 comment 改进 修改命名 otelclient.otelExporterOtlpEndpoint to otelclient.endpoint禁用 TLS 加密&#xff0c;移除 otelclient.insecureMode避免命名缩写影响代码的阅读理解把两个函数&#xff08;createTraceExp和createTraceProvider&#…

Softing物联网(IoT)方案之OT/IT数据集成

一 利用数据提高效率和绩效 多年以来数据集成和工业物联网一直在推动着市场的发展&#xff0c;目前我们已经能够集成并成功使用先进的技术、大量的传感器和复杂的数据格式等。而在工业物联网或工业4.0中&#xff0c;还有运营技术&#xff08;OT&#xff09;和信息技术&#xf…

会C++还需要再去学Python吗?

提到的C、数据结构与算法、操作系统、计算机网络和数据库技术等确实是计算机科学中非常重要的基础知识领域&#xff0c;对于软件开发和计算机工程师来说&#xff0c;它们是必备的核心知识。掌握这些知识对于开发高性能、可靠和安全的应用程序非常重要。Python作为一种脚本语言&…