孙鑫 VC++深入详解第15课——多线程

2023-12-25 15:40

本文主要是介绍孙鑫 VC++深入详解第15课——多线程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.简单的多线程实例

步骤:

①全局函数ThreadProc

②创建进程CreateThread

③关闭进程CloseHandle

④让主线程休眠 Sleep()

代码:

#include <iostream>
#include <stdlib.h>
#include <Windows.h>
using namespace std;DWORD WINAPI Fun1Proc(LPVOID lpParameter);
HANDLE hMetux;
void main()
{hMetux = CreateMutex(NULL,FALSE,NULL);HANDLE hThread1;hThread1 = CreateThread(NULL,0,Fun1Proc,NULL,0,NULL);CloseHandle(hThread1);cout<<"main thread is runing"<<endl;Sleep(1000);cin.get();cin.get();}DWORD WINAPI Fun1Proc(LPVOID lpParameter)
{cout<<"thread1 is running"<<endl;return 0;
}

2.多线程实现火车售票

①创建两个线程同时出售火车票

②线程中火车票 ticket>0则出售,否则结束

#include <Windows.h>
#include <stdlib.h>
#include <iostream>
using namespace std;
DWORD WINAPI FunOneProc(LPVOID lpParam);
DWORD WINAPI FunTwoProc(LPVOID lpParam);
int ticket = 100;
int main()
{HANDLE hThreadOne;HANDLE hThreadTwo;hThreadOne = CreateThread(NULL,0,FunOneProc,NULL,0,NULL);hThreadTwo = CreateThread(NULL,0,FunTwoProc,NULL,0,NULL);CloseHandle(hThreadOne);CloseHandle(hThreadTwo);Sleep(4000);cin.get();cin.get();return 0;
}DWORD WINAPI FunOneProc(LPVOID lpParam)
{while (TRUE){if (ticket > 0){cout<<"One Thread tickets:"<<ticket--<<endl;} else{break;}}return 0;
}DWORD WINAPI FunTwoProc(LPVOID lpParam)
{while (TRUE){if (ticket > 0){cout<<"Two Thread tickets:"<<ticket--<<endl;} else{break;}	}return 0;
}


运行结果:


结果分析:

开始看到这个结果的时候,我大惊了一下。想着10099,9897这么大的数字从何而来,我的设置ticket =100。想了一番,这么大的数字应该分开看 10099 为100,99 ;9897为98,97.

      为什么会这样呢。孙鑫老师的书里的运行结果也没有这么奇葩。我自己想了一下,孙鑫老师电脑估计比较老土配置不高,是单核的cpu。而我因为游戏时间比较多,买了双核的cpu,并行性高。当进程One在运行时输出One Thread tickets:,恰好到了进程Two,然后输出:Two Thread tickets。然后又到了进程One,输出100,进程Two输出99。所以最后结果就呈现上面的奇葩了。`(*∩_∩*)′


3.利用互斥对象实现同步

①声明全局互斥变量:HANDLE hMutex

②创建互斥对象:CreateMutex

③在保护代码前加:WaitForSingleObject

④保护代码后加:ReleaseMutex

注意事项:创建的互斥对象应该在创建进程前面

#include <Windows.h>
#include <stdlib.h>
#include <iostream>
using namespace std;
DWORD WINAPI FunOneProc(LPVOID lpParam);
DWORD WINAPI FunTwoProc(LPVOID lpParam);
int ticket = 100;
HANDLE hMutex;
int main()
{HANDLE hThreadOne;HANDLE hThreadTwo;hMutex = CreateMutex(NULL,FALSE,NULL);hThreadOne = CreateThread(NULL,0,FunOneProc,NULL,0,NULL);hThreadTwo = CreateThread(NULL,0,FunTwoProc,NULL,0,NULL);CloseHandle(hThreadOne);CloseHandle(hThreadTwo);Sleep(4000);cin.get();cin.get();return 0;
}DWORD WINAPI FunOneProc(LPVOID lpParam)
{while (TRUE){WaitForSingleObject(hMutex,INFINITE);if (ticket > 0){cout<<"One Thread tickets:"<<ticket--<<endl;} else{break;}ReleaseMutex(hMutex);}return 0;
}DWORD WINAPI FunTwoProc(LPVOID lpParam)
{while (TRUE){WaitForSingleObject(hMutex,INFINITE);if (ticket > 0){cout<<"Two Thread tickets:"<<ticket--<<endl;} else{break;}	ReleaseMutex(hMutex);}return 0;
}
运行结果:



4.多线程实现Chat聊天室

实现步骤:

1.加载套接字库

①在 stdafx.h 文件中 添加 #include <afxsock.h>// MFC 套接字扩展

②在App类中的InitInstance函数中添加如下代码:

if (!AfxSocketInit())
{AfxMessageBox(IDP_SOCKETS_INIT_FAILED);return FALSE;
}

2.初始化套接字绑定IP和端口号

①在OnInitDialog函数中添加自定义函数InitSocket;

②InitSocket函数的实现

BOOL CUdpSrvDlg::InitSocket()
{m_socket = socket(AF_INET,SOCK_DGRAM,0);if (INVALID_SOCKET == m_socket){AfxMessageBox(_T("套接字创建失败!"));return FALSE;}SOCKADDR_IN sockAddr;sockAddr.sin_addr.S_un.S_addr = htonl(INADDR_ANY);sockAddr.sin_family = AF_INET;sockAddr.sin_port = htons(6000);//绑定套接字int retval = bind(m_socket,(SOCKADDR*)&sockAddr,sizeof(SOCKADDR));if (SOCKET_ERROR == retval){closesocket(m_socket);AfxMessageBox(_T("绑定套接字失败"));return FALSE;}return TRUE;
}

3.用多线程实现接收端功能

①定义结构体:RECVPARAM

struct RECVPARAM 
{HWND hwnd;//对话框句柄SOCKET socket;//已创建的套接字
};

②创建接收线程 hThread

    InitSocket();	RECVPARAM *pRecvParam = new RECVPARAM;pRecvParam->hwnd = m_hWnd;pRecvParam->socket = m_socket;//创建接收线程HANDLE hThread = CreateThread(NULL,0,RecvProc,(LPVOID)pRecvParam,0,NULL);//关闭该接收进程句柄,释放其引用计数CloseHandle(hThread);

③实现接收线程函数:RecvPRoc

 在类内定义,必须使用静态变量

static DWORD WINAPI RecvProc(LPVOID lpParam);

函数实现:

DWORD WINAPI CUdpSrvDlg::RecvProc(LPVOID lpParam)
{//获取主线程传递的套接字和窗口句柄SOCKET sock = ((RECVPARAM*)lpParam)->socket;HWND hwnd = ((RECVPARAM*)lpParam)->hwnd;delete lpParam;SOCKADDR_IN addrFrom;int len = sizeof(SOCKADDR);char recvBuf[200];char tempBuf[300];int retval;while (TRUE){//接收数据retval = recvfrom(sock,recvBuf,200,0,(SOCKADDR*)&addrFrom,&len);if (SOCKET_ERROR == retval){break;}sprintf_s(tempBuf,300,"%s say: %s",inet_ntoa(addrFrom.sin_addr),recvBuf);::PostMessage(hwnd,WM_RECVDATA,0,(LPARAM)tempBuf);}return 0;
}

4.将接收的信息显示在窗口上

①自定义消息宏

#define WM_RECVDATA WM_USER +1

②设置消息响应函数原型的声明

afx_msg LRESULT OnRecvData(WPARAM wParam,LPARAM lParam);

③添加消息映射

ON_MESSAGE(WM_RECVDATA,OnRecvData)

④添加消息函数的定义

LRESULT CUdpSrvDlg::OnRecvData(WPARAM wParam,LPARAM lParam)
{//取出接收到的数据CString str = (char*)lParam;CString strTemp;//获得已有数据GetDlgItemText(IDC_EDIT_RECV,strTemp);str += "\r\n";str += strTemp;//显示所有接收到的数据SetDlgItemText(IDC_EDIT_RECV,str);return 0;
}

5.设置发送端的信息

实现函数:

void CUdpSrvDlg::OnBnClickedBtnSend()
{// TODO: 在此添加控件通知处理程序代码//获取对方IPDWORD dwIP;((CIPAddressCtrl*)GetDlgItem(IDC_IPADDRESS1))->GetAddress(dwIP);SOCKADDR_IN addrTo;addrTo.sin_addr.S_un.S_addr = htonl(dwIP);addrTo.sin_family = AF_INET;addrTo.sin_port = htons(6000);CString strSend;//获得待发送数据GetDlgItemText(IDC_EDIT_SEND,strSend);//发送数据sendto(m_socket,strSend,strSend.GetLength()+1,0,(SOCKADDR*)&addrTo,sizeof(SOCKADDR));//清空发送编辑框中的内容SetDlgItemText(IDC_EDIT_SEND,"");	
}


6.小技巧:

①EDIT控件 MultiLine设置多行显示。

②default button设置TRUE,这是响应ENTER键。

这篇关于孙鑫 VC++深入详解第15课——多线程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Oracle的to_date()函数详解

《Oracle的to_date()函数详解》Oracle的to_date()函数用于日期格式转换,需要注意Oracle中不区分大小写的MM和mm格式代码,应使用mi代替分钟,此外,Oracle还支持毫... 目录oracle的to_date()函数一.在使用Oracle的to_date函数来做日期转换二.日

Java实现任务管理器性能网络监控数据的方法详解

《Java实现任务管理器性能网络监控数据的方法详解》在现代操作系统中,任务管理器是一个非常重要的工具,用于监控和管理计算机的运行状态,包括CPU使用率、内存占用等,对于开发者和系统管理员来说,了解这些... 目录引言一、背景知识二、准备工作1. Maven依赖2. Gradle依赖三、代码实现四、代码详解五

Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单

《Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单》:本文主要介绍Springboot的ThreadPoolTaskScheduler线... 目录ThreadPoolTaskScheduler线程池实现15分钟不操作自动取消订单概要1,创建订单后

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

Mysql 中的多表连接和连接类型详解

《Mysql中的多表连接和连接类型详解》这篇文章详细介绍了MySQL中的多表连接及其各种类型,包括内连接、左连接、右连接、全外连接、自连接和交叉连接,通过这些连接方式,可以将分散在不同表中的相关数据... 目录什么是多表连接?1. 内连接(INNER JOIN)2. 左连接(LEFT JOIN 或 LEFT

Java中switch-case结构的使用方法举例详解

《Java中switch-case结构的使用方法举例详解》:本文主要介绍Java中switch-case结构使用的相关资料,switch-case结构是Java中处理多个分支条件的一种有效方式,它... 目录前言一、switch-case结构的基本语法二、使用示例三、注意事项四、总结前言对于Java初学者

Linux内核之内核裁剪详解

《Linux内核之内核裁剪详解》Linux内核裁剪是通过移除不必要的功能和模块,调整配置参数来优化内核,以满足特定需求,裁剪的方法包括使用配置选项、模块化设计和优化配置参数,图形裁剪工具如makeme... 目录简介一、 裁剪的原因二、裁剪的方法三、图形裁剪工具四、操作说明五、make menuconfig

详解Java中的敏感信息处理

《详解Java中的敏感信息处理》平时开发中常常会遇到像用户的手机号、姓名、身份证等敏感信息需要处理,这篇文章主要为大家整理了一些常用的方法,希望对大家有所帮助... 目录前后端传输AES 对称加密RSA 非对称加密混合加密数据库加密MD5 + Salt/SHA + SaltAES 加密平时开发中遇到像用户的

Springboot使用RabbitMQ实现关闭超时订单(示例详解)

《Springboot使用RabbitMQ实现关闭超时订单(示例详解)》介绍了如何在SpringBoot项目中使用RabbitMQ实现订单的延时处理和超时关闭,通过配置RabbitMQ的交换机、队列和... 目录1.maven中引入rabbitmq的依赖:2.application.yml中进行rabbit

C语言线程池的常见实现方式详解

《C语言线程池的常见实现方式详解》本文介绍了如何使用C语言实现一个基本的线程池,线程池的实现包括工作线程、任务队列、任务调度、线程池的初始化、任务添加、销毁等步骤,感兴趣的朋友跟随小编一起看看吧... 目录1. 线程池的基本结构2. 线程池的实现步骤3. 线程池的核心数据结构4. 线程池的详细实现4.1 初