C++简单的TCP/IP服务端与客户端(附完整代码)

一. TCP服务端

(一)创建一个TCP服务端

可大致分为以下5个步骤:

 1.初始化环境
 2.创建监听套接字
 3.监听套接字与IP地址及端口绑定
 4.监听套接字
 5.等待客户端连接

1. 初始化环境

WSADATA	wsaData;

WSAStartup(MAKEWORD(2, 2), &wsaData);

WSAStartup()函数的调用指明Windows Sockets API的版本号及获得特定Windows Sockets实现的细节,应用程序或DLL只能在一次成功的WSAStartup()调用之后才能调用进一步的Windows Sockets API函数。
该函数执行成功后返回0。

2. 创建监听套接字

m_listenSocket = socket(AF_INET, SOCK_STREAM, 0));

AF_INET:指定使用IPV4协议簇
SOCK_STREAM:字节流,数据有保障的传输
0:不希望指定协议
该函数执行成功返回一个新的套接字,否则返回:INVALID_SOCKET。

3. 监听套接字与IP地址及端口绑定

sockaddr_in sockadd = { 0, };

sockadd.sin_family = AF_INET;//IPV4协议簇
sockadd.sin_port = htons(m_uPort);//监听端口
sockadd.sin_addr.S_un.S_addr = htonl(INADDR_ANY);//监听本机任意IP
bind(m_listenSocket, (struct sockaddr*) & sockadd, sizeof(sockadd));

m_listenSocket:已经创建好的监听套接字
INADDR_ANY:如果机器存在多网卡,客户端连接任意一个IP地址均可建立通信
执行失败时返回: SOCKET_ERROR

4. 监听套接字

listen(m_listenSocket, 1);

m_listenSocket:绑定IP地址及端口的监听地址
1:指定最大允许同时连接的客户端数

5. 等待客户端连接

m_clientSocket = accept(m_listenSocket, (struct sockaddr*) & addr, &addrlen);

提取套接字m_listenSocket上挂起连接队列的第一个连接,然后返回新套接字m_clientSocket,如过要与客户端通信就需要通过m_clientSocket来发送或接收数据
如果暂时没有客户端连接过来,该函数将阻塞在此处,直到新连接到来才会返回。

(二)数据收发

1. 接收数据

const int iBufSize = 1024;
char recvBuf[iBufSize] = { 0, };
auto iRecvSize = recv(m_clientSocket, recvBuf, iBufSize, 0);//若不支持C++11及以上,auto改为int

m_clientSocket:收发数据时,均需要用到accept()返回的客户端套接字
recvBuf:接收数据缓冲区,收到的数据就保存在该数组中
iBufSize:recvBuf数组的字节长度,最多接收到的数据长度,该缓冲区也只能接收这么多,否则会溢出,造成程序异常
0:将数据从TCP从输入队列中剪切到recvBuf,且不做其他操作
返回值:如果没有发生错误,recv将返回接收到的字节数,recvBuf参数指向的缓冲区将包含接收到的数据。如果连接已经正常关闭,则返回值为零。否则,将返回一个SOCKET_ERROR值
如果套接字m_clientSocket上没有可用的传入数据,则recv调用阻塞并等待数据并不会返回,除非定义了其他阻塞规则,此处并没有指定

2. 发送数据

std::string strMsg = "hello";
send(m_clientSocket, strMsg.c_str(), strMsg.length(), 0);

m_clientSocket:收发数据时,均需要用到accept()返回的客户端套接字
strMsg.c_str():发送数据首地址
strMsg.length():数据长度
0:无特别指定调用
返回值:如果没有发生错误,send返回发送的字节总数,它可能小于len参数中请求发送的字节数。否则,将返回SOCKET_ERROR的值

(三)服务端代码

CTcpServer.h

//CTcpServer.h
#pragma once

#include <string>
#include <winsock2.h>

#pragma comment(lib,"ws2_32")//Standard socket API.

class CTcpServer
{
public:
	CTcpServer(std::string strIp, unsigned int uPort);
	virtual ~CTcpServer();

