使用互斥对象:

#include <windows.h>
#include <iostream> #define THREADCOUNT 6 HANDLE ghMutex; DWORD WINAPI RunThreadFunction(LPVOID lpParam); int main(void)
{
HANDLE aThread[THREADCOUNT];
DWORD ThreadID; // 创建一个无所有者的互斥锁
ghMutex = CreateMutex(
NULL,
FALSE,
NULL); if (ghMutex == NULL)
{
std::cout << "CreateMutex error: " << GetLastError() << std::endl;
ExitProcess(2);
} for(int i = 0; i < THREADCOUNT; i++)
{
aThread[i] = CreateThread(
NULL,
0,
(LPTHREAD_START_ROUTINE) RunThreadFunction,
NULL,
0,
&ThreadID); if( aThread[i] == nullptr )
{
std::cout << "CreateThread error: " << GetLastError() << std::endl;
ExitProcess(3);
}
} // 等到所有线程都终止
WaitForMultipleObjects(THREADCOUNT, aThread, TRUE, INFINITE); for(int i = 0; i < THREADCOUNT; i++ )
CloseHandle(aThread[i]); // 关闭所有线程句柄并释放内存 CloseHandle(ghMutex); return 0;
} DWORD WINAPI RunThreadFunction(LPVOID lpParam)
{
UNREFERENCED_PARAMETER(lpParam); DWORD dwWaitResult; for(int i = 0; i < 5; i++)
{
// 获得互斥权
dwWaitResult = WaitForSingleObject(
ghMutex,
INFINITE); switch (dwWaitResult)
{
case WAIT_OBJECT_0:
try {
std::cout << i << " Thread " << GetCurrentThreadId() << " is run..."<< std::endl;
}
catch (...) {
if (! ReleaseMutex(ghMutex)) // 释放互斥权
{
std::cout << "ReleaseMutex error" << std::endl;
}
}
break;
// 线程处于不确定状态
case WAIT_ABANDONED:
return FALSE;
}
} return TRUE;
}

  代码运行结果:

  这里会发现线程ID都是同一个...这是因为,这段代码中,在try语句块忘记释放锁了,所以最初执行的线程执行完`RunThreadFunction`函数中的循环后都没有释放掉,所以出现了这种情况。

  如果直接去掉互斥锁会看到输出十分混乱...原因也在于std::cout的输出是拼接的。如果用printf来输出会好一些,但输出的 i 可能会是混乱的,如下图:

  因为C++中没有finally关键字,final好像也不是用来做异常处理的,在try中也加上锁后的运行效果:

  可以看出执行顺序为:因为线程数定义为6个,所以就创建6个线程来同时执行一个函数,函数中的5次循环会使得阻塞,但代码中加了互斥锁,为了保持同步,所以其他线程会等待当前这个进程中所有线程执行完...

  下面是在MSDN看到的一个创建线程的例子,如果想要传递其他类型的参数且没有类型转换的警告,可以这样做(用到了结构体):

