前几天简单对C和C++中的创建多线程的函数进行了测试,这篇随笔就简单介绍一下创建线程的相关函数。

  C中三个创建线程函数:pthread_create()、_beginthread()、CreateThread()

  三个关闭线程函数:pthread_exit()、_endthread()、ExitThread()

  头文件分别是:pthread.h、process.h、processthreadsapi.h

  pthread_create()函数原型:

int pthread_create(pthread_t *tidp,
const pthread_attr_t *attr,
(void*)(*start_rtn)(void*),
void *arg);

  _beginthread()函数原型:

uintptr_t _beginthread(void( *start_address )( void * ),
unsignedstack_size,void *arglist);

  _beginthread()函数是调用CreateThread()函数来实现的,但多了一些功能,是C标准库中提供的函数。而CreateThread()函数创建一个线程在调用进程的虚拟地址空间内执行,是操作系统中的函数。可参考解释:http://bbs.csdn.net/topics/21693。

  CreateThread()函数原型:

HANDLE WINAPI CreateThread(
_In_opt_ LPSECURITY_ATTRIBUTES lpThreadAttributes,
_In_ SIZE_T dwStackSize,
_In_ LPTHREAD_START_ROUTINE lpStartAddress,
_In_opt_ LPVOID lpParameter,
_In_ DWORD dwCreationFlags,
_Out_opt_ LPDWORD lpThreadId
);

  pthread_exit()函数原型:

void pthread_exit(void *retval);

  _endthread()函数原型:

void _endthread( void );
void _endthreadex(
unsigned retval
);

  ExitThread()函数原型:

VOID WINAPI ExitThread(
_In_ DWORD dwExitCode
);

  貌似要使用pthread_exit();需要安全释放,否则会出现内存泄露:pthread_exit()

  下面测试一下各个函数:

  pthread.h:

#include <stdio.h>
#include <stdlib.h>
#include <pthread.h> #define MAXTHREADS 5 void* Run(void* args)
{
printf("%d:Hello World!\n", *((int*)args));
pthread_exit(NULL);
return NULL;
} int main()
{
pthread_t tid[MAXTHREADS];
pthread_attr_t attr;
void*status;
int ret; pthread_attr_init(&attr);//初始化attr
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);//设置获得attr的线程为可连接的:PTHREAD_CREATE_JOINABLE 原型为 0 for(int i = 0; i < MAXTHREADS; i++)
{
if((ret = pthread_create(&tid[i], &attr, Run, (void*)&i)) != 0)//获得attr,则线程可连接
{
puts("Pthread_Create error!");
exit(1);
}
pthread_join(tid[i], &status);//等待线程结束,结束的线程便不能再被连接
} pthread_attr_destroy(&attr);//销毁attr资源 return 0;
}

  pthread_join等待其他线程结束后再连接能连接的函数,已经结束的线程便不能再被连接。所以输出结果可看到是按顺序执行的。感觉就是加上了互斥锁,不知道理解对不对?

  process.h:

#include <stdio.h>
#include <stdlib.h>
#include <process.h> #define MAXTHREAD 5 void run(void*i)
{
printf("%d:Hello world!\n", *((int*)i));
} int main()
{
for(int i = 0; i < MAXTHREAD; i++) {
_beginthread(&run, 0, (void*)&i);
}
_endthread(); return 0;
}

  消除了类型强转的警告,但程序无法正常退出,有时也无法正常进行。

--------------------update 2018-02-24 16:09:01---------------------

  processthreadsapi.h:

#include <windows.h>

#define MAXTHREAD 5

HANDLE ghMutex;

DWORD WINAPI Run_Process(LPVOID p) {

    WaitForSingleObject(ghMutex, INFINITE);
try
{
TCHAR s = *((int*)p)+'0';
MessageBoxA(NULL, TEXT("Hello world!"), TEXT(&s), MB_ICONWARNING | MB_OK);
ReleaseMutex(ghMutex);
}
catch(...)
{
if (! ReleaseMutex(ghMutex))
MessageBoxA(NULL, TEXT("ReleaseMutex Error!"), TEXT("ERROR"), MB_ICONERROR | MB_OK);
} return GetLastError();
} int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE HPrevInstance, PSTR szCmdLine, int iCmdShow)
{
DWORD dwThreadIdArray[MAXTHREAD];
HANDLE hThreadArray[MAXTHREAD]; if((ghMutex = CreateMutex(NULL, FALSE, NULL)) == NULL)
{
ExitProcess(1);
} for(int i = 0; i < MAXTHREAD; i++) {
hThreadArray[i] = CreateThread(
NULL,
0,
Run_Process,
(LPVOID)&i,
0,
&dwThreadIdArray[i]); if (hThreadArray[i] == NULL)
{
ExitProcess(3);
}
} WaitForMultipleObjects(MAXTHREAD, hThreadArray, TRUE, INFINITE); for(int i = 0; i < MAXTHREAD; i++)
{
CloseHandle(hThreadArray[i]);
} CloseHandle(ghMutex); return 0;
}

  参考我的这篇随笔:

http://www.cnblogs.com/darkchii/p/8447863.html

-------------------------------

  参考资料:

http://www.man7.org/linux/man-pages/man3/pthread_exit.3.html
http://www.man7.org/linux/man-pages/man3/pthread_create.3.html
http://www.cppblog.com/prayer/archive/2012/04/23/172427.html
https://msdn.microsoft.com/en-us/library/windows/desktop/ms682659(v=vs.85).aspx
https://msdn.microsoft.com/en-us/library/windows/desktop/ms682453(v=vs.85).aspx
https://msdn.microsoft.com/zh-cn/library/hw264s73.aspx

  