	//初始化网络服务端
	bool InitServer();

	//发送数据
	bool SendMsg(const std::string& strMsg);

	//接收数据并打印
	bool RecvMsg();

private:
	unsigned int m_uPort;//监听端口
	std::string m_strIp;//用于监听本机指定IP地址

	SOCKET m_listenSocket = NULL;//监听套接字
	SOCKET m_clientSocket = NULL;//客户端套接字
};

CTcpServer.cpp

//CTcpServer.cpp
#include <iostream>
#include "CTcpServer.h"

CTcpServer::CTcpServer(std::string strIp, unsigned int uPort) :
	m_strIp(strIp),
	m_uPort(uPort)
{
}

CTcpServer::~CTcpServer()
{
	if (m_clientSocket)
	{
		closesocket(m_clientSocket);
		m_clientSocket = NULL;
	}

	if (m_listenSocket)
	{
		closesocket(m_listenSocket);
		m_listenSocket = NULL;
	}

	WSACleanup();
}

bool CTcpServer::InitServer()
{
	WSADATA	wsaData;

	//1. 初始化环境
	if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0)
	{
		std::cout << "Init Windows Socket Failed!\n";
		return false;
	}

	//2. 创建监听套接字
	if ((m_listenSocket = socket(AF_INET, SOCK_STREAM, 0)) == INVALID_SOCKET)
	{
		std::cout << "Create socket failed!\n";
		return false;
	}

	//协议
	sockaddr_in sockadd = { 0, };

	sockadd.sin_family = AF_INET;//IPV4协议簇
	sockadd.sin_port = htons(m_uPort);//监听端口
	sockadd.sin_addr.S_un.S_addr = htonl(INADDR_ANY);//监听本机任意IP

	//3. 监听套接字与IP地址及端口绑定
	if (bind(m_listenSocket, (struct sockaddr*) & sockadd, sizeof(sockadd)) == SOCKET_ERROR)
	{
		closesocket(m_listenSocket);
		m_listenSocket = INVALID_SOCKET;
		std::cout << "Socket bind failed!\n";
		return false;
	}

	//4. 监听套接字
	if (listen(m_listenSocket, 1) == SOCKET_ERROR)
	{
		closesocket(m_listenSocket);
		m_listenSocket = INVALID_SOCKET;
		std::cout << "Socket listen failed!\n";
		return false;
	}

	sockaddr_in addr = { 0, };
	int addrlen = sizeof(addr);

	//5. 等待客户端连接
	m_clientSocket = accept(m_listenSocket, (struct sockaddr*) & addr, &addrlen);

	if (m_clientSocket == SOCKET_ERROR)
	{
		closesocket(m_clientSocket);
		m_clientSocket = INVALID_SOCKET;
		std::cout << "Socket accept failed!\n";
		return false;
	}

	return true;
}

bool CTcpServer::SendMsg(const std::string& strMsg)
{
	if (!m_clientSocket) return false;

	if (send(m_clientSocket, strMsg.c_str(), strMsg.length(), 0) != INVALID_SOCKET)
	{
		std::cout << "发送成功:" << strMsg << "\n";
		return true;
	}
	else
	{
		std::cout << "发送失败!\n";
		return false;
	}
}

bool CTcpServer::RecvMsg()
{
	if (!m_clientSocket) return false;

	const int iBufSize = 1024;
	char recvBuf[iBufSize] = { 0, };
	auto iRecvSize = recv(m_clientSocket, recvBuf, iBufSize, 0);//若不支持C++11及以上,auto改为int

	if (iRecvSize <= 0)
	{
		std::cout << "接收失败!\n";
		return false;
	}
	else
	{
		std::cout << "接收成功:" << recvBuf << "\n";
		return true;
	}
}
#include <iostream>
#include "CTcpServer.h"