#include <windows.h>
#include <tchar.h>
#include <strsafe.h> #define MAX_THREADS 3
#define BUF_SIZE 255 DWORD WINAPI MyThreadFunction( LPVOID lpParam );
void ErrorHandler(LPTSTR lpszFunction); typedef struct MyData {
int val1;
int val2;
} MYDATA, *PMYDATA; int _tmain()
{
PMYDATA pDataArray[MAX_THREADS];
DWORD dwThreadIdArray[MAX_THREADS];
HANDLE hThreadArray[MAX_THREADS]; for( int i=0; i<MAX_THREADS; i++ )
{
pDataArray[i] = (PMYDATA) HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(MYDATA)); if( pDataArray[i] == NULL )
{
ExitProcess(2);
} pDataArray[i]->val1 = i;
pDataArray[i]->val2 = i+100; hThreadArray[i] = CreateThread(
NULL,
0,
MyThreadFunction,
pDataArray[i],
0,
&dwThreadIdArray[i]); if (hThreadArray[i] == NULL)
{
ErrorHandler((LPTSTR)TEXT("CreateThread"));
ExitProcess(3);
}
} WaitForMultipleObjects(MAX_THREADS, hThreadArray, TRUE, INFINITE); for(int i=0; i<MAX_THREADS; i++)
{
CloseHandle(hThreadArray[i]);
if(pDataArray[i] != NULL)
{
HeapFree(GetProcessHeap(), 0, pDataArray[i]);
pDataArray[i] = NULL;
}
} return 0;
} DWORD WINAPI MyThreadFunction( LPVOID lpParam )
{
HANDLE hStdout;
PMYDATA pDataArray; TCHAR msgBuf[BUF_SIZE];
size_t cchStringSize;
DWORD dwChars; hStdout = GetStdHandle(STD_OUTPUT_HANDLE);
if( hStdout == INVALID_HANDLE_VALUE )
return 1; pDataArray = (PMYDATA)lpParam; StringCchPrintf(msgBuf, BUF_SIZE, TEXT("Parameters = %d, %d\n"), pDataArray->val1, pDataArray->val2);
StringCchLength(msgBuf, BUF_SIZE, &cchStringSize);
WriteConsole(hStdout, msgBuf, (DWORD)cchStringSize, &dwChars, NULL); return 0;
} void ErrorHandler(LPTSTR lpszFunction)
{
LPVOID lpMsgBuf;
LPVOID lpDisplayBuf;
DWORD dw = GetLastError(); FormatMessage(
FORMAT_MESSAGE_ALLOCATE_BUFFER |
FORMAT_MESSAGE_FROM_SYSTEM |
FORMAT_MESSAGE_IGNORE_INSERTS,
NULL,
dw,
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
(LPTSTR) &lpMsgBuf,
0, NULL ); lpDisplayBuf = (LPVOID)LocalAlloc(LMEM_ZEROINIT, (lstrlen((LPCTSTR) lpMsgBuf) + lstrlen((LPCTSTR) lpszFunction) + 40) * sizeof(TCHAR));
StringCchPrintf((LPTSTR)lpDisplayBuf, LocalSize(lpDisplayBuf) / sizeof(TCHAR), TEXT("%s failed with error %d: %s"), lpszFunction, dw, lpMsgBuf);
MessageBox(NULL, (LPCTSTR) lpDisplayBuf, TEXT("Error"), MB_OK); LocalFree(lpMsgBuf);
LocalFree(lpDisplayBuf);
}

  