C/C++ - 多线程的更多相关文章

  1. Python中的多进程与多线程(一)

    一.背景 最近在Azkaban的测试工作中,需要在测试环境下模拟线上的调度场景进行稳定性测试.故而重操python旧业,通过python编写脚本来构造类似线上的调度场景.在脚本编写过程中,碰到这样一个 ...

  2. 多线程爬坑之路-Thread和Runable源码解析之基本方法的运用实例

    前面的文章:多线程爬坑之路-学习多线程需要来了解哪些东西?(concurrent并发包的数据结构和线程池,Locks锁,Atomic原子类) 多线程爬坑之路-Thread和Runable源码解析 前面 ...

  3. 多线程爬坑之路-学习多线程需要来了解哪些东西?(concurrent并发包的数据结构和线程池,Locks锁,Atomic原子类)

    前言:刚学习了一段机器学习,最近需要重构一个java项目,又赶过来看java.大多是线程代码,没办法,那时候总觉得多线程是个很难的部分很少用到,所以一直没下决定去啃,那些年留下的坑,总是得自己跳进去填 ...

  4. Java多线程

    一:进程与线程 概述:几乎任何的操作系统都支持运行多个任务,通常一个任务就是一个程序,而一个程序就是一个进程.当一个进程运行时,内部可能包括多个顺序执行流,每个顺序执行流就是一个线程.   进程:进程 ...

  5. .NET基础拾遗(5)多线程开发基础

    Index : (1)类型语法.内存管理和垃圾回收基础 (2)面向对象的实现和异常的处理基础 (3)字符串.集合与流 (4)委托.事件.反射与特性 (5)多线程开发基础 (6)ADO.NET与数据库开 ...

  6. Java多线程基础——对象及变量并发访问

    在开发多线程程序时,如果每个多线程处理的事情都不一样,每个线程都互不相关,这样开发的过程就非常轻松.但是很多时候,多线程程序是需要同时访问同一个对象,或者变量的.这样,一个对象同时被多个线程访问,会出 ...

  7. C#多线程之线程池篇3

    在上一篇C#多线程之线程池篇2中,我们主要学习了线程池和并行度以及如何实现取消选项的相关知识.在这一篇中,我们主要学习如何使用等待句柄和超时.使用计时器和使用BackgroundWorker组件的相关 ...

  8. C#多线程之线程池篇2

    在上一篇C#多线程之线程池篇1中,我们主要学习了如何在线程池中调用委托以及如何在线程池中执行异步操作,在这篇中,我们将学习线程池和并行度.实现取消选项的相关知识. 三.线程池和并行度 在这一小节中,我 ...

  9. C#多线程之线程池篇1

    在C#多线程之线程池篇中,我们将学习多线程访问共享资源的一些通用的技术,我们将学习到以下知识点: 在线程池中调用委托 在线程池中执行异步操作 线程池和并行度 实现取消选项 使用等待句柄和超时 使用计时 ...

  10. C#多线程之线程同步篇3

    在上一篇C#多线程之线程同步篇2中,我们主要学习了AutoResetEvent构造.ManualResetEventSlim构造和CountdownEvent构造,在这一篇中,我们将学习Barrier ...

随机推荐

  1. Travel in desert

    传送门 不算难吧 应该有思路的 还是太水了吧 (而且和货车运输很像的啊 ---------------------------------------------------------------- ...

  2. Ant安装及环境配置

    首先说一下啥事ant 所以装ant的前提是得装java 点击查看怎么安装JDK 然后安装ant,其实挺简单的 官网下载http://ant.apache.org 所有版本https://www.apa ...

  3. spark streaming读取kakfka数据手动维护offset

    在spark streaming读取kafka的数据中,spark streaming提供了两个接口读取kafka中的数据,分别是KafkaUtils.createDstream,KafkaUtils ...

  4. ES-windows版本设置远程访问

    1,官网下载 2,下载完解压 3,修改配置文件 elasticsearch.yml network.host: 0.0.0.0http.port: 9200transport.host: localh ...

  5. Java常量/变量

    1. 常量 /* 常量:在程序运行期间,固定不变的量. 常量的分类: 1. 字符串常量:凡是用双引号引起来的部分,叫做字符串常量.例如:"abc"."Hello" ...

  6. 洛谷 P1880 [NOI1995]石子合并(区间DP)

    嗯... 题目链接:https://www.luogu.org/problem/P1880 这道题特点在于石子是一个环,所以让a[i+n] = a[i](两倍长度)即可解决环的问题,然后注意求区间最小 ...

  7. IDEA部署项目,并结合Shell脚本运行Java程序

    一.概述 在实际开发中,我们写好的代码,往往打成war包或jar包,通过winscp或其他软件将其上传至服务器,然而这样非常大的一个弊端就是不利于开发,为什么这么说呢?假如我们刚刚将springboo ...

  8. java中4种常用线程池

    一.线程池 线程池:说白了,就是一种线程使用模式.线程过多会带来调度开销,进而影响整体性能.而线程池维护着多个线程,等待着监督管理者分配可并发执行的任务,这避免了在处理短时间任务时创建与销毁线程的代价 ...

  9. 计算机二级-C语言-程序填空题-190115记录-fprintf()函数和fscanf()函数的使用。

    //给定程序,函数fun的功能是:将自然数1~10以及它们的平方根写到名为myflie3.txt的文本文件中,然后再顺序读出显示在屏幕上. //重难点:fprintf()函数和fscanf()函数的使 ...

  10. 13 DFT变换的性质

    DFT变换的性质 线性性质 \[ \begin{aligned} y[n]&=ax[n]+bw[n]\xrightarrow{DFT}Y[k]=\sum_{n=0}^{N-1}(ax[n]+ ...