您的位置:首页 > 汽车 > 时评 > 有限责任公司公司章程范本_网络营销以什么为基础_外贸营销网站建设介绍_东莞网站推广营销网站设计

有限责任公司公司章程范本_网络营销以什么为基础_外贸营销网站建设介绍_东莞网站推广营销网站设计

2025/1/8 0:58:10 来源:https://blog.csdn.net/qq_59940419/article/details/144429845  浏览:    关键词:有限责任公司公司章程范本_网络营销以什么为基础_外贸营销网站建设介绍_东莞网站推广营销网站设计
有限责任公司公司章程范本_网络营销以什么为基础_外贸营销网站建设介绍_东莞网站推广营销网站设计

文章目录

  • 前言
  • 一、异步I/O模型
  • 二、多线程
  • 三、利用异步I/O模型和多线程解决阻塞问题
  • 四、实现客户端队列
  • 总结


前言

C++打造局域网聊天室第八课: 异步I/O模型及多线程


一、异步I/O模型

解决阻塞与非阻塞的好方法就是利用Windows提供的各种异步I/O模型
select选择模型
WSAAsyncSelect异步选择模型
WSAEventSelect事件选择模型
重叠I/O模型
完成端口模型

这里使用select选择模型

int select(int nfds, // 忽略,只是为了与Berkeley套接字兼容。一般调用函数时直接写0即可fd_set* readfds, // 可读性Socket集合,比较常用。数据可读、连接已经关闭、重启或中断、如果listen函数已经调用,且有一个连续未处理,那么调用accept函数将会成功fd_set* writefds, // 可写性Socket集合,数据能够发送、如果一个非阻塞连续调用正在被处理,连接已经成功fd_set* exceptfds, // 异常性Socket集合,如果一个非阻塞连续调用正在被处理,连接试图失败、OOB数据可读const struct timeval* timeout //超时时间
);

封装了一个函数:

// 第一个参数接受一个socket,想要用select监视的套接字句柄;第二个参数为超时时间;第三个参数为监视可读性还是可写性
BOOL SOCKET_Select(SOCKET hSocket, int nTimeOut, BOOL bRead) 
{fd_set fdset; // 定义select函数所需的fd_set对象结构timeval tv; // 定义超时时间对象FD_ZERO(&fdset);//对fdset结构进行清零操作FD_SET(hSocket, &fdset); //把传进来的socket放入fdset结构中nTimeOut = nTimeOut > 1000 ? 1000 : nTimeOut; // 最多只等1秒tv.tv_sec = 0; // 以秒为单位tv.tv_usec = nTimeOut; //以毫秒为单位int iRet = 0;if (bRead) // 可读性{iRet = select(0, &fdset, NULL, NULL, &tv);}else //可写性{iRet = select(0, NULL, &fdset, NULL, &tv);}if (iRet <= 0) // select函数返回值小于等于0说明有错误发生或者超时,超时即为即便等待了nTimeOut时间,socket仍旧没有返回状态{return FALSE;}else if(FD_ISSET(hSocket, &fdset)) // 判断传进来的socket是否在返回的socket集合当中,表示是否可读或可写{return TRUE;}return FALSE;

说明:select函数其实就是去缓冲区看一眼,看看有没有东西,如果有就返回其Socket的集合。之后就可以根据具体状态来调用相应函数。

二、多线程

1.进程是线程的容器,一个进程中至少要有一个主线程,线程才是真正执行代码的主体。
2.线程可以“并发”运行,从而提高程序的执行效率。在单核处理器计算机上,多线程表现为一个线程处理一会,保存状态,取执行另外线程,来回切换执行。
3.不要没有限制的去开线程,线程过多会使得CPU的调度切换线程的时间增大,结果适得其反。
4.线程间的同步问题:虽然Windows为我们提供了好多方法实现线程同步,但还是要小心谨慎,切勿粗心大意,否则程序出问题会不知道问题在哪。
5.如果做服务器确实需要处理成千上万个客户请求,可以使用线程池方面知识。
6.创建线程的API函数:CreateThread();MFC中创建线程的函数:AFxBeginThread(),最终调用的也是CreateThread,只是将一些常用参数赋上默认的值。
在这里插入图片描述
lpsa
安全属性的指针,表示安全级别。默认NULL

dwStackSize
线程的堆栈大小,默认0

pfnThreadProc
线程函数的地址,线程要去执行的代码需要依赖该函数

pvParam
传递给线程过程的参数,即线程函数的地址参数

dwCreationFlags
创建线程的初始化状态 (如CREATE_SUSPENDED创建线程后不让其马上运行,不需要的话写0即可).

pdwThreadId
返回值,返回线程的ID,不关心的话可以传递NULL

三、利用异步I/O模型和多线程解决阻塞问题

上节课采用点击开启服务器后的响应函数中直接写新建Socket、绑定和监听等操作,产生了阻塞现象,这里我们将上节课的代码注释掉,利用利用异步I/O模型和多线程解决阻塞问题。

在点击开启服务器后创建一个线程,在线程函数中运行上述工作。如果阻塞会阻塞新创建的线程,而维持消息界面的主线程不会阻塞。由于CreateThread函数的返回值是一个句柄,首先在chartroomDlg.h头文件中声明一个句柄,然后在chartroomDlg.cpp源文件中CchartroomDlg类的构造函数中初始化句柄。
在这里插入图片描述
在这里插入图片描述
然后可以创建新线程,注释或删除上节课的代码后在函数OnBnClickedButton3()中继续实现
在这里插入图片描述

这里的线程函数地址需要我们去实现,由于该程序服务器端和客户端集于一身,为了避免混淆,新创建一个源文件Server.cpp完成服务端的相关代码

#include "pch.h"
#include "framework.h"
#include "chartroom.h"
#include "chartroomDlg.h"
#include "afxdialogex.h"// 第一个参数接受一个socket,想要用select监视的套接字句柄;第二个参数为超时时间;第三个参数为监视可读性还是可写性
BOOL SOCKET_Select(SOCKET hSocket, int nTimeOut, BOOL bRead)
{fd_set fdset; // 定义select函数所需的fd_set对象结构timeval tv; // 定义超时时间对象FD_ZERO(&fdset);//对fdset结构进行清零操作FD_SET(hSocket, &fdset); //把传进来的socket放入fdset结构中nTimeOut = nTimeOut > 1000 ? 1000 : nTimeOut; // 最多只等1秒tv.tv_sec = 0; // 以秒为单位tv.tv_usec = nTimeOut; //以毫秒为单位int iRet = 0;if (bRead) // 可读性{iRet = select(0, &fdset, NULL, NULL, &tv);}else //可写性{iRet = select(0, NULL, &fdset, NULL, &tv);}if (iRet <= 0) // select函数返回值小于等于0说明有错误发生或者超时,超时即为即便等待了nTimeOut时间,socket仍旧没有返回状态{return FALSE;}else if (FD_ISSET(hSocket, &fdset)) // 判断传进来的socket是否在返回的socket集合当中,表示是否可读或可写{return TRUE;}return FALSE;
}DWORD WINAPI ListenThreadFunc(LPVOID pParam)
{CchartroomDlg* pChartRoom = (CchartroomDlg*)pParam; // 将参数强制转化为主对话框类,以便使用主对话框类的一些成员变量ASSERT(pChartRoom != NULL);// 新建pChartRoom->m_ListenSock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);if (pChartRoom->m_ListenSock = INVALID_SOCKET) // 如果新建失败{AfxMessageBox(_T("新建SOCKET失败!"));return FALSE;}int iPort = pChartRoom->GetDlgItemInt(IDC_EDIT7);if (iPort <= 0 || iPort > 65535) // 对端口值进行判断{AfxMessageBox(_T("请输入合适的端口:1-65535"));goto __Error_End;}sockaddr_in service;service.sin_family = AF_INET; //与新建socket第一个参数的值一样service.sin_addr.s_addr = INADDR_ANY; //获取本机所有可能得到的IP,一般有几块网卡就有几个IP//cout << service.sin_addr.s_addr;service.sin_port = htons(iPort); //将端口传递给sin_port成员,htons为字节顺序转换函数,利用该函数是因为常用的CUP字节顺序与网络字节顺序相反// 例如地址0x12345678,host:0x78 0x56 0x34 0x12; net:0x12 0x34 0x56 0x78。h为host(主机),n为network。htons即为将主机的字节顺序转化为net顺序if (bind(pChartRoom->m_ListenSock, (sockaddr*)&service, sizeof(sockaddr_in)) == SOCKET_ERROR)// 第一个参数为需要绑定的socket;第二个参数为一个sockaddr_in结构,需要进行一个强制类型转换;第三个参数为该结构的大小{AfxMessageBox(_T("绑定端口失败!"));goto __Error_End;}// 监听:第一个参数为传递的socket,第二个参数为监听序列中允许同时存在的客户端最大数目if (listen(pChartRoom->m_ListenSock, 5) == SOCKET_ERROR){AfxMessageBox(_T("监听失败!"));goto __Error_End;}// 利用循环的目的:服务器为一对多客户端,不能只接受一个客户端的连接而不管其他客户端。因此利用循环来接受客户端连接while (1){// 利用异步I/O模型来使用acceptif (SOCKET_Select(pChartRoom->m_ListenSock, 100, TRUE))// 看一下队列中是否存在客户端连接请求{sockaddr_in clientAddr;int iLen = sizeof(sockaddr_in);// 第一个参数为socket,第二个参数为一个sockaddr_in结构,需要进行一个强制类型转换,第三个参数为该结构的大小的地址。// 返回的socket为服务端与接受的客户端通信的socket,即利用accsocket进行与其他客户端的通信,m_ListenSock只是在服务器端进行监听SOCKET accSock = accept(pChartRoom->m_ListenSock, (struct sockaddr *)&clientAddr , &iLen);if (accSock == INVALID_SOCKET){continue;}// 其他操作Sleep(100); // 让其他程序使用cpu,避免一直被循环占用}}__Error_End:closesocket(pChartRoom->m_ListenSock);return TRUE;
}

这里上述的func要替换为Server.cpp中的函数名ListenThreadFunc,并且由于此时函数的定义和使用不在一个源文件中,这里添加一个头文件Server.h,在Server.h中声明一下函数ListenThreadFunc,并在chartroomDlg.cpp中include头文件Server.h,这样既可以在源文件chartroomDlg.cpp中使用函数ListenThreadFunc()。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
同时,由于在函数ListenThreadFunc中将参数制转化为主对话框类,因此CreateThread的第四个参数要传入this指针,即传入自己的地址。
在这里插入图片描述

四、实现客户端队列

为了实现一个客户端发送的消息可以传达给其他多个客户端。

每个客户端对应一个结点,此结点包含客户端的一些信息,如IP,Socket句柄,线程句柄等。
客户端结点可以定义一个类,直接在Server.h中定义

class CClientitem {
public:CString m_surlp; // 客户端的IPSOCKET m_Socket; // 客户端的socketHANDLE hThread; // 客户端的线程句柄CchartroomDlg* m_pMainWnd;CClientitem() // 构造函数{m_pMainWnd = NULL;m_Socket:INVALID_SOCKET;hThread = NULL;}};

存储客户端结点的队列有很多方法,如自己实现链表、STL库等,这里利用MFC提供的CArray模板类实现。

在chartroomDlg.h头文件中定义,这里同样要注意:由于使用了在Server.h中定义的类CClientitem,因此要在chartroomDlg.h中包含Server.h头文件,
在这里插入图片描述
在这里插入图片描述

同时,由于在Server.h头文件中定义类CClientitem时使用了CchartroomDlg类,因此要在Server.h头文件中声明一下类CchartroomDlg
在这里插入图片描述


总结

C++打造局域网聊天室第八课: 异步I/O模型及多线程

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com