C/C++ — CreateThread 相关 API的更多相关文章

  1. OpenGL FrameBufferCopy相关Api比较(glCopyPixels,glReadPixels,glCopyTexImage2D,glFramebufferTexture2D)

    OpenGL FrameBufferCopy相关Api比较 glCopyPixels,glReadPixels,glCopyTexImage2D,glFramebufferTexture2D 标题所述 ...

  2. QQ音乐的各种相关API

    QQ音乐的各种相关API 分类: oc2014-01-29 15:34 2676人阅读 评论(2) 收藏 举报 基本上论坛里做在线音乐的都在用百度的API,进来发现百度的API不仅歌曲的质量不可以保证 ...

  3. addChildViewController相关api深入剖析

    注:本文根据个人的实践和理解写成,若有不当之处欢迎斧正和探讨! addChildViewController是一个从iOS5开始支持的api接口,相关的一系列的接口是用来处理viewcontrolle ...

  4. [原创]java WEB学习笔记44:Filter 简介,模型,创建,工作原理,相关API,过滤器的部署及映射的方式,Demo

    本博客为原创:综合 尚硅谷(http://www.atguigu.com)的系统教程(深表感谢)和 网络上的现有资源(博客,文档,图书等),资源的出处我会标明 本博客的目的:①总结自己的学习过程,相当 ...

  5. 关于iOS中用AudioFile相关API解码或播放AAC_HE_V2时仅仅能识别单声首22.05k採样率的问题

    关于iOS中用AudioFile相关API解码或播放AAC_HE_V2时仅仅能识别单声首22.05k採样率的问题 在官方AQPlayer Demo 和 aqofflinerender中.都用了Audi ...

  6. TCP/IP协议栈源码图解分析系列10:linux内核协议栈中对于socket相关API的实现

    题记:本系列文章的目的是抛开书本从Linux内核源代码的角度详细分析TCP/IP协议栈内核相关技术 轻松搞定TCP/IP协议栈,原创文章欢迎交流, byhankswang@gmail.com linu ...

  7. 【Socket编程】Java中网络相关API的应用

    Java中网络相关API的应用 一.InetAddress类 InetAddress类用于标识网络上的硬件资源,表示互联网协议(IP)地址. InetAddress类没有构造方法,所以不能直接new出 ...

  8. java 11 移除的一些其他内容,更简化的编译运行程序,Unicode 10,移除了不太使用的JavaEE模块和CORBA技术,废除Nashorn javascript引擎,不建议使用Pack200 相关api

    移除的一些其他内容 移除项 移除了com.sun.awt.AWTUtilities 移除了sun.misc.Unsafe.defineClass, 使用java.lang.invoke.MethodH ...

  9. python 以单例模式封装logging相关api实现日志打印类

    python 以单例模式封装logging相关api实现日志打印类   by:授客QQ:1033553122 测试环境: Python版本:Python 2.7   实现功能: 支持自由配置,如下lo ...

随机推荐

  1. docker删除mysql镜像失败Error response from daemon: conflict: unable to delete 8809d5286227 (must be forced) - image is being used by stopped container 1a2a427273b3

    错误解析:这是由于要删除的目标镜像中有容器存在,故无法删除镜像 解决办法:先删除镜像中的容器,再删除该镜像.

  2. stream.js

    <script src='stream-min.js'></script> 下载 stream.js 2Kb minified streams是什么? Streams 是一个操 ...

  3. 吴裕雄 python 神经网络——TensorFlow实现AlexNet模型处理手写数字识别MNIST数据集

    import tensorflow as tf # 输入数据 from tensorflow.examples.tutorials.mnist import input_data mnist = in ...

  4. C++中map的介绍用法以及Gym题目:Two Sequences

    Map是STL的一个关联容器,它提供一对一(其中第一个可以称为关键字(key),每个关键字只能在map中出现一次,第二个可能称为该关键字的值(value))的数据 处理能力,由于这个特性,它完成有可能 ...

  5. redis介绍、单机安装以及java调用

    什么是redis Redis是用C语言开发的一个开源的高性能键值对(key-value)数据库.和传统的关系型数据库不一样,不一定遵循传统数据库的一些基本要求(非关系型的.分布式的.开源的.水平可扩展 ...

  6. pip install cv2 安装报错

    pip install cv2 安装报错是一个常见现象: ERROR: Could not find a version that satisfies the requirement cv2 (fro ...

  7. php 基础 自动类型转换

    1.自动类型转换:表示运算的时候,Boolean,Null,String等类型,会先自动转为Integer或Float类型 null-->0 true-->1 false-->0 S ...

  8. Linux - 查看端口占用、开放情况

    1. lsof -i : 2. nmap 127.0.0.1 3. sudo netstat -tunlp (不加sudo看不见PID) 4. gufw 参考 https://askubuntu.co ...

  9. SpringBoot2.x过后static下的静态资源无法访问

    @Configuration public class CorsConfig extends WebMvcConfigurationSupport {/** * 添加静态资源文件,外部可以直接访问地址 ...

  10. 给博客页面添加 live2d 小萝莉

    添加依赖 在页脚HTML代码的地方添加下面的代码: <script src="https://eqcn.ajz.miesnfu.com/wp-content/plugins/wp-3d ...