int main()
{
    {
        CTcpServer tcpServer("127.0.0.1", 6005);

        if (!tcpServer.InitServer())
            getchar();

        for (int i = 0; i != 3; ++i)
        {
        	//接收数据成功后,向客户端返回"answer"
            if (tcpServer.RecvMsg())
            {
                std::string strAnswerMsg{ "answer" };

                tcpServer.SendMsg(strAnswerMsg);
            }
        }
    }//接收3次数据后出此大括号,tcpServer对象被析构,客户端连接被关闭
    
    return 0;
}

二. TCP客户端

创建一个TCP客户端

1. 初始化环境
2. 创建一个新的套接字
3. 建立连接

客户端的创建比较简单,只需要三步,创建好了之后就可以进行数据的收发了
短链接:客户端与服务端建立连接之后,进行短暂的收发数据之后即断开连接(closesocket())称为短链接
长连接:两者建立连接之后需要长时间保持该连接的成为长连接,在不需要数据交互时可以每隔一定时间其中一方发送一个心跳报文(内容可以是约定的任意值,比如字符串“heartBeat”)到另一方,另一方收到心跳报文后立即恢复一个应答(同样是约定的任意值),发送方在一定次数没有收到应答 或者 接收方一定次数没有收到心跳报文时,均可判定对方断线,此时可以关闭套接字或其他业务逻辑

还是直接上代码吧

CTcpClient.h

//CTcpClient.h
#pragma once

#include <string>
#include <winsock2.h>

#pragma comment(lib,"ws2_32")//Standard socket API.

class CTcpClient
{
public:
	CTcpClient(std::string strServerIp, unsigned uServerPort);
	virtual ~CTcpClient();

	//建立连接
	bool InitConnect();

	//发送数据
	bool SendMsg(const std::string& strMsg);

	//接收数据并打印
	bool RecvMsg();

private:
	SOCKET m_socket = INVALID_SOCKET;
	std::string m_strServerIp;//服务端监听IP地址
	unsigned int m_uServerPort = -1;//服务端监听端口
	struct addrinfo* m_servAddrInfo = NULL;//服务端地址结构链表
};


CTcpClient.cpp

//CTcpClient.cpp

#include <sstream>
#include <iostream>
#include <Ws2tcpip.h>
#include <mstcpip.h>
#include "CTcpClient.h"

CTcpClient::CTcpClient(std::string strServerIp, unsigned uServerPort) :
	m_strServerIp(strServerIp),
	m_uServerPort(uServerPort)
{
}

CTcpClient::~CTcpClient()
{
	if (m_socket != INVALID_SOCKET)
		closesocket(m_socket);

	WSACleanup();
	freeaddrinfo(m_servAddrInfo);
}

bool CTcpClient::InitConnect()
{
	WSADATA	wsaData;

	//1. 初始化环境
	if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0)
	{
		std::cout << "Init Windows Socket Failed!\n";
		return false;
	}

	addrinfo hints = { 0, };//协议无关(IPV4 or IPV6)

	hints.ai_flags = AI_NUMERICHOST;
	hints.ai_family = AF_UNSPEC;
	hints.ai_socktype = SOCK_STREAM;
	hints.ai_protocol = IPPROTO_TCP;

	std::stringstream ssPort;

	ssPort << m_uServerPort;

	//获取服务端地址结构
	if (getaddrinfo(m_strServerIp.c_str(), ssPort.str().c_str(), &hints, &m_servAddrInfo) != 0)
	{
		std::cout << "Get server addrInfo failed!\n";
		return false;
	}

	if (m_socket != INVALID_SOCKET)
		closesocket(m_socket);

	m_socket = INVALID_SOCKET;

	//2. 创建一个新的套接字
	if ((m_socket = socket(m_servAddrInfo->ai_family, m_servAddrInfo->ai_socktype, m_servAddrInfo->ai_protocol)) == SOCKET_ERROR)
		return false;

	//3. 建立连接
	int iResult = connect(m_socket, m_servAddrInfo->ai_addr, (int)m_servAddrInfo->ai_addrlen);

	if (iResult == SOCKET_ERROR)
	{
		iResult = WSAGetLastError();
		
		if (iResult != WSAEWOULDBLOCK)
		{
			std::cout << "Connect server failed!\n";
			closesocket(m_socket);
			m_socket = INVALID_SOCKET;
			return false;
		}
	}

	std::cout << "Connect Server succeed!\n";
	return true;
}

