C++下跨平台websocket库及使用示例

2024-04-28 14:36

本文主要是介绍C++下跨平台websocket库及使用示例,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

websocketpp库使用非常方便,开源,可跨平台,内部实现全部是head文件,无cpp,接口简单易用。源码路径比如 https://github.com/zaphoyd/websocketpp.git

使用方法是,先下载源码包并复制道工程的include目录下。

--include

   --websocketpp

然后在代码#include "utility_client.hpp"即可调用其接口进行websocket通信。以linux为例具体使用的方法代码示例:

websockettalker.h:

#ifndef WEBSOCKETTALKER_H
#define WEBSOCKETTALKER_H
#include "dao.h"
#include "utility_client.hpp"
#include "pthread.h"
#include "vector"using namespace std;
class CloudTalker;
class WebsocketTalker
{
public:WebsocketTalker(CloudTalker* talker, const string& remoteAddr, const string& name);~WebsocketTalker();string GetStatus();string GetMessage();bool Subscribe(const string& deviceId, const string& token);void handleNbData(const string& msg);private:CloudTalker* mTalker;static void* thread_loop(void* pVoid);pthread_t recv_thread;websocket_endpoint mEndpoint;int mId;string addr = "";bool should_exit = false;vector<DevInfo> mImeiTable;DAO* mDevDbDao = NULL;
};#endif // WEBSOCKETTALKER_H

websockettalker.cpp:

#include "websockettalker.h"
#include "cloudtalker.h"
#include "udptalker.h"
#include <boost/algorithm/string.hpp>using namespace boost;
#define JSON_PATH_DEV "path/to/devDB.json"
WebsocketTalker::WebsocketTalker(CloudTalker* talker, const string& remoteAddr, const string& name)
{string messageclient;addr = "ws://" + remoteAddr + ":8081/webSocketServer?" + name; //格式为cout <<"full address is:" <<addr <<endl;mTalker = talker;mDevDbDao = new DAO();pthread_create(&recv_thread, nullptr, thread_loop, this);
}/***********************************************************
* 描述: 处理数据
* 入参: wholeMsg - 要处理的数据
* 出参: 无
* 返回: 无
************************************************************/
void WebsocketTalker::handleData(const string& wholeMsg)
{cout <<endl;}/***********************************************************
* 描述: websocket通信线程
* 入参: pVoid - WebsocketTalker指针
* 出参: 无
* 返回: 无
************************************************************/
void* WebsocketTalker::thread_loop(void* pVoid)
{WebsocketTalker* p = (WebsocketTalker*) pVoid;string status = "";string wholeMsg = "";int cont = 0;unsigned int len = 0;while(!p->should_exit){p->mImeiTable.clear();printf("WEBSOCKET WANT TO CREATE CONNECT FOR NBIOT DATA\n");p->mDevDbDao->getNBAndShareImeiAll(JSON_PATH_DEV, p->mImeiTable);len = p->mImeiTable.size();if(len == 0){int cnt = 0;printf("nb/share dev list is zero\n");while(!p->should_exit && cnt++ <4){usleep(500000);}cnt = 0;continue;}p->mId = p->mEndpoint.connect(p->addr);if (p->mId == -1){cout << "Created connection failed! mId=" << p->mId<< endl;}else{while(!p->should_exit){usleep(50000);status = p->GetStatus();if(status.length() && status != "Connecting")break;}if(p->should_exit){printf("exit 1\n");break;}for(unsigned int i=0; i<len; i++){cout<<"subscribe dev: " << p->mImeiTable[i].IMEi<< "with token: "<<p->mImeiTable[i].Token<<endl;p->Subscribe(p->mImeiTable[i].IMEi, p->mImeiTable[i].Token);cont = 0;while(!p->should_exit){usleep(50000);status = p->GetStatus();if(status == "Message")break;if(cont++ > 40){printf("websocket data receive subscribe status timeout\n");//cont = 0;break;}}if(p->should_exit || cont > 40){printf("exit 2\n");break;}wholeMsg = p->GetMessage();printf("[%s(%d)%s] msg=%s\n", __FUNCTION__, __LINE__, __TIME__, wholeMsg.data());cont = 0;/*循环接收消息 */while(!p->should_exit){if(cont++ > 40){printf("websocket data receive timeout\n");//cont = 0;break;}status = p->GetStatus();//printf("get status: %s\n", status.c_str());if(status == "Message"){wholeMsg = p->GetMessage();printf("[%s(%d)%s;] GOT SUBSCRIBED DATA, msg=%s\n", __FUNCTION__, __LINE__, __TIME__, wholeMsg.data());p->handleNbData(wholeMsg);cont = 0;break;}usleep(50000);}if(p->should_exit){printf("exit 3\n");break;}}p->mEndpoint.close(p->mId, websocketpp::close::status::normal, "close");if(p->should_exit){printf("exit 3\n");break;}}usleep(2000000);//sleep long time could be.}return NULL;
}
string WebsocketTalker::GetStatus()
{return mEndpoint.get_metadata(mId)->get_status();
}string WebsocketTalker::GetMessage()
{return mEndpoint.get_metadata(mId)->get_message();
}bool WebsocketTalker::Subscribe(const string& deviceId, const string& token)
{string str;str = "{\"type\":\"address\",\"msg\":\"" + deviceId + "\",\"token\":\"" + token +"\"}";printf("[%s(%d)] %s\n", __FUNCTION__, __LINE__, str.data());mEndpoint.send(mId, str);return true;
}WebsocketTalker::~WebsocketTalker()
{printf("exit wstalker\n");if(mDevDbDao != NULL){delete mDevDbDao;}should_exit = true;pthread_join(recv_thread, NULL);printf("exit wstalker done\n");
}

 dao.h:

#ifndef DAO_H
#define DAO_H#include <iostream>
#include <fstream>
#include <string>
#include <vector>
#include "json/json.h"#ifndef UNUSED
#define UNUSED(x) (void)(x)
#endifusing namespace std;typedef struct
{string IMEi;string Type;string Token;
}DevInfo;class DAO{public:DAO(){}// 10. get device information transform to <vector> from json, "device information contain {IMEI, type, token, index}"int getNBAndShareImeiAll(const char* dfile_path, vector<DevInfo>& imeiTable){fstream dev_f;dev_f.open(dfile_path, ios::in | ios::out);if (!dev_f.is_open()){cout << "Open file error !" << endl;}Json::Value val;Json::Value root_arr;JSONCPP_STRING errs;//bool did;DevInfo devInfo;Json::CharReaderBuilder rbuilder;rbuilder["collectComments"] = false;bool parse_ok = Json::parseFromStream(rbuilder, dev_f, &root_arr, &errs);if(!parse_ok){cout << "Parse json file error !" << endl;}else{const Json::Value arrayObj = root_arr;cout << " ------ Printf Device information: ------ " << endl;try{for (unsigned int i = 0; i < arrayObj.size(); i++){if(arrayObj[i]["Type"].asString() == "LORA"){devInfo ={arrayObj[i]["IMEi"].asString(),arrayObj[i]["Type"].asString(),arrayObj[i]["Token"].asString()};imeiTable.push_back(devInfo);}}}catch (const Json::LogicError &ex){cout << " Parse json string error!" << endl;}}dev_f.close();return 0;}};#endif // DAO_H

 devDB.json:

[{"IMEi" : "6789999","Token" : "jfkl","Type" : "LORA"},{"IMEi" : "00124b00","Token" : "0","Type" : "xxx"},]

json的解析使用jsoncpp库,可自行安装。 

这篇关于C++下跨平台websocket库及使用示例的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/943506

相关文章

使用Redis快速实现共享Session登录的详细步骤

《使用Redis快速实现共享Session登录的详细步骤》在Web开发中,Session通常用于存储用户的会话信息,允许用户在多个页面之间保持登录状态,Redis是一个开源的高性能键值数据库,广泛用于... 目录前言实现原理:步骤:使用Redis实现共享Session登录1. 引入Redis依赖2. 配置R

使用Python的requests库调用API接口的详细步骤

《使用Python的requests库调用API接口的详细步骤》使用Python的requests库调用API接口是开发中最常用的方式之一,它简化了HTTP请求的处理流程,以下是详细步骤和实战示例,涵... 目录一、准备工作:安装 requests 库二、基本调用流程(以 RESTful API 为例)1.

使用Python开发一个Ditto剪贴板数据导出工具

《使用Python开发一个Ditto剪贴板数据导出工具》在日常工作中,我们经常需要处理大量的剪贴板数据,下面将介绍如何使用Python的wxPython库开发一个图形化工具,实现从Ditto数据库中读... 目录前言运行结果项目需求分析技术选型核心功能实现1. Ditto数据库结构分析2. 数据库自动定位3

Python yield与yield from的简单使用方式

《Pythonyield与yieldfrom的简单使用方式》生成器通过yield定义,可在处理I/O时暂停执行并返回部分结果,待其他任务完成后继续,yieldfrom用于将一个生成器的值传递给另一... 目录python yield与yield from的使用代码结构总结Python yield与yield

Go语言使用select监听多个channel的示例详解

《Go语言使用select监听多个channel的示例详解》本文将聚焦Go并发中的一个强力工具,select,这篇文章将通过实际案例学习如何优雅地监听多个Channel,实现多任务处理、超时控制和非阻... 目录一、前言:为什么要使用select二、实战目标三、案例代码:监听两个任务结果和超时四、运行示例五

python使用Akshare与Streamlit实现股票估值分析教程(图文代码)

《python使用Akshare与Streamlit实现股票估值分析教程(图文代码)》入职测试中的一道题,要求:从Akshare下载某一个股票近十年的财务报表包括,资产负债表,利润表,现金流量表,保存... 目录一、前言二、核心知识点梳理1、Akshare数据获取2、Pandas数据处理3、Matplotl

Java使用Thumbnailator库实现图片处理与压缩功能

《Java使用Thumbnailator库实现图片处理与压缩功能》Thumbnailator是高性能Java图像处理库,支持缩放、旋转、水印添加、裁剪及格式转换,提供易用API和性能优化,适合Web应... 目录1. 图片处理库Thumbnailator介绍2. 基本和指定大小图片缩放功能2.1 图片缩放的

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

MySQL中EXISTS与IN用法使用与对比分析

《MySQL中EXISTS与IN用法使用与对比分析》在MySQL中,EXISTS和IN都用于子查询中根据另一个查询的结果来过滤主查询的记录,本文将基于工作原理、效率和应用场景进行全面对比... 目录一、基本用法详解1. IN 运算符2. EXISTS 运算符二、EXISTS 与 IN 的选择策略三、性能对比

MySQL常用字符串函数示例和场景介绍

《MySQL常用字符串函数示例和场景介绍》MySQL提供了丰富的字符串函数帮助我们高效地对字符串进行处理、转换和分析,本文我将全面且深入地介绍MySQL常用的字符串函数,并结合具体示例和场景,帮你熟练... 目录一、字符串函数概述1.1 字符串函数的作用1.2 字符串函数分类二、字符串长度与统计函数2.1