1. 服务端

#include <iostream>
#include <signal.h>
#include <forward_list>
#include <thread>
#include <WinSock2.h>
#include <Windows.h>
#pragma comment(lib, "Ws2_32.lib") class Server {
constexpr static int DEFAULT_BACKLOG = 10;
public:
Server(): _listener(INVALID_SOCKET), _should_exit(false) {}
~Server() {
for (auto it = _threads.begin(); it != _threads.end(); it++) {
it->join();
}
} // 判断 server 是否有效
operator bool() const {
return _listener != INVALID_SOCKET;
} // 监听并接受客户端连接
void listen_and_serve(unsigned short port, int backlog= DEFAULT_BACKLOG) {
// 初始化 WSA
WORD socketVersion = MAKEWORD(2, 2);
WSADATA wsaData;
if (WSAStartup(socketVersion, &wsaData) != 0) {
std::cerr << "cannot start up\n";
return;
} // 创建套接字
_listener = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (_listener == INVALID_SOCKET) {
std::cerr << "cannot create listener\n";
return;
} // 设置套接字选项
int value = 1;
if (setsockopt(_listener, SOL_SOCKET, SO_REUSEADDR, (char*)&value, sizeof(value)) == SOCKET_ERROR) {
std::cerr << "cannot set option: reuse_address\n";
} // 绑定地址
memset(&addr, 0, sizeof(sockaddr_in));
sockaddr_in addr;
addr.sin_family = AF_INET;
addr.sin_port = htons(port);
addr.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
if (bind(_listener, (LPSOCKADDR)&addr, sizeof(addr)) == SOCKET_ERROR) {
std::cerr << "bind failed\n";
return;
} // 监听连接
if (listen(_listener, backlog) == SOCKET_ERROR) {
std::cerr << "listen failed\n";
return;
} SOCKET conn;
std::cout << "listening ...\n"; // 接受连接
while (!_should_exit) {
conn = accept(_listener, NULL, NULL);
if (conn == INVALID_SOCKET) {
std::cerr << "accept failed\n";
} // 启动线程去处理客户端连接
_threads.push_front(std::thread(SetServer::handle_connection, conn));
}
} // 关闭、完成善后工作
void close() {
_should_exit = true;
closesocket(_listener);
WSACleanup();
} // 处理客户端连接
static void handle_connection(SOCKET conn) {
char buf[16] = { 0 };
int n = recv(conn, buf, sizeof(buf), 0);
send(conn, buf, n, 0);
closesocket(conn);
} private:
SOCKET _listener;
std::forward_list<std::thread> _threads;
bool _should_exit;
}; Server server; // 信号处理器
void handler(int signum) {
std::cout << "exit ...\n";
if (server) {
server.close();
}
exit(EXIT_SUCCESS);
} int main() {
// 处理 ctrl+c
signal(SIGINT, handler); server.listen_and_serve(6666);
}

2. 客户端

#pragma once

#include <iostream>
#include <WS2tcpip.h>
#include <WinSock2.h>
#include <Windows.h>
#pragma comment(lib, "Ws2_32.lib") const char* SERVER_IP = "127.0.0.1";
const short SERVER_PORT = 6666; int main() {
// 初始化 WSA
WORD socketVersion = MAKEWORD(2, 2);
WSADATA wsaData;
if (WSAStartup(socketVersion, &wsaData) != 0) {
std::cerr << "cannot start up\n";
return 1;
} // 创建套接字
SOCKET sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (sock == INVALID_SOCKET) {
std::cerr << "cannot create socket\n";
return 1;
} // 发起连接
sockaddr_in addr;
addr.sin_family = AF_INET;
addr.sin_port = htons(SERVER_PORT);
inet_pton(AF_INET, SERVER_IP, &addr.sin_addr);
if (connect(sock, (LPSOCKADDR)&addr, sizeof(addr)) == SOCKET_ERROR) {
std::cerr << "connect failed\n";
return 1;
} // 发送、接收数据
const char* msg = "hello";
char buf[16] = { 0 };
send(sock, msg, strlen(msg), 0);
int n = recv(sock, buf, sizeof(buf), 0);
std::cout << buf << std::endl; closesocket(sock);
return 0;
}

参考资料

https://blog.csdn.net/weixinhum/article/details/81742248