bool CTcpClient::SendMsg(const std::string& strMsg)
{
	if (!m_socket) return false;

	if (send(m_socket, strMsg.c_str(), strMsg.length(), 0) != INVALID_SOCKET)
	{
		std::cout << "发送成功:" << strMsg << "\n";
		return true;
	}
	else
	{
		std::cout << "发送失败!\n";
		return false;
	}
}

bool CTcpClient::RecvMsg()
{
	if (!m_socket) return false;

	const int iBufSize = 1024;
	char recvBuf[iBufSize] = { 0, };
	auto iRecvSize = recv(m_socket, recvBuf, iBufSize, 0);//若不支持C++11及以上,auto改为int

	if (iRecvSize <= 0)
	{
		std::cout << "接收失败!\n";
		return false;
	}
	else
	{
		std::cout << "接收成功:" << recvBuf << "\n";
		return true;
	}
}

#include <iostream>
#include "CTcpClient.h"

int main()
{
    {
        CTcpClient tcpClient("127.0.0.1", 6005);

        if (!tcpClient.InitConnect())
            getchar();

        std::string strAskMsg{ "ask" };

        for (int i = 0; i != 3; ++i)
        {
            if (tcpClient.SendMsg(strAskMsg))
            {
                tcpClient.RecvMsg();
            }
        }
    }
    
    getchar();
    
    return 0;
}

运行截图

