C++中网络通信

1. 网络通信流程

1.1 流式套接字(TCP)

在这里插入图片描述
上图对应代码流程为:
服务端

  1. 创建套接字(函数:socket
  2. 服务器绑定ip和端口号(函数:bind
  3. 监听,等待客户端连接(函数:listen
  4. 接收客户端连接请求(函数:accept
  5. 接收请求数据(函数:recv
  6. 发送响应数据(函数:send
  7. 关闭套接字(函数:closesocket

客户端

  1. 创建套接字(函数:socket
  2. 服务器绑定ip和端口号(函数:bind
  3. 请求连接(函数:connect
  4. 发送响应数据(函数:send
  5. 接收请求数据(函数:recv
  6. 关闭套接字(函数:closesocket

示例:
头文件
这里需要特别注意,windows下的头文件如下

//socket头文件
#include "winsock.h"
//socket库的lib
#pragma comment(lib,"ws2_32.lib")

而Linux下头文件则是:

#include <sys/types.h>
#include <sys/socket.h>
#include <netdb.h>

服务端创建基本流程

//创建套接字,第一个参数表示ip协议族,第二个参数表示数据流方式,第三个参数是TCP协议。
    SOCKET socksvr = socket(AF_INET,SOCK_STREAM,IPPROTO_TCP);
    if(INVALID_SOCKET == socksvr){
        return ;
    }

    struct sockaddr_in svraddr = {0};
    svraddr.sin_family = AF_INET;//代表Internet协议族
    //htons函数是将u_short型变量从主机字节顺序变为TCP/IP网络字节顺序
    svraddr.sin_port = htons(5678);
    //htonl函数是将u_long型变量从主机字节顺序变为TCP/IP网络字节顺序
    svraddr.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
    //绑定,将服务器套接字与服务器套接字地址绑定
    bind(socksvr,(struct sockaddr *)&svraddr,sizeof(svraddr));//指定名字,类型,长度。绑定套接字
    //监听
    listen(socksvr,SOMAXCONN);//第一个参数是套接字,第二个参数是等待连接队列的最大长度
    //建立客户端套接字地址,主要是为了接受客户端返回参数之用
    struct sockaddr_in clientaddr = {0};
    int nlen = sizeof(clientaddr);

    //以下是建立客户端套接字并建立连接函数,有一个确认过程
    SOCKET sockclient = accept(socksvr,(struct sockaddr *)&clientaddr,&nlen);
    //先接收后发送,由上面知,数据已在sockclient中,我们只需读此结构便可知晓数据
    CHAR szText[100] = {0};
   //接收缓冲区数据 
	recv(sockclient,szText,100,0); //接收函数,一直处于侦听模式,等待服务器端发送数据的到来。
	printf("%s\n",szText);
	CHAR szSend[100] = "Hello Client";
	send(sockclient,szSend,sizeof(szSend),0);//发送函数。
	closesocket(sockclient);
	closesocket(socksvr);

recv函数的返回值是接收的数据的字节数。在判断客户端是否已经断开连接的时候,可以根据recv的返回值判断,如果recv的返回值小于等于0,则客户端已经断开了连接。

客户端的流程和服务端基本一样,

//创建socket
	SOCKET sockclient = socket(AF_INET,SOCK_STREAM,IPPROTO_TCP);
	if(INVALID_SOCKET == sockclient)
	{ 
		return;
    }
	//连接服务器,建立服务器端套接字地址
	struct sockaddr_in addr = {0};
	addr.sin_family = AF_INET;
	addr.sin_port = htons(5678);
	//对于inet_addr()函数,它是把“xxx.xxx.xxx.xxx”形式表示的IPV4地址,转换为IN_ADDR结构体能够
	//接收的形式(unsigned long型,因为IN_ADDR结构体中的负责接收的S_addr成员变量的类型是unsigned long型)
	addr.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");//本机ip
	//向服务器发出连接请求,当然我们也可以通过connet函数的返回值判断到底有无连接成功。
	int iRetVal = connect(sockclient,(struct sockaddr*)&addr,sizeof(addr));
	//后面发送数据和接收数据和服务端一样,这里省略

1.2 数据报套接字(UDP)

在这里插入图片描述
上图对应代码流程为:
服务端

  1. 创建套接字(函数:socket
  2. 服务器绑定ip和端口号(函数:bind
  3. 接收请求数据(函数:recvfrom
  4. 发送响应数据(函数:sendto
  5. 关闭套接字(函数:closesocket

客户端

  1. 创建套接字(函数:socket
  2. 服务器绑定ip和端口号(函数:bind
  3. 发送响应数据(函数:sendto
  4. 接收请求数据(函数:recvfrom
  5. 关闭套接字(函数:closesocket

示例:
头文件

//socket头文件
#include "winsock.h"
//socket库的lib
#pragma comment(lib,"ws2_32.lib")

服务端创建基本流程

//创建套接字,第一个参数表示ip协议族,第二个参数表示数据流方式,第三个参数是TCP协议。
    SOCKET socksvr = socket(AF_INET,SOCK_STREAM,IPPROTO_TCP);
    if(INVALID_SOCKET == socksvr){
        return ;
    }

    struct sockaddr_in svraddr = {0};
    svraddr.sin_family = AF_INET;//代表Internet协议族
    //htons函数是将u_short型变量从主机字节顺序变为TCP/IP网络字节顺序
    svraddr.sin_port = htons(5678);
    //htonl函数是将u_long型变量从主机字节顺序变为TCP/IP网络字节顺序
    svraddr.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
    //绑定,将服务器套接字与服务器套接字地址绑定
    bind(socksvr,(struct sockaddr *)&svraddr,sizeof(svraddr));//指定名字,类型,长度。绑定套接字
    CHAR szRecv[100] = {0};
	struct sockaddr_in clientaddr = {0};
	int nLen = sizeof(clientaddr);
	/*下面函数前四个参数同TCP接收数据函数recv()一样,后两个中,一个是返回发送*******/
	/*数据地址的主机的地址,包括IP地址以及端口号,最后一个为地址长度的地址。*******/
	/*此函数中,先是服务器端的套接字,后是客户端的地址*/
 	//从后往前读此函数
	recvfrom(socksvr,szRecv,100,0,(struct sockaddr*)&clientaddr,&nLen);//构造ip地址
	printf("%s\n",szRecv);
	//注1:该程序也可以向客户端发送数据。
	//注2:服务器端中,必须也是先接收后发送,不然,我们无法知道客户端的地址。下面函数中clientaddr已知晓
	CHAR szSend[100] = "hello udp client";
	//从前往后读此函数
	sendto(socksvr,szSend,100,0,(struct sockaddr*)&clientaddr,nLen);//发送时构造ip地址和端口。
	//关闭socket
	closesocket(socksvr);

客户端基本一致,只是bind可以省略

//创建SOCKET ,ip协议族,数据报方式,udp协议。
	SOCKET sockclient = socket(AF_INET,SOCK_DGRAM,IPPROTO_UDP);
	if(INVALID_SOCKET == sockclient)
	{
	     return ;
	}
	//数据收发,服务器端套接字地址
struct sockaddr_in svraddr = {0};
svraddr.sin_family = AF_INET;
svraddr.sin_port = htons(5780);
svraddr.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");//指定服务器端的IP与端口。
CHAR szSend[100] = "hello udp server";
/*此函数先是客户端的套接字,然后是服务器端地址*/
//简单理解为:从函数前面的客户端套接字的发送数据缓存区中将数发送给服务器端地址
sendto(sockclient,szSend,100,0,(struct sockaddr*)&svraddr,sizeof(svraddr));//发送时构造ip地址和端口。

//注:该程序也可以接收服务器端回传的数据。
CHAR szRecv[100];  
//简单理解为:从函数后面的服务器端地址中取数到客户端套接字的接收缓冲区szRecv中
int len = sizeof(svraddr);
recvfrom(sockclient,szRecv,100,0,(struct sockaddr*)&svraddr,&len);
printf("%s \n",szRecv);
//关闭socket
closesocket(sockclient);

参考博客
C++ TCP/IP通信

  • 0
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 实现网络通信监控需要通过以下几个步骤来实现: 1. 监听网络通信:采用代理服务器的方式,将网络流量导向到一个间设备上,然后在该设备上监测网络通信。可以使用网络抓包工具,如Wireshark,来捕获并分析网络数据包。 2. 分析网络数据:对捕获到的网络数据包进行解析和分析,提取关键信息,如源地址、目标地址、协议类型、端口号等。可以采用数据包解析库,如Scapy或Pcap来辅助分析。 3. 实时监控:将分析得到的网络信息进行处理,并实时监控网络数据的流向和传输状态。可以使用编程语言,如Python或Java,利用网络编程库,如socket或scoket.io来开发监控程序。 4. 报警机制:当监测到异常或危险情况时,通过邮件、短信或弹窗等方式进行即时报警通知。可以设置关键指标的阈值,并通过条件判断来触发报警。 5. 数据记录与分析:将监测到的网络数据进行记录,并可以对其进行分析和统计,以便后续分析网络流量、检测网络攻击或优化网络性能。 6. 可视化展示:将监测到的网络数据以可视化的方式展示在监控界面上,可以使用图表、表格、地图等形式展示相关信息,方便用户查看和理解。 总之,实现网络通信监控需要对网络数据进行分析和处理,并利用报警机制和数据分析来提升网络安全性和性能。同时,合理的可视化展示可以帮助用户直观地了解网络通信的情况。 ### 回答2: C实现网络通信监控可以通过以下步骤实现。 首先,需要使用C语言的网络编程库,例如sockets库,以便与网络通信进行交互。可以使用socket()函数创建一个套接字,并使用bind()函数将其绑定到特定的IP地址和端口号上。 接下来,可以使用listen()函数将套接字设置为监听状态,以便接受传入的网络连接。可以创建一个循环,使用accept()函数接受连接请求,并为每个连接创建一个新的线程或进程进行处理。 在每个连接,可以使用read()函数从套接字读取数据,或使用write()函数将数据写入套接字。可以根据需要制定通信协议和数据格式。 为了实现网络通信的监控,可以添加适当的监控机制。例如,可以使用select()函数来监视套接字的状态变化,以便在有新的数据可读或写入时及时做出相应的处理。 此外,还可以使用日志记录或打印输出等方式,将网络通信过程的各个步骤和数据进行记录,以便进行监控和故障排除。可以使用文件操作函数如fprintf()来写入日志文件,或使用printf()函数将日志信息打印到控制台。 最后,为了确保网络通信监控的稳定性和可靠性,还可以添加错误处理机制。例如,可以检测套接字连接的异常或错误,及时关闭连接并释放资源,以避免潜在的资源泄露或崩溃。 总之,通过以上步骤和技术,可以使用C语言实现网络通信监控,并对网络通信进行有效地监控和管理。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值