计算机网络-------------服务器代码实现(TCP/IP协议)

本文主要是介绍计算机网络-------------服务器代码实现(TCP/IP协议),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

服务器
我们把服务器比作是餐厅,那么我们要开一个餐厅,应该要做如下几步:
1.选择要开什么店(火锅店、KFC、烧烤…),(加载库:WSAStartUp())
2.地方找好了,那么我们就需要找个店长,帮我们看店(创建套接字:socket())
3.店长找好了,接下来我们需要让这个店长在店里工作,(绑定:bind())
4…一切准备工作完成之后,我们就需要接待客人了,观察是否来客人了(监听:listen())
5.来客人时,我们需要接待客人(接收连接:accept())
6.安排客人坐下后,我们就需要让客人点餐,(接收数据:recv())
7.客人点完餐,我们收到客人的菜单后,需要给客人回复(send())

8.客人吃完了,店长下班(关闭套接字:closesocket())
9.店长下班后,要把店关了(卸载库:WSACleanup())

以上就是一个服务器工作的过程,实现代码如下(封装到TCPNet类里了):对于一个类,显然有初始化,关闭操作,另外又需要收发数据,因此,TCPNet.h文件里代码如下

#pragma once
#include "INet.h"
#include "Kernel.h"class TCPNet:public INet
{
public:TCPNet(void);~TCPNet(void);
public:bool InitNetWork();void UnInitNetWork();bool SendData(SOCKET sockWaiter,char* szbuf,int nLen);void RecvData(SOCKET sockWaiter);
private:SOCKET m_sockListen;bool  m_bFlagQuit;
};

下面是TCPNet.cpp里的代码,首先是构造函数:

TCPNet::TCPNet()
{m_sockListen = NULL;m_bFlagQuit = true;
}

初始化函数:
1.加载库

//1.加载库WORD wVersionRequested;WSADATA wsaData;int err;/* Use the MAKEWORD(lowbyte, highbyte) macro declared in Windef.h */wVersionRequested = MAKEWORD(2, 2);err = WSAStartup(wVersionRequested, &wsaData);if (err != 0) {/* Tell the user that we could not find a usable *//* Winsock DLL.                                  */printf("WSAStartup failed with error: %d\n", err);return false;}/* Confirm that the WinSock DLL supports 2.2.*/
/* Note that if the DLL supports versions greater    */
/* than 2.2 in addition to 2.2, it will still return */
/* 2.2 in wVersion since that is the version we      */
/* requested.                                        */if (LOBYTE(wsaData.wVersion) != 2 || HIBYTE(wsaData.wVersion) != 2) {/* Tell the user that we could not find a usable *//* WinSock DLL.                                  */printf("Could not find a usable version of Winsock.dll\n");UnInitNetWork();return false;}elseprintf("The Winsock 2.2 dll was found okay\n");

加载库代码看着挺复杂,其实这部分很简单,只需要到查看帮助文档,然后进入帮助文档后,下面有示例代码,将需要的部分复制过来即可(注意头文件也要导入)
在这里插入图片描述2.创建套接字

//2.雇个店长--m_sockListen = socket(AF_INET,SOCK_STREAM,IPPROTO_TCP );if(m_sockListen == INVALID_SOCKET ){UnInitNetWork();return false;}

3.绑定

//3.店长去店里工作()--sockaddr_in  addrServer;addrServer.sin_family =AF_INET;addrServer.sin_addr.S_un.S_addr = 0;addrServer.sin_port = htons(8899);if(SOCKET_ERROR == bind(m_sockListen,(const SOCKADDR *)&addrServer,sizeof(sockaddr_in))){UnInitNetWork();return false;}

4.监听

//4.宣传--listen(); 一旦开始监听,一直处于监听状态if(SOCKET_ERROR  == listen(m_sockListen,10)){UnInitNetWork();return false;}

下面个是关闭函数实现:

void TCPNet::UnInitNetWork()
{m_bFlagQuit  =false;auto ite = m_lstThread.begin();while(ite != m_lstThread.end()){if(WAIT_TIMEOUT == WaitForSingleObject(*ite,100))TerminateThread(*ite,-1);CloseHandle(*ite);*ite = NULL;ite++;}if(m_sockListen){closesocket(m_sockListen);m_sockListen = NULL;}WSACleanup();
}

下面是接收数据函数实现