在这里插入图片描述
有任何问题,欢迎留言

  • 21
    点赞
  • 102
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
### 回答1: 好的,下面是关于 C 语言下 Socket 组播服务端客户端代 码的示例: 服务端代码: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <sys/socket.h> #include <netinet/in.h> #include <unistd.h> #define PORT 8080 #define MAXLINE 1024 int main() { int sockfd; char buffer[MAXLINE]; char* message = "Hello, Multicast!"; struct sockaddr_in servaddr; // 创建套接字 sockfd = socket(AF_INET, SOCK_DGRAM, 0); memset(&servaddr, 0, sizeof(servaddr)); servaddr.sin_family = AF_INET; servaddr.sin_addr.s_addr = htonl(INADDR_ANY); servaddr.sin_port = htons(PORT); // 绑定套接字 if(bind(sockfd, (struct sockaddr*)&servaddr, sizeof(servaddr)) < 0) { perror("bind failed"); exit(EXIT_FAILURE); } // 循环发送数据 while(1) { if(sendto(sockfd, (const char*)message, strlen(message), MSG_CONFIRM, (const struct sockaddr*)&servaddr, sizeof(servaddr)) < 0) { perror("sendto failed"); exit(EXIT_FAILURE); } sleep(1); } close(sockfd); return 0; } ``` 客户端代码: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <sys/socket.h> #include <netinet/in.h> #include <unistd.h> #define PORT 8080 #define MAXLINE 1024 int main() { int sockfd; char buffer[MAXLINE]; struct sockaddr_in servaddr; // 创建套接字 sockfd = socket(AF_INET, SOCK_DGRAM, 0); memset(&servaddr, 0, sizeof(servaddr)); servaddr.sin_family = AF_INET; servaddr.sin_addr.s_addr = inet_addr("239.0.0.1"); servaddr.sin_port = htons(PORT); // 加入组播组 struct ip_mreq mreq; mreq.imr_multiaddr.s_addr = inet_addr("239.0.0.1"); mreq.imr_interface.s_addr = htonl(INADDR_ANY); if(setsockopt(sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq, sizeof(mreq)) < 0) { perror("setsockopt failed"); exit(EXIT_FAILURE); } // 循环接收数据 while(1) { memset(buffer, 0, MAXLINE); if(recvfrom(sockfd, (char*)buffer, MAXLINE, MSG_WAITALL, (struct sockaddr*)&servaddr, sizeof(servaddr)) < 0) { perror("recvfrom failed"); exit(EXIT_FAILURE); } printf("Message received: %s\n", buffer); } close(sockfd); return 0; } ``` 这段示例代码展示了如何创建基于 UDP 协议的 Socket 组播服务端客户端,其中服务端会不断向组播地址发送消息,而客户端会监听这个组播地址并接收到服务端发来的消息。 ### 回答2: 下面是一个使用Socket进行组播的服务端客户端的代码示例: 服务端代码: ```python import socket # 创建UDP socket,并绑定端口 server_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) server_socket.bind(('', 5000)) # 加入组播组 group_ip = '239.0.0.1' group_port = 5555 group = socket.inet_aton(group_ip) + socket.inet_aton('0.0.0.0') server_socket.setsockopt(socket.IPPROTO_IP, socket.IP_ADD_MEMBERSHIP, group) while True: # 接收客户端发送的数据 data, address = server_socket.recvfrom(1024) # 打印接收到的数据 print(f"接收到来自 {address} 的消息: {data.decode()}") # 将接收到的数据发送给组播组 server_socket.sendto(data, (group_ip, group_port)) # 关闭socket连接 server_socket.close() ``` 客户端代码: ```python import socket # 创建UDP socket,并设置套接字为广播类型 client_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) client_socket.setsockopt(socket.SOL_SOCKET, socket.SO_BROADCAST, 1) while True: # 获取用户输入消息 message = input("请输入要发送的消息(退出请输入q): ") if message == 'q': break # 将消息发送给组播组 client_socket.sendto(message.encode(), ('<broadcast>', 5000)) # 接收服务端发送的数据 data, address = client_socket.recvfrom(1024) # 打印接收到的数据 print(f"接收到来自 {address} 的回复: {data.decode()}") # 关闭socket连接 client_socket.close() ``` 以上代码中,服务端通过创建一个UDP socket,并绑定指定端口,然后加入指定的组播组。客户端通过创建一个UDP socket并设置为广播类型,然后发送消息给组播组。服务端接收到客户端发送的消息后,将消息发送给组播组,客户端收到组播消息后打印出来。通过这种方式,服务端客户端可以进行组播通信。 ### 回答3: 组播(Multicast)是一种基于UDP的通信协议,它允许一个服务器同时向多个客户端发送数据。下面是Socket组播服务端客户端的示例代码: 服务端代码: ```python import socket def multicast_server(): multicast_group = '224.0.0.1' server_address = ('', 10000) # 创建组播套接字 sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) # 设置套接字为可复用 sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) # 绑定地址和端口 sock.bind(server_address) # 将套接字加入组播组 group = socket.inet_aton(multicast_group) mreq = struct.pack('4sL', group, socket.INADDR_ANY) sock.setsockopt(socket.IPPROTO_IP, socket.IP_ADD_MEMBERSHIP, mreq) while True: data, address = sock.recvfrom(1024) print(f'Received message: {data.decode()} from {address}') # 关闭套接字 sock.close() if __name__ == '__main__': multicast_server() ``` 客户端代码: ```python import socket def multicast_client(): multicast_group = '224.0.0.1' server_address = ('', 10000) # 创建组播套接字 sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) # 设置套接字为可复用 sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) # 绑定地址和端口 sock.bind(server_address) # 将套接字加入组播组 group = socket.inet_aton(multicast_group) mreq = struct.pack('4sL', group, socket.INADDR_ANY) sock.setsockopt(socket.IPPROTO_IP, socket.IP_ADD_MEMBERSHIP, mreq) while True: message = input('Enter message to send: ') # 发送消息到组播组 sock.sendto(message.encode(), (multicast_group, 10000)) # 关闭套接字 sock.close() if __name__ == '__main__': multicast_client() ``` 以上是一个基本的组播示例,服务端通过创建组播套接字绑定地址和端口,并加入组播组。客户端同样创建组播套接字并加入组播组,然后从用户输入获取消息并发送到组播组。通过组播可以实现向多个客户端同时发送相同的消息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值