线程的创建以及使用(MFC/C/C++)

这篇具有很好参考价值的文章主要介绍了线程的创建以及使用(MFC/C/C++)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

目录

一、创建线程的几种方法

1、使用POSIX线程库(pthread)(C)

2、 使用Windows线程库(Win32 API)(C)

3、使用C++11标准库

二、线程的应用(常用实例,附上完整代码)

1、CreateThread创建线程的实例


一、创建线程的几种方法

在C/C++中,有多种方法可以创建线程,以下列出了其中几种常用的方法:

1、使用POSIX线程库(pthread)(C)

        POSIX线程库是一套用于多线程编程的标准库,在大部分类UNIX系统上都可以使用。可以使用pthread_create()函数来创建线程。

// C
#include <pthread.h>

void *threadFunc(void *arg) {
    // 线程的代码逻辑
    return NULL;
}

int main() {
    pthread_t tid;
    pthread_create(&tid, NULL, threadFunc, NULL);
    // 主线程的代码逻辑
    pthread_join(tid, NULL); // 等待子线程执行完毕
    return 0;
}

在Linux的项目中的创建线程的实例:

#include <pthread.h>

// 创建线程
void fun()
{
    pthread_t tid;
    int err;
    err = pthread_create(&tid, NULL, createGetTextThread, this);
}

// 线程走的函数
void* createGetTextThread(void *arg)//this是哪个类,就转换那个类的指针即可
{
    Ctrl *p = (Ctrl *)arg;
    while(true)
    {
        //todo
      
        QThread::msleep(150);
    }
    return nullptr;
}
2、 使用Windows线程库(Win32 API)(C)

        在Windows操作系统上,可以使用Win32 API来创建线程。可以使用CreateThread()函数来创建线程。

(1)CreateThread()的原型为:

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

参数说明:

  • lpThreadAttributes:线程的安全属性,默认为NULL。
  • dwStackSize:线程堆栈的大小,默认为0,表示使用默认的堆栈大小。
  • lpStartAddress:线程的入口函数地址,可以是一个函数指针或者线程函数的名字。
  • lpParameter:传递给线程函数的参数。
  • dwCreationFlags:线程的创建标志,可以设置一些额外的选项,如CREATE_SUSPENDED(创建后暂停)等。
  • lpThreadId:用于接收线程ID的指针。

CreateThread函数创建一个新的线程,并返回一个指向线程的句柄。通过这个句柄,可以对线程进行一些操作,如等待线程结束、挂起或恢复线程等。

(2)下面是一个简单的示例,演示如何使用CreateThread创建一个新的线程:

// c
#include <windows.h>

DWORD WINAPI threadFunc(LPVOID lpParam) {
    // 线程的代码逻辑
    return 0;
}

int main() {
    HANDLE hThread = CreateThread(NULL, 0, threadFunc, NULL, 0, NULL);
    // 主线程的代码逻辑
    WaitForSingleObject(hThread, INFINITE); // 等待子线程执行完毕
    CloseHandle(hThread);
    return 0;
}
3、使用C++11标准库

        在C++11标准中,引入了std::thread类,可以方便地创建和管理线程。

// c++
#include <thread>

void threadFunc() {
    // 线程的代码逻辑
}

int main() {
    std::thread t(threadFunc); // 创建线程
    // 主线程的代码逻辑
    t.join(); // 等待子线程执行完毕
    return 0;
}

4、AfxBeginThread (MFC)

        AfxBeginThread 是一个 MFC(Microsoft Foundation Class)函数,用于创建一个新的线程。它的含义是在应用程序中启动一个新的线程,以便在后台执行一些任务,而不会阻塞主线程的运行。

AfxBeginThread 函数的原型如下:
CWinThread* AfxBeginThread(
AFX_THREADPROC pfnThreadProc,
LPVOID pParam,
int nPriority = THREAD_PRIORITY_NORMAL,
UINT nStackSize = 0,
DWORD dwCreateFlags = 0,
LPSECURITY_ATTRIBUTES lpSecurityAttrs = NULL
);

参数说明:

  • pfnThreadProc:指向线程函数的指针,该线程函数将在新线程中执行。
  • pParam:传递给线程函数的参数。
  • nPriority:新线程的优先级,默认为 THREAD_PRIORITY_NORMAL。
  • nStackSize:新线程的堆栈大小,默认为 0,表示使用默认堆栈大小。
  • dwCreateFlags:创建线程的标志,默认为 0。
  • lpSecurityAttrs:线程安全属性,默认为 NULL。

