C++ 线程的创建、挂起、唤醒和结束 &&&& 利用waitForSingleObject 函数陷入死锁的问题解决
最近在写一个CAN总线的上位机软件,利用CAN转USB的设备连到电脑上,进行数据的传输。在接收下位机发送的数据的时候采用的在线程中持续接收数据。
1、在连接设备的函数中,开启线程。
//在创建线程的时候,将线程挂起,挂起的线程可设置下面的m_bAutoDelete 等属性,再进行线程的唤醒 m_pThread = AfxBeginThread(ReceiveThread,this,,CREATE_SUSPENDED,NULL);
m_pThread->m_bAutoDelete = false;
2、线程唤醒
ResumeThread(m_pThread->m_hThread);
3、线程挂起
SuspendThread(m_pThread->m_hThread);
4、线程结束
线程结束呢,网上最推荐的方法是线程函数正常返回,即某个变量达到某个标准,退出循环,结束线程。
UINT CTest_OilDlg::ReceiveThread(void *param)
{
while()
{
Sleep();
if(dlg->m_connect == )
{
break;
}
// 其他操作
}
}
上面满足m_connect == 0,即设备断开该循环就结束,线程函数就会进行正常返回。在结束线程的地方写如下代码
::WaitForSingleObject(m_pThread->m_hThread,INFINITE);
delete m_pThread;
m_pThread = NULL;//不太懂
WaitForSingleObject该函数是等待线程运行结束,即线程结束后置空线程。但是在使用该函数的时候,子线程里面不能有更新界面的操作,比如更新某个控件。因为这个函数是个阻塞函数,不仅阻塞线程也会阻塞消息。也就是说该函数要等待线程结束,而线程中如果有更新界面的操作,则需要界面的这个主线程给一个反应,但是此时界面的主线程被waitForSingleObject函数阻塞着,也就是进入了一个死锁的境遇。
这个时候可以改用MsgWaitForMultipleObjects这个函数,这个函数是微软针对waitForSingleObject会阻塞消息给出的一个函数,MsgWaitForMultipleObjects不会阻塞消息。
也就是上面的代码可以改写成如下:
DWORD dwRet = ;
MSG msg;
while(true)
{
//等待处理数据线程结束,和等待消息队列中的任何消息
dwRet = MsgWaitForMultipleObjects(,&m_pThread->m_hThread,false,INFINITE,QS_ALLINPUT);
//dwRet = WaitForSingleObject(m_pThread->m_hThread,50);
switch (dwRet)
{
case WAIT_OBJECT_0:
break;
case WAIT_OBJECT_0 + :
//get the message from Queue and dispatch it to specific window
PeekMessage(&msg,NULL,,,PM_REMOVE);
DispatchMessage(&msg);
continue;
default:
break;
}
break;
}
//CloseHandle(m_pThread->m_hThread);
delete m_pThread;
m_pThread = NULL;//不太懂
关于WaitForMultipleObjects(参考https://www.cnblogs.com/shangdawei/p/4015772.html)
DWORD WaitForMultipleObjects(
DWORD dwCount, //等待的内核对象个数
CONST HANDLE* phObjects, //一个存放被等待的内核对象句柄的数组
BOOL bWaitAll, //是否等到所有内核对象为已通知状态后才返回
DWORD dwMilliseconds); //等待时间
HANDLE h[]; //句柄数组
//三个进程句柄
h[] = hProcess1;
h[] = hProcess2;
h[] = hProcess3;
DWORD dw = WaitForMultipleObjects(, h, FALSE, ); //等待3个进程结束
switch (dw)
{
case WAIT_FAILED:
// 函数呼叫失败
break; case WAIT_TIMEOUT:
// 超时
break; case WAIT_OBJECT_0 + :
// h[0](hProcess1)所代表的进程结束
break; case WAIT_OBJECT_0 + :
// h[1](hProcess2)所代表的进程结束
break; case WAIT_OBJECT_0 + :
// h[2](hProcess3)所代表的进程结束
break;
}
C++ 线程的创建、挂起、唤醒和结束 &&&& 利用waitForSingleObject 函数陷入死锁的问题解决的更多相关文章
- 线程概念( 线程的特点,进程与线程的关系, 线程和python理论知识,线程的创建)
参考博客: https://www.cnblogs.com/xiao987334176/p/9041318.html 线程概念的引入背景 进程 之前我们已经了解了操作系统中进程的概念,程序并不能单独运 ...
- python 全栈开发,Day41(线程概念,线程的特点,进程和线程的关系,线程和python 理论知识,线程的创建)
昨日内容回顾 队列 队列 : 先进先出.数据进程安全 队列实现方式: 管道 + 锁 生产者消费者模型 : 解决数据供需不平衡 管道 双向通信 数据进程不安全 EOFError: 管道是由操作系统进行引 ...
- python全栈开发,Day41(线程概念,线程的特点,进程和线程的关系,线程和python理论知识,线程的创建)
昨日内容回顾 队列 队列:先进先出.数据进程安全 队列实现方式:管道+锁 生产者消费者模型:解决数据供需不平衡 管道 双向通信,数据进程不安全 EOFError: 管道是由操作系统进行引用计数的 必须 ...
- 基础学习day11--多线程一线程的创建,运行,同步和锁
一.线程基本概述 1.1.进程和线程 进程:一个应用程序一般都是一个进程,正在进行的程序 每一个进程最少都有一个线程,都有一个执行顺序,该顺序是一个执行路径或者一个控制单元 线程:进程中一个独立的控制 ...
- 漫谈并发编程(二):java线程的创建与基本控制
java线程的创建 定义任务 在java中使用任务这个名词来表示一个线程控制流的代码段,用Runnable接口来标记一个任务,该接口的run方法为线程运行的代码段. public ...
- java 添加一个线程、创建响应的用户界面 。 演示示例代码
javajava 添加一个线程.创建响应的用户界面 . 演示示例代码 来自thinking in java 4 21章 部分的代码 夹21.2.11 thinking in java 4免费下载: ...
- ACE线程管理机制-线程的创建与管理
转载于:http://www.cnblogs.com/TianFang/archive/2006/12/04/581369.html 有过在不同的操作系统下用c++进行过多线程编程的朋友对那些线程处理 ...
- Java多线程系列 基础篇02 线程的创建和运行
1.线程创建的方式常用有两种 1. 继承 Thread 类创建线程 2. 实现 Runnable 接口创建线程 2.Thread 和 Runnable的区别 Thread和Runnable的相同点:都 ...
- C/C++ Muti-Thread多线程编程学习(之)线程Thread | 创建、运行、结束
文章目录 前言 线程 Thread 创建线程 CreateThread _beginthread _beginthreadex pthread_create 线程运行 结束线程 前言 多线程(Mu ...
随机推荐
- Centos配置nginx反向代理8090端口到80端口
下面,我就来说说怎么反向代理自己的项目到默认80端口. 1)安装nginx:yum install nginx -y 2)启动nginx:service nginx start或者systemctl ...
- [C语言]删除用户自定义后缀名的所有文件
环境:win7 IDE:DEV-C++ 编译器:GCC 编译结果:Success 运行结果:Success 使用说明: 1.输入需要查询的目录,比如e: 2.输入需要删除的后缀名:比如:txt 注意: ...
- MySQL 温故知心(三)
MySQL锁概述 相对其他数据库而言,MySQL的锁机制比较简单,其最显著的特点是不同的存储引擎支持不同的锁机制.比如,MyISAM和MEMORY存储引擎采用的是表级锁(table-level loc ...
- 后缀自动机模板 SAM
一点疑问: 当创建nq节点时,要不要把nq的cnt标记赋值为1? 讲道理nq节点也是代表一个子串啊,不过网上的模板都没赋值. 2017.9.18 update: 把memset部分重写,改成用节点用到 ...
- 两句话,实现android 4.4以上实现沉浸式状态栏
效果图如下,就是状态栏和actionbar保持一致的颜色,非常漂亮 1:在Activity的OnCreate函数 if (Build.VERSION.SDK_INT >= Build.VERSI ...
- NodeJS NPM HTTPS
npm config set registry http://registry.npmjs.org/
- centos升级vim
vim7爆出严重安全漏洞,升级到vim8,过程如下: yum remove vim -y yum install ncurses-devel -y 如果失败,提示unknown host apt.sw ...
- WeX5基础
最近在研究微信app开发,使用的是WeX5,在这里把一些基础知识点记录下来,忘记了可以翻阅查看. 一:开发后端服务 1.建立数据源:窗口--首选项--studio配置--数据源--增加--数据源类型选 ...
- 浅谈boost.variant的几种访问方式
前言 variant类型在C++14并没有加入,在cppreference网站上可以看到该类型将会在C++17加入,若想在不支持C++17的编译器上使用variant类型,我们可以通过boost的va ...
- ElasticSearch(一)ElasticSearch的应用场景及为什么要选择ElasticSearch?
先了解一下数据的分类 结构化数据又可以称之为行数据,存储在数据库里,可以用二维表结构来逻辑表达实现的数据.其实就是可以能够用数据或者统一的结构加以表示的数据.比如在数据表存储商品的库存,可以用整型表示 ...