当前位置: 首页 > web >正文

C++ 学习 网络编程 2025年6月17日19:56:47

网络编程 

C++网络编程允许开发者创建能够通过网络进行通信的应用程序。以下是C++网络编程的主要方面和常用技术:

基础概念
  1. 套接字(Socket)编程:网络通信的基础

  2. 协议:TCP(可靠连接)和UDP(无连接)

  3. IP地址和端口:标识网络中的主机和服务

常用API

Berkeley套接字(BSD套接字)

实现了一个 基础的TCP服务器,它会监听本地的8080端口,接受客户端连接,并进行简单的数据收发。

#include <sys/socket.h>   // 提供socket相关函数和数据结构
#include <netinet/in.h>   // 包含IP地址和端口号的定义
#include <arpa/inet.h>    // 提供IP地址转换函数// 创建TCP套接字
// AF_INET表示IPv4协议,SOCK_STREAM表示面向连接的TCP套接字
// 返回值:成功返回套接字描述符,失败返回-1
int sockfd = socket(AF_INET, SOCK_STREAM, 0);// 配置服务器地址结构体
struct sockaddr_in serv_addr;
serv_addr.sin_family = AF_INET;          // 使用IPv4地址族
serv_addr.sin_port = htons(8080);        // 设置端口号为8080(htons将主机字节序转为网络字节序)
serv_addr.sin_addr.s_addr = INADDR_ANY;  // 允许接收任意网卡(所有IP地址)的连接// 将套接字绑定到指定地址和端口
// 成功返回0,失败返回-1
bind(sockfd, (struct sockaddr*)&serv_addr, sizeof(serv_addr));// 开始监听连接请求
// 参数5表示等待连接队列的最大长度(即允许的未完成连接数)
listen(sockfd, 5);// 接受客户端连接(阻塞调用,直到有客户端连接)
struct sockaddr_in cli_addr;            // 用于存储客户端地址信息
socklen_t clilen = sizeof(cli_addr);    // 客户端地址结构体长度
// 返回值newsockfd是专门与这个客户端通信的新套接字描述符
int newsockfd = accept(sockfd, (struct sockaddr*)&cli_addr, &clilen);// 发送数据到客户端
// buffer:要发送的数据缓冲区
// strlen(buffer):实际发送的数据长度
// 返回值:成功返回发送的字节数,失败返回-1
send(newsockfd, buffer, strlen(buffer), 0);// 从客户端接收数据
// buffer:接收数据的缓冲区
// sizeof(buffer):缓冲区最大容量
// 返回值:成功返回接收的字节数(0表示连接关闭),失败返回-1
recv(newsockfd, buffer, sizeof(buffer), 0);

 

  1.   创建socket >创建一个用于TCP通信的套接字(类似"电话机")
  2.   绑定bind>将套接字与指定的IP和端口绑定(类似"插电话线到插座")
  3.   监听listen>启动监听模式,等待客户端连接(类似"电话待机状态")
  4.   接受accept>接受客户端的连接请求(类似"接听电话")
  5.   通信send/recv>数据 收recv / 发send

 

典型应用场景
  1. 简单聊天程序:服务器接收客户端消息并回复。

  2. 远程控制:客户端发送指令,服务器执行操作。

  3. 文件传输:通过TCP可靠传输文件。

Windows套接字(Winsock)