网络编程 — Windows TCP服务端和客户端的更多相关文章

  1. 网络编程 — Linux TCP服务端和客户端

    1. 服务端 #include <stdlib.h> #include <string.h> #include <errno.h> #include <sig ...

  2. 基于Select模型的Windows TCP服务端和客户端程序示例

    最近跟着刘远东老师的<C++百万并发网络通信引擎架构与实现(服务端.客户端.跨平台)>,Bilibili视频地址为C++百万并发网络通信引擎架构与实现(服务端.客户端.跨平台),重新复习下 ...

  3. python网络编程:socket、服务端、客户端

    本文内容: socket介绍 TCP: 服务端 客户端 UDP: 服务端 客户端 首发时间:2018-02-08 01:14 修改: 2018-03-20 :重置了布局,增加了UDP 什么是socke ...

  4. python网络编程-TCP服务端的开发

    #TCP服务端开发 2 #方法说明 3 """ 4 bind(host,port)表示绑定端口号,host是ip地址,ip地址一般不进 行绑定,表示本机的任何一个ip地址 ...

  5. [javaSE] 网络编程(TCP服务端客户端互访阻塞)

    客户端给服务端发送数据,服务端收到数据后,给客户端反馈数据 客户端: 获取Socket对象,new出来,构造参数:String的ip地址,int的端口号 调用Socket对象的getOutputStr ...

  6. TCP/IP网络编程之多线程服务端的实现(二)

    线程存在的问题和临界区 上一章TCP/IP网络编程之多线程服务端的实现(一)的thread4.c中,我们发现多线程对同一变量进行加减,最后的结果居然不是我们预料之内的.其实,如果多执行几次程序,会发现 ...

  7. TCP/IP网络编程之多进程服务端(二)

    信号处理 本章接上一章TCP/IP网络编程之多进程服务端(一),在上一章中,我们介绍了进程的创建和销毁,以及如何销毁僵尸进程.前面我们讲过,waitpid是非阻塞等待子进程销毁的函数,但有一个不好的缺 ...

  8. 【转】TCP/UDP简易通信框架源码,支持轻松管理多个TCP服务端(客户端)、UDP客户端

    [转]TCP/UDP简易通信框架源码,支持轻松管理多个TCP服务端(客户端).UDP客户端 目录 说明 TCP/UDP通信主要结构 管理多个Socket的解决方案 框架中TCP部分的使用 框架中UDP ...

  9. swoole创建TCP服务端和客户端

    服务端: server.php <?php //创建Server对象,监听 127.0.0.1:9501端口    $serv = new swoole_server("127.0.0 ...

随机推荐

  1. 超简单的 Docker部署 SpringBoot项目 步骤

    很久之前就用过,一直没有好好写篇博客,今天就总结一下 创建一个 SpringBoot项目 创建一个SpringBoot项目并打成jar包,结构如图 编写 Dockerfile文件 FROM java: ...

  2. C# 编译机器码过程原理之再谈反射

    一.引言 我们知道在Java中有虚拟机,代码运行时虚拟机把Java语言编译成与机器无关的字节码,然后再把字节码编译成机器指令执行,那么在.NET中程序是如何运行的呢? 其实运行原理是一样的,.NET中 ...

  3. apk获取sha1值的方法

    安卓应用都使用一个签名证书(.keystore或.jks文件)签名,签名证书可以由jdk生成.证书生成后,证书就有其sha1值. 使用此证书打包后的安卓应用,也有其一样的sha1值.md5值和sha2 ...

  4. 如何将离线计算业务的成本降低65%——弹性容器服务EKS「竞价实例」上线

    在容器化的应用场景中,大数据计算是其中很大并且业务应用在逐渐增加的一个热门领域,包括越来越多的人工智能企业,都在使用容器技术来支持业务中的大量计算任务.降低成本.提升资源利用率也是当前这部分用户非常期 ...

  5. Shiro认证详解

    Shiro shiro是一个java的安全框架 官网地址 http://shiro.apache.org/ 目录 Shiro Shiro综述 过滤器 注解 整合Shiro 1. 配置SecurityM ...

  6. Oracle-序列-存储过程-视图-索引-触发器

    课程介绍 1. 约束(掌握) 2. 序列(掌握) 3. 索引(掌握) 4. 视图(掌握) 5. 存储过程(掌握) 6. 自定义函数(掌握) 7. 触发器(掌握) 数据库对象的命名规则 1.对象名称必须 ...

  7. sql文件转换为excel文件

    最近经常需要把sql整理成excel,本人比较懒,所以写一个小工具,用到了jxl包.以前没有接触过,正好了解一下. 一.基础知识       jxl操作excel包括对象 Workbook,Sheet ...

  8. 1001 害死人不偿命的(3n+1)猜想 (15分)

    卡拉兹(Callatz)猜想: 对任何一个正整数 n,如果它是偶数,那么把它砍掉一半:如果它是奇数,那么把 (3n+1) 砍掉一半.这样一直反复砍下去,最后一定在某一步得到 n=1.卡拉兹在 1950 ...

  9. ES6模板字符串及字符串的扩展方法

    一.ES6模板字符串 传统定义字符串的方式是: const str='hello es2015,this is a string' ES6新增了一种定义字符串的方式用反引号进行标识 const str ...

  10. 微信小程序--投票小程序设计与实现(图片、视频发布、分组、审核、排名)

    投票微信小程序设计与实现(图片.视频发布.分组.审核.排名) ​ 之前接到一个需求,设计一个类似H5 投票系统之类的小程序,我绞尽脑汁,冥思苦想,最后终于做了出来. 再次感谢 @文晓港 的ColorU ...