AfxBeginThread 函数将返回一个 CWinThread 对象的指针,可以使用该指针来管理和控制新线程的行为,如等待线程结束、获取线程ID等。

总之,AfxBeginThread 函数用于创建一个新的线程,并在新线程中执行指定的函数,以实现多线程编程。

实例:

// 声明一个变量,继承于CWinThread
UINT ReceiveMsgThreadID;
CThreadReceiveRespMsg *pThreadReceiveMsg;

//启动线程	
pThreadReceiveMsg = (CThreadReceiveRespMsg *)::AfxBeginThread(RUNTIME_CLASS(CThreadReceiveRespMsg), THREAD_PRIORITY_NORMAL);
if (pThreadReceiveMsg != NULL)
{
	log(_T("线程1启动成功 \n"));
	ReceiveMsgThreadID = pThreadReceiveMsg->m_nThreadID;
}
else
{
	log(_T("线程1启动失败 \n"));
}

以上是几种常用的创建线程的方法,选择哪种方法取决于你的需求和运行环境。

二、线程的应用(常用实例,附上完整代码)

1、CreateThread创建线程的实例
// 头文件
#pragma once
#include "singleton.h"
#include <windows.h>

class CThreadBase
{
public:
	CThreadBase(void);
	virtual ~CThreadBase(void);

public:
	// 开启线程
	void vStart();

	// 结束线程
	virtual void vStop();

	// 线程是否可运行
	virtual bool isCanRun();

	// 子类重写此函数用于接收线程逻辑
	virtual int circle();

	// 入锁、解锁临界区,不同的编译环境修改成其他的同步条件即可
	void _lock();
	void _unLock();
	
protected:
	DWORD m_dThreadId;						// 线程ID	
	bool m_bCanRun;							// 是否可以运行
	int m_nThreadHandle;					// 线程句柄

#ifdef WINVER
	CRITICAL_SECTION m_criSection;		// 临界区,定义一个,需要数据同步的自取即可
#endif
};
// 源文件
#include "ThreadBase.h"
//#include <afxwin.h>

// 线程函数
DWORD WINAPI threadProc(LPVOID pArgs)
{
	if (0 == pArgs)
		return 0;

	CThreadBase* pThreadBase = (CThreadBase*)pArgs;
	pThreadBase->circle();
	return 0;
}

CThreadBase::CThreadBase(void)
{
	m_bCanRun = false;
	m_dThreadId = 0;
	m_nThreadHandle = NULL;

#ifdef WINVER
	InitializeCriticalSection(&m_criSection);
#endif
}

CThreadBase::~CThreadBase(void)
{
}

void CThreadBase::vStart()
{
	m_nThreadHandle = (int)CreateThread(NULL, 0, threadProc, this, 0, &m_dThreadId);
	m_bCanRun = true;
}

void CThreadBase::vStop()
{
	m_bCanRun = false;
}

bool CThreadBase::isCanRun()
{
	return m_bCanRun;
}

int CThreadBase::circle()
{
	return TRUE;
}

void CThreadBase::_lock()
{
#ifdef WINVER
	EnterCriticalSection(&m_criSection);
#endif
}

void CThreadBase::_unLock()
{
#ifdef WINVER
	LeaveCriticalSection(&m_criSection);
#endif
}

使用时候:

(1)直接继承这个类,子类重写circle() 这个函数即可,这个函数用于接收线程的逻辑;

(2)想让线程执行的功能,可以写在子类的circle()中;文章来源地址https://www.toymoban.com/news/detail-558086.html