#include <winsock2.h>    // Windows Socket API头文件
#include <ws2tcpip.h>    // 用于IP地址转换等扩展功能
#include <iostream>
#pragma comment(lib, "ws2_32.lib")  // 自动链接Winsock库int main() {// 1. 初始化Winsock(Windows特有)WSADATA wsaData;if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0) {  // 请求2.2版本std::cerr << "WSAStartup failed: " << WSAGetLastError() << std::endl;return 1;}std::cout << "Winsock初始化成功!" << std::endl;// 2. 创建TCP套接字SOCKET serverSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);if (serverSocket == INVALID_SOCKET) {std::cerr << "socket创建失败: " << WSAGetLastError() << std::endl;WSACleanup();return 1;}std::cout << "TCP套接字创建成功!" << std::endl;// 3. 配置服务器地址sockaddr_in serverAddr;serverAddr.sin_family = AF_INET;              // IPv4地址族serverAddr.sin_port = htons(8080);            // 监听8080端口serverAddr.sin_addr.s_addr = INADDR_ANY;      // 监听所有本地IP地址// inet_pton(AF_INET, "127.0.0.1", &serverAddr.sin_addr); // 也可指定IP// 4. 绑定套接字到地址if (bind(serverSocket, (sockaddr*)&serverAddr, sizeof(serverAddr)) == SOCKET_ERROR) {std::cerr << "绑定失败: " << WSAGetLastError() << std::endl;closesocket(serverSocket);WSACleanup();return 1;}std::cout << "套接字绑定到0.0.0.0:8080成功!" << std::endl;// 5. 开始监听if (listen(serverSocket, SOMAXCONN) == SOCKET_ERROR) {  // SOMAXCONN是最大队列长度std::cerr << "监听失败: " << WSAGetLastError() << std::endl;closesocket(serverSocket);WSACleanup();return 1;}std::cout << "正在监听端口8080..." << std::endl;// 6. 接受客户端连接sockaddr_in clientAddr;int clientAddrLen = sizeof(clientAddr);SOCKET clientSocket = accept(serverSocket, (sockaddr*)&clientAddr, &clientAddrLen);if (clientSocket == INVALID_SOCKET) {std::cerr << "接受连接失败: " << WSAGetLastError() << std::endl;closesocket(serverSocket);WSACleanup();return 1;}// 打印客户端信息char clientIP[INET_ADDRSTRLEN];inet_ntop(AF_INET, &clientAddr.sin_addr, clientIP, INET_ADDRSTRLEN);std::cout << "客户端连接来自: " << clientIP << ":" << ntohs(clientAddr.sin_port) << std::endl;// 7. 收发数据const char* welcomeMsg = "欢迎连接到TCP服务器!";if (send(clientSocket, welcomeMsg, strlen(welcomeMsg), 0) == SOCKET_ERROR) {std::cerr << "发送失败: " << WSAGetLastError() << std::endl;}char recvBuf[1024];int bytesReceived = recv(clientSocket, recvBuf, sizeof(recvBuf), 0);if (bytesReceived > 0) {recvBuf[bytesReceived] = '\0';  // 添加字符串结束符std::cout << "收到客户端消息: " << recvBuf << std::endl;}// 8. 清理资源closesocket(clientSocket);closesocket(serverSocket);WSACleanup();std::cout << "服务器已关闭。" << std::endl;return 0;
}

现代C++网络库

Boost.Asio

服务器端代码示例