void TCPNet::RecvData(SOCKET sockWaiter)
{int nPackSize;char*pszbuf = NULL;while(m_bFlagQuit){//接收包大小int nRelReadNum = recv(sockWaiter,(char*)&nPackSize,sizeof(int),0);if(nRelReadNum <0){if(GetLastError() == 10054){auto ite = m_mapThreadIdToSocket.begin();while(ite != m_mapThreadIdToSocket.end()){if(sockWaiter == ite->first){closesocket(sockWaiter);m_mapThreadIdToSocket.erase(ite);return ;}ite++;}}continue;}//接收数据包pszbuf =new char[nPackSize];int noffset =0;while(nPackSize){nRelReadNum = recv(sockWaiter,pszbuf+noffset,nPackSize,0);noffset+= nRelReadNum;nPackSize -= nRelReadNum;}//处理//cout<<pszbuf<<endl;m_pKernel->DealData(pszbuf,sockWaiter);delete []pszbuf;pszbuf= NULL;}
}

因为,在使用TCP协议的时候可能会出现粘包问题,所以为了解决粘包问题,采用先发送包大小,在发送数据包的方式。在接收数据时由于不一定一次能接收完数据,所以是循环接收数据。
下面是发送数据代码:

bool TCPNet::SendData(SOCKET sockWaiter,char* szbuf,int nLen)
{if(!sockWaiter || !szbuf || nLen <0)return false;if(send(sockWaiter,(const char*)&nLen,sizeof(int),0) <=0)return false;if(send(sockWaiter,szbuf,nLen,0) <=0)return false;return true;
}

以上就是服务器实现代码,如果大家发现那里有问题,欢迎批评指正,请多多关注哈~

这篇关于计算机网络-------------服务器代码实现(TCP/IP协议)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/596620

相关文章

jupyter代码块没有运行图标的解决方案

《jupyter代码块没有运行图标的解决方案》:本文主要介绍jupyter代码块没有运行图标的解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录jupyter代码块没有运行图标的解决1.找到Jupyter notebook的系统配置文件2.这时候一般会搜索到

基于Python实现高效PPT转图片工具

《基于Python实现高效PPT转图片工具》在日常工作中,PPT是我们常用的演示工具,但有时候我们需要将PPT的内容提取为图片格式以便于展示或保存,所以本文将用Python实现PPT转PNG工具,希望... 目录1. 概述2. 功能使用2.1 安装依赖2.2 使用步骤2.3 代码实现2.4 GUI界面3.效

MySQL更新某个字段拼接固定字符串的实现

《MySQL更新某个字段拼接固定字符串的实现》在MySQL中,我们经常需要对数据库中的某个字段进行更新操作,本文就来介绍一下MySQL更新某个字段拼接固定字符串的实现,感兴趣的可以了解一下... 目录1. 查看字段当前值2. 更新字段拼接固定字符串3. 验证更新结果mysql更新某个字段拼接固定字符串 -

java实现延迟/超时/定时问题

《java实现延迟/超时/定时问题》:本文主要介绍java实现延迟/超时/定时问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java实现延迟/超时/定时java 每间隔5秒执行一次,一共执行5次然后结束scheduleAtFixedRate 和 schedu

Java Optional避免空指针异常的实现

《JavaOptional避免空指针异常的实现》空指针异常一直是困扰开发者的常见问题之一,本文主要介绍了JavaOptional避免空指针异常的实现,帮助开发者编写更健壮、可读性更高的代码,减少因... 目录一、Optional 概述二、Optional 的创建三、Optional 的常用方法四、Optio

在Android平台上实现消息推送功能

《在Android平台上实现消息推送功能》随着移动互联网应用的飞速发展,消息推送已成为移动应用中不可或缺的功能,在Android平台上,实现消息推送涉及到服务端的消息发送、客户端的消息接收、通知渠道(... 目录一、项目概述二、相关知识介绍2.1 消息推送的基本原理2.2 Firebase Cloud Me

Spring Boot项目中结合MyBatis实现MySQL的自动主从切换功能

《SpringBoot项目中结合MyBatis实现MySQL的自动主从切换功能》:本文主要介绍SpringBoot项目中结合MyBatis实现MySQL的自动主从切换功能,本文分步骤给大家介绍的... 目录原理解析1. mysql主从复制(Master-Slave Replication)2. 读写分离3.

Redis实现延迟任务的三种方法详解

《Redis实现延迟任务的三种方法详解》延迟任务(DelayedTask)是指在未来的某个时间点,执行相应的任务,本文为大家整理了三种常见的实现方法,感兴趣的小伙伴可以参考一下... 目录1.前言2.Redis如何实现延迟任务3.代码实现3.1. 过期键通知事件实现3.2. 使用ZSet实现延迟任务3.3

基于Python和MoviePy实现照片管理和视频合成工具

《基于Python和MoviePy实现照片管理和视频合成工具》在这篇博客中,我们将详细剖析一个基于Python的图形界面应用程序,该程序使用wxPython构建用户界面,并结合MoviePy、Pill... 目录引言项目概述代码结构分析1. 导入和依赖2. 主类:PhotoManager初始化方法:__in

springboot filter实现请求响应全链路拦截

《springbootfilter实现请求响应全链路拦截》这篇文章主要为大家详细介绍了SpringBoot如何结合Filter同时拦截请求和响应,从而实现​​日志采集自动化,感兴趣的小伙伴可以跟随小... 目录一、为什么你需要这个过滤器?​​​二、核心实现:一个Filter搞定双向数据流​​​​三、完整代码