到了这里,关于线程的创建以及使用(MFC/C/C++)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • MFC 非线程创建模态化窗口 实现工具栏拓展

    1.1 在Dlg.h文件中声明变量和定义资源ID 1.2 在资源视图中导入Bitmap资源图片 可以在右击资源符号中查看具体ID值 1.3 在Dlg.cpp文件中的OnInitDialog()方法中导入Bitmap与按钮ID 1.4 开启鼠标停靠信息提示 先在Dlg.h中增加函数声明 再Dlg.cpp中实现 增加消息映射宏开启提示 1.5 效果呈现 2.

    2024年02月10日
    浏览(27)
  • 什么是线程?线程和进程的关系?如何创建/查看线程?

    1.1.1 什么是线程 进程进一步细化为线程, 是程序内部的一条执行路径. 一个进程中至少有一个线程. 每个线程之间都可以按照顺讯执行自己的代码. 多个线程之间\\\"同时\\\"执行多份代码. 1.1.2 线程存在的意义 ① “并发编程\\\"成为\\\"刚需” 单核 CPU 的发展遇到了瓶颈. 要想提高算力, 就

    2024年02月15日
    浏览(29)
  • 进程与线程、线程创建、线程周期、多线程安全和线程池(ThreadPoolExecutor)

    什么进程? 进程是 资源(CPU、内存等)分配 的基本单位,它是程序执行时的一个实例。程序运行时系统就会创建一个进程,并为它分配资源,然后把该进程放入进程就绪队列,进程调度器选中它的时候就会为它分配CPU时间,程序开始真正运行。 什么是线程? 线程是操作系

    2024年02月14日
    浏览(28)
  • 僵尸进程的避免 守护进程的创建 线程的创建,阻塞,数据传递 5.15

    父子进程相关知识: exit(int status):结束当前调用的进程,自动刷新缓存 标准库函数 头文件:#include stdlib.h _exit(int status) : 结束当前调用的进程,不刷新缓存 系统调用函数 头文件:#include unistd.h 子进程先于父进程结束,父进程没有回收子进程的剩余资源 1、子进程结束,通知

    2024年02月05日
    浏览(30)
  • Java并发(三)----创建线程的三种方式及查看进程线程

    例如: 输出 注意:这里通过 @Slf4j 注解打印的日志 把【线程】和【任务】(要执行的代码)分开 Thread 代表线程 Runnable 可运行的任务(线程要执行的代码) 例如: 输出 Java 8 以后可以使用 lambda 精简代码 小结 方法1 是把线程和任务合并在了一起,方法2 是把线程和任务分开

    2023年04月24日
    浏览(39)
  • Java并发(1)--线程,进程,以及缓存

    进程 进程是程序的一次执行过程,系统程序的基本单位。有自己的main方法,并且主要由主方法运行起来的基本上就是进程。 线程 线程与进程相似,但线程是一个比进程更小的执行单位。一个进程在其执行的过程中可以产生多个线程。与进程不同的是同类的多个线程共享 堆

    2024年04月16日
    浏览(36)
  • Java进程线程介绍创建和执行销毁并理解线程安全和线程池 Native Method

    进程和线程都是一个控制流程。 一个进程通常对应于一个程序。 一个程序可以由多个不同的线程构成。 一个进程就是一个应用程序 一个应用程序面对多个用户则多个进程 一个进程则多个线程 多个线程共享资源且携带数据操作 多进程资源隔离 多线程的核心在于多个代码块

    2024年02月02日
    浏览(39)
  • MFC中使用多线程

    一、在MFC中使用多线程,可以通过CWinThread类来实现。下面是一个简单的示例,演示了如何在MFC应用程序中创建和使用多线程: 在你的MFC应用程序中包含头文件 \\\"afxmt.h\\\",该头文件包含了多线程相关的类和函数。 创建一个派生自CWinThread的自定义线程类。示例代码如下: 在自定

    2024年02月11日
    浏览(35)
  • AfxBeginThread线程函数的使用(MFC)

    在MFC中,开启线程函数推荐使用 AfxBeginThread 函数,返回的是CWinThread类,可以响应消息 线程函数在接受的参数需要进行 类型转换 需传入多个参数时可将多个参数定义为一个 结构体 ,然后再传递给线程函数 可将当前窗口指针作为参数传递到线程函数中,线程函数即可调用窗

    2024年02月07日
    浏览(29)
  • 关于进程、线程、协程的概念以及Java中的应用

    本文将从“操作系统”、“Java应用”上两个角度来探究这三者的区别。 在我本人的疑惑中,我有以下3个问题。 在“多道程序环境下”,允许多个程序并发执行,此时它们将失去封闭性,并具有间断性以及不可再现性的特征,因此需要引入进程的概念。 进程是程序执行的过

    2024年02月08日
    浏览(50)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包