#include <boost/asio.hpp>
#include <iostream>// 使用Boost.Asio命名空间
using namespace boost::asio;
using ip::tcp;int main() {try {// 1. 创建I/O执行上下文(事件循环核心)io_service io_service;// 2. 创建TCP接收器(监听8080端口)//    tcp::v4()表示IPv4,tcp::endpoint指定监听所有地址的8080端口tcp::acceptor acceptor(io_service, tcp::endpoint(tcp::v4(), 8080));std::cout << "服务器启动,监听端口8080..." << std::endl;// 3. 等待客户端连接(同步阻塞方式)tcp::socket socket(io_service);  // 创建套接字对象acceptor.accept(socket);         // 阻塞直到有客户端连接// 获取客户端端点信息std::string client_ip = socket.remote_endpoint().address().to_string();unsigned short client_port = socket.remote_endpoint().port();std::cout << "客户端连接来自: " << client_ip << ":" << client_port << std::endl;// 4. 向客户端发送欢迎消息std::string message = "Hello from Boost.Asio server!";boost::system::error_code error;write(socket, buffer(message), error);  // 同步写入数据// 检查发送是否成功if (error) {std::cerr << "发送失败: " << error.message() << std::endl;} else {std::cout << "消息已发送至客户端" << std::endl;}// 5. 接收客户端数据(可选)boost::asio::streambuf receive_buffer;read_until(socket, receive_buffer, "\n", error);  // 读取直到换行符if (!error) {std::istream is(&receive_buffer);std::string client_message;std::getline(is, client_message);std::cout << "收到客户端消息: " << client_message << std::endl;} else {std::cerr << "接收错误: " << error.message() << std::endl;}// 6. 关闭连接(RAII机制会自动关闭)} catch (std::exception& e) {std::cerr << "异常: " << e.what() << std::endl;return 1;}return 0;
}

 客户端测试代码示例

#include <boost/asio.hpp>
#include <iostream>using namespace boost::asio;
using ip::tcp;int main() {try {io_service io_service;// 1. 解析服务器地址和端口tcp::resolver resolver(io_service);tcp::resolver::query query("127.0.0.1", "8080");tcp::resolver::iterator endpoint_iterator = resolver.resolve(query);// 2. 创建并连接套接字tcp::socket socket(io_service);boost::asio::connect(socket, endpoint_iterator);// 3. 接收服务器消息boost::asio::streambuf receive_buffer;read_until(socket, receive_buffer, "\n");std::istream is(&receive_buffer);std::string server_message;std::getline(is, server_message);std::cout << "服务器说: " << server_message << std::endl;// 4. 发送响应消息std::string message = "Hello from client!\n";write(socket, buffer(message));} catch (std::exception& e) {std::cerr << "异常: " << e.what() << std::endl;}return 0;
}

# 编译服务器(假设保存为server.cpp)
g++ -std=c++11 server.cpp -o server -lboost_system

# 编译客户端
g++ -std=c++11 client.cpp -o client -lboost_system

# 运行(两个终端分别执行)
./server
./client

POCO网络库 

#include <Poco/Net/TCPServer.h>
#include <Poco/Net/TCPServerConnection.h>
#include <Poco/Net/TCPServerConnectionFactory.h>class MyConnection: public Poco::Net::TCPServerConnection {
public:MyConnection(const StreamSocket& s): TCPServerConnection(s) {}void run() {StreamSocket& ss = socket();ss.sendBytes("Hello from POCO server!", 22);}
};// 创建服务器
TCPServer srv(new TCPServerConnectionFactoryImpl<MyConnection>(), 8080);
srv.start();

HTTP客户端/服务器

使用cpp-httplib库

#include <httplib.h>// 服务器
httplib::Server svr;
svr.Get("/hi", [](const httplib::Request&, httplib::Response& res) {res.set_content("Hello World!", "text/plain");
});
svr.listen("localhost", 8080);// 客户端
httplib::Client cli("localhost", 8080);
auto res = cli.Get("/hi");
if (res && res->status == 200) {std::cout << res->body << std::endl;
}

http://www.xdnf.cn/news/14598.html

相关文章:

  • MySQL的Sql优化经验总结
  • 浅谈开发者重构的时机选择
  • 如何确定驱动480x320分辨率的显示屏所需的MCU主频
  • DBeaver数据库管理工具的简介、下载安装与优化配置
  • [IMX][UBoot] 02.源码目录
  • Python格式化工具推荐
  • Java中final修饰符
  • 第五章:执行计划分析 - 读懂MySQL的执行策略
  • 一款完美适配mobile、pad、web三端的博客网站UI解决方案
  • 《单光子成像》第六章 预习2025.6.15
  • 【驱动设计的硬件基础】I²C
  • 数据质量-如何构建高质量的大模型数据集
  • Understanding Human Hands in Contact at Internet Scale
  • Python基于Flask的医疗问句中的实体识别算法的研究(附源码,文档说明)
  • 【Dify系列】【Dify 核心功能】【应用类型】【五】【工作流】
  • C++ new知识点详解
  • 调和级数 敛散性
  • 一些杂想20250615
  • SAP顾问职位汇总(第24周)
  • 【Lean4编程入门】 Lean 4 中的 `inductive` 类型定义注解例子解析
  • 电商数据采集的技术分享
  • 【Bug:docker】--docker的wsl版本问题
  • 人工智能-准确率(Precision)、召回率(Recall) 和 F1 分数
  • 1、Java基础语法通关:从变量盒子到运算符魔法
  • NGINX Google Performance Tools 模块`ngx_google_perftools_module`
  • Mkdocs 阅读时间统计插件
  • 【第四十周】眼动追踪基础
  • 【C/C++】内核开发之进程调度大纲
  • Claude Code 是什么?
  • 【论文解读】LLaMA-Berry: 将“refine”当作动作解决数学推理