網絡庫libhv介紹

? ? ? libhv是一個類似于libevent、libev、libuv的跨平臺網絡庫,提供了更易用的接口和更豐富的協議,用來開發TCP/UDP/SSL/HTTP/WebSocket/MQTT 客戶端/服務端。源碼地址:https://github.com/ithewei/libhv,最新發布版本為v1.3.3,License為BSD-3-Clause。

? ? ? libhv特性

? ? ? (1).跨平臺:Linux、Windows、macOS、Android、iOS、BSD、Solaris

? ? ? (2).高性能事件循環:網絡IO事件、定時器事件、空閑事件、自定義事件、信號

? ? ? (3).TCP/UDP 客戶端/服務端/代理

? ? ? (4).TCP支持心跳、重連、轉發、多線程安全寫入和關閉等功能

? ? ? (5).內置常見的拆包模式:固定包長、分界符、頭部長度字段

? ? ? (6).RUDP支持:WITH_KCP

? ? ? (7).SSL/TLS支持:可選WITH_OPENSSL、WITH_GNUTLS或WITH_MBEDTLS

? ? ? (8).HTTP客戶端/服務器支持:https http1/x http2 grpc

? ? ? (9).HTTP支持靜態文件服務、目錄服務、正向/反向代理服務、同步/異步API處理器

? ? ? (10).HTTP支持RESTful風格、路由、中間件、keep-alive長連接、chunked分塊、SSE等功能

? ? ? (11).WebSocket服務端/客戶端

? ? ? (12).MQTT客戶端

? ? ? libhv在Windows上編譯,build.sh內容如下:

#! /bin/bashif [ $# != 1 ]; thenecho "Error: requires one parameters:  Relese or Debug"echo "For example: $0 Debug"exit -1
fiif [ $1 != "Release"  ] && [ $1 != "Debug" ]; thenecho "Error: the sparameter can only be Release or Debug"exit -1
fimkdir -p build && cd buildcmake \-G"Visual Studio 17 2022" -A x64 \${cuda_options} \-DCMAKE_BUILD_TYPE=$1 \-DCMAKE_CONFIGURATION_TYPES=$1 \-DCMAKE_INSTALL_PREFIX=../install \..
cmake --build . --target install --config $1rc=$?
if [[ ${rc} != 0 ]]; thenecho -e "\033[0;31mError: there are some errors in the above operation, please check: ${rc}\033[0m"exit ${rc}
elseecho "build completed"
fi

? ? ? HTTP客戶端測試代碼如下:

int test_libhv_http_client()
{const std::string server_url{ "http://192.168.19.205:8080" };HttpRequest request{};request.method = HTTP_GET;request.url =  server_url + "/api/test1";hv::HttpClient client{};HttpResponse response{};if (auto ret = client.send(&request, &response); ret == 0) {auto j = hv::Json::parse(response.body);if (j.contains("time")) {std::cout << "server time: " << j["time"] << std::endl;} else {std::cerr << "Error: missing time field: " << response.body << std::endl;}} else {std::cerr << "Error: failed to request: " << ret << std::endl;}constexpr char image_name[]{ "../../../testdata/cat.jpg" };std::ifstream in_file(image_name, std::ios::binary | std::ios::ate); if (!in_file.is_open()) {std::cerr << "Error: fail to open file: " << image_name << std::endl;return -1;}size_t file_size = in_file.tellg();std::unique_ptr<unsigned char[]> data(new unsigned char[file_size]);in_file.seekg(0);in_file.read(reinterpret_cast<char*>(data.get()), file_size);auto base64_data = hv::Base64Encode(data.get(), file_size);hv::Json j = {{"image_name", image_name},{"image_data", base64_data},{"image_size", file_size}};HttpRequest request2{};request2.method = HTTP_POST;request2.url = server_url + "/api/test2";request2.body = j.dump();request2.headers["Content-Type"] = "application/json";request2.timeout = 2;HttpResponse response2{};if (auto ret = client.send(&request2, &response2); ret == 0) {if (response2.status_code == HTTP_STATUS_OK) {// 200hv::Json j = hv::Json::parse(response2.body);if (!j.contains("image_size") || !j.contains("image_time")) {std::cerr << "Error: missing image_size or image_time: " << response2.body << std::endl;return -1;}std::cout << "image was created at: " << j["image_time"] << ", image size: " << j["image_size"] << std::endl;} else {std::cerr << "status code: " << response2.status_code << ", status message: " << response2.status_message() << ", body: " << response2.body << std::endl;return -1;}} else {std::cerr << "Error: failed to send, error code: " << ret << std::endl;return -1;}return 0;
}

? ? ? 執行結果如下圖所示:

? ? ? HTTP服務端測試代碼如下:

int test_libhv_http_server()
{namespace fs = std::filesystem;hv::HttpService router{};router.GET("/api/test1", [](const HttpContextPtr& ctx) {std::cout << "client: ip addr: " << ctx->ip() << ", port: " << ctx->port() << ", method: " << ctx->method() << std::endl;auto get_local_time = [] {auto time = std::chrono::system_clock::to_time_t(std::chrono::system_clock::now());std::tm* tm = std::localtime(&time);std::stringstream buffer;buffer << std::put_time(tm, "%Y-%m-%d %H:%M:%S");return buffer.str();};hv::Json j = {{"status", "success"},{"time", get_local_time()}};return ctx->send(j.dump());});router.POST("/api/test2", [](const HttpContextPtr& ctx) {try {std::cout << "client: ip addr: " << ctx->ip() << ", port: " << ctx->port() << ", method: " << ctx->method() << std::endl;auto j = hv::Json::parse(ctx->body());if (!j.contains("image_name") || !j.contains("image_data") || !j.contains("image_size")) {ctx->setStatus(HTTP_STATUS_BAD_REQUEST);return ctx->send(R"({"error":"missing image_name or image_data or image_size"})");}auto data = hv::Base64Decode(j["image_data"].get<std::string>().c_str());if (data.length() != j["image_size"]) {ctx->setStatus(HTTP_STATUS_PRECONDITION_FAILED);return ctx->send(R"({"error":"data length mismatch"})");}fs::path image_path = j["image_name"].get<std::string>();auto image_name = image_path.filename();std::ofstream out_file(image_name.string(), std::ios::binary);out_file.write(data.data(), data.length());out_file.close();auto get_time = [](fs::file_time_type tp) {using namespace std::chrono;auto sctp = time_point_cast<system_clock::duration>(tp - fs::file_time_type::clock::now() + system_clock::now());auto tt = system_clock::to_time_t(sctp);std::tm* gmt = std::localtime(&tt); // UTC: std::gmtime(&tt);std::stringstream buffer;buffer << std::put_time(gmt, "%Y-%m-%d %H:%M:%S");return buffer.str();};auto image_time = get_time(fs::last_write_time(image_name));std::cout << "image was created at: " << image_time << std::endl;auto get_file_size = [](std::uintmax_t size) {float s1 = size / 1024. / 1024 / 1024;float s2 = size / 1024. / 1024;float s3 = size / 1024.;if (s1 > 1)return std::make_tuple(s1, std::string(" GB"));if (s2 > 1)return std::make_tuple(s2, std::string(" MB"));if (s3 > 1)return std::make_tuple(s3, std::string(" KB"));return std::make_tuple(static_cast<float>(size), std::string(" Bytes"));};auto [image_size, suffix] = get_file_size(static_cast<std::intmax_t>(fs::file_size(image_name)));std::cout << "image size: " << image_size << suffix << std::endl;ctx->setContentType(APPLICATION_JSON);hv::Json j2 = {{"status", "success"},{"image_time", image_time},{"image_size", std::format("{:.4f}{}", image_size, suffix)}};return ctx->send(j2.dump());} catch (const std::exception& e) {ctx->setStatus(HTTP_STATUS_INTERNAL_SERVER_ERROR);hv::Json j3 = { "error", e.what() };return ctx->send(j3.dump());}});hv::HttpServer server{};server.port = 8080;//server.worker_threads = 2;server.service = &router;//server.run(); // blockingserver.start(); // non-blockingwhile (true) {hv_delay(1000);}return 0;
}

? ? ? 執行結果如下圖所示:

? ? ? GitHub:https://github.com/fengbingchun/OpenSSL_Test

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/pingmian/83920.shtml
繁體地址,請注明出處:http://hk.pswp.cn/pingmian/83920.shtml
英文地址,請注明出處:http://en.pswp.cn/pingmian/83920.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

施耐德特價型號伺服電機VIA0703D31A1022、常見故障

?? ?一、啟動類故障? ?電機無法啟動? ?可能原因?&#xff1a;電源未接通、制動器未釋放、接線錯誤或控制器故障。?解決措施?&#xff1a; 檢查電源線路及斷路器狀態&#xff1b;驗證制動器是否打開&#xff08;帶制動器型號&#xff09;&#xff1b;核對電機與控制器…

【Redis從入門到精通實戰文章匯總】

&#x1f4da;博客主頁&#xff1a;代碼探秘者 ?專欄&#xff1a;文章正在持續更新ing… ?C語言/C&#xff1a;C&#xff08;詳細版&#xff09; 數據結構&#xff09; 十大排序算法 ?Java基礎&#xff1a;JavaSE基礎 面向對象大合集 JavaSE進階 Java版數據結構JDK新特性…

MCP 技術完全指南:微軟開源項目助力 AI 開發標準化學習

引言 在人工智能快速發展的今天&#xff0c;如何讓 AI 模型與客戶端應用程序之間建立標準化的交互機制&#xff0c;已成為開發者們亟待解決的關鍵問題。微軟近期開源的 mcp-for-beginners 項目&#xff0c;為我們提供了一個系統性學習 Model Context Protocol (MCP) 的絕佳機會…

SQL進階之旅 Day 20:鎖與并發控制技巧

【JDK21深度解密 Day 20】鎖與并發控制技巧 文章簡述 在高并發的數據庫環境中&#xff0c;鎖與并發控制是保障數據一致性和系統穩定性的核心機制。本文作為“SQL進階之旅”系列的第20天&#xff0c;深入探討SQL中的鎖機制、事務隔離級別以及并發控制策略。文章從理論基礎入手…

Qt(part 2)1、Qwindow(菜單欄,工具欄,狀態欄),鉚接部件,核心部件 ,2、添加資源文件 3、對話框

1、Qwindow tips&#xff1a;1&#xff0c;首先為什么創建出的對象基本都是指針形式&#xff0c;個人覺得是對象樹的原因&#xff08;自動釋放內存&#xff09;&#xff0c;指針來訪問成員函數->的形式。2&#xff0c;菜單欄只能一個的&#xff0c;放窗口基本Set&#xff0c…

一款“短小精悍的”手機錄屏軟件

這個時代&#xff0c;手機自帶錄屏功能已經不是什么稀奇的事情了&#xff0c;但是手機自帶的錄屏功能不都是完美的&#xff0c;無法靜音錄屏、、不能修改畫質、不能剪輯、不能自定義水印......emmm.....貌似除了錄屏就什么都不會 今天分享的這款軟件——ADV屏幕錄制漢化版&…

力扣HOT100之二分查找:153. 尋找旋轉排序數組中的最小值

這道題是上一道題&#xff1a;33. 搜索旋轉排序數組的前置題&#xff0c;有點沒看懂力扣為什么要這樣安排題目順序&#xff0c;應該把這道題按排在前面才對啊。。。這道題的思路已經在上一道題的思路中說過了&#xff0c;這里就直接復制粘貼上一篇博客中的內容了。 我們閱讀完題…

libiec61850 mms協議異步模式

之前項目中使用到libiec61850庫&#xff0c;都是服務端開發。這次新的需求要接收服務端的遙測數據&#xff0c;這就涉及到客戶端開發了。 客戶端開發沒搞過啊&#xff0c;挑戰不少&#xff0c;但是人不就是通過戰勝困難才成長的嘛。通過查看libiec61850的客戶端API發現&#xf…

【 知你所想 】基于ernie-x1-turbo推理模型實現趣味猜心游戲

&#x1f31f; 項目特點 &#x1f916; 智能AI&#xff1a;基于文心一言大模型&#xff0c;具有強大的推理能力&#x1f3af; 實時思考&#xff1a;展示AI的思考過程&#xff0c;讓你了解AI是如何推理的&#x1f3ae; 互動性強&#xff1a;通過簡單的"是/否"問答&…

Excel 模擬分析之單變量求解簡單應用

正向求解 利用公式根據貸款總額、還款期限、貸款利率&#xff0c;求每月還款金額 反向求解 根據每月還款能力&#xff0c;求最大能承受貸款金額 參數&#xff1a; 目標單元格&#xff1a;求的值所在的單元格 目標值&#xff1a;想要達到的預期值 可變單元格&#xff1a;變…

關于easyexcel動態下拉選問題處理

前些日子突然碰到一個問題&#xff0c;說是客戶的導入文件模版想支持部分導入內容的下拉選&#xff0c;于是我就找了easyexcel官網尋找解決方案&#xff0c;并沒有找到合適的方案&#xff0c;沒辦法只能自己動手并分享出來&#xff0c;針對Java生成Excel下拉菜單時因選項過多導…

【Qt】之【Get√】【Bug】通過值捕獲(或 const 引用捕獲)傳進 lambda,會默認復制成 const

通過值捕獲&#xff08;或 const 引用捕獲&#xff09;傳進 lambda&#xff0c;會默認復制成 const。 背景 匿名函數外部定義 QSet<QString> nameSet,需要傳入匿名函數使用修改 connect(dlg, ..., [nameSet](...) {nameSet.insert(name); // ? 這里其實是 const QSet…

css元素的after制作斜向的刪除線

<div class"price_div"></div>.price_div{position: relative; } ::after{content: ;position: absolute;left: 0;top: 50%;width: 100%;height: 2px;background: #FF186B;transform: rotate(-5deg); }

uniapp map組件的基礎與實踐

UniApp 中的 map 組件用于在應用中展示地圖,并且支持在地圖上添加標記、繪制線條和多邊形等功能。以下是一些基本用法: 1. 基本結構 首先,確保你在頁面的 .vue 文件中引入了 map 組件。以下是創建一個簡單地圖的基本代碼結構: <template><view class="con…

深入理解PHP安全漏洞:文件包含與SSRF攻擊全解析

深入理解PHP安全漏洞&#xff1a;文件包含與SSRF攻擊全解析 前言 在Web安全領域&#xff0c;PHP應用程序的安全問題一直備受關注。本文將深入探討兩種常見的PHP安全漏洞&#xff1a;文件包含漏洞和服務器端請求偽造(SSRF)&#xff0c;幫助開發者理解漏洞原理、利用方式以及防…

MS358A 低功耗運算放大器 車規

MS358A 低功耗運算放大器 車規 產品簡述 MS358A 是雙通道運算放大器&#xff0c;具有低功耗、寬電源電壓范圍、高單位增益帶寬的特性。在特定情況下&#xff0c;壓擺率可以達到0.4V/μs 。每個通道的靜態電流 (5V) 只有 430μA 。 MS358A輸入共模范圍可以到地&#xff0c;同時…

n8n + AI Agent:AI 自動化生成測試用例并支持導出 Excel

n8n + AI Agent:AI 自動化生成測試用例并支持導出 Excel 最終成果展示一、準備工作二、手把手搭建工作流第一步:創建手動觸發器 (Chat Trigger)第二步:創建 AI Agent 節點第三步:為 AI Agent 植入 DeepSeek AI 模型第四步:解析AI的響應 (Code)第五步:生成Excel文件 (Conv…

5.1 HarmonyOS NEXT系統級性能調優:內核調度、I/O優化與多線程管理實戰

HarmonyOS NEXT系統級性能調優&#xff1a;內核調度、I/O優化與多線程管理實戰 在HarmonyOS NEXT的全場景生態中&#xff0c;系統級性能調優是構建流暢、高效應用的關鍵。通過內核調度精細化控制、存儲與網絡I/O深度優化&#xff0c;以及多線程資源智能管理&#xff0c;開發者…

?線性注意力 vs. 傳統注意力:效率與表達的博弈新解

?核心結論?&#xff1a;線性注意力用計算復雜度降維換取全局建模能力&#xff0c;通過核函數和結構優化補足表達缺陷 一、本質差異&#xff1a;兩種注意力如何工作&#xff1f; ?特性?傳統注意力&#xff08;Softmax Attention&#xff09;線性注意力&#xff08;Linear At…

github中main與master,master無法合并到main

文章目錄 遇到問題背景怎么做 遇到問題 上傳 github 時候&#xff0c;發現傳上去的是 master&#xff0c;但是 github 竟然還有一個 main 背景 github 采用 main 替代 master 作為主分支不是出于技術背景&#xff0c;而是出于 2020 年全球范圍內興起的 “Black Lives Matter…