MFC 创建UI 线程 || 无法解析的外部符号 "public: virtual struct CRuntimeClass

本文介绍了MFC中UI线程和工作线程的应用方法,包括如何创建UI线程及工作线程,解决常见错误,如外部符号未解析的问题,并提供了一个具体的CAddDeviceApp类实例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

MFC 分UI线程和工作线程,一般现在的应用程序都是一个主UI线程和N个工作线程来完成工作。主UI线程获取到工作线程发送的信息来刷新界面。


下面的方发是使用UI线程的方发。

1、UI线程,继承CWinThread类

 1 class CAddDeviceApp : public CWinThread
 2 {
 3     DECLARE_DYNCREATE(CAddDeviceApp)
 4 protected:
 5     CAddDeviceApp();
 6 public:
 7     virtual BOOL InitInstance();
 8     virtual int ExitInstance();
 9 protected:
10     virtual ~CAddDeviceApp();
11     DECLARE_MESSAGE_MAP()
12 
13 };


线程InitInstance

 1 BOOL CAddDeviceApp::InitInstance()
 2 {
 3     CSecondThreadDlg dlg;
 4     m_pMainWnd = &dlg;
 5     INT_PTR nResponse = dlg.DoModal();
 6     if (nResponse == IDOK)
 7     {
 8     }
 9     else if (nResponse == IDCANCEL)
10     {
11     }
12     return TRUE;
13 }
m_pMainWnd = &dlg; 加上这个之后,UI线程会独立处理消息循环,启动的UI线程DoModal对话框不会阻塞主线程的对话框。

启动UI线程:
1 CAddDeviceApp * pThread =  (CAddDeviceApp*)AfxBeginThread(RUNTIME_CLASS(CAddDeviceApp));

2、工作线程
目前工作在MFC中使用的工作线程有:MFC线程,C Run运行时线程,Boost线程。

1 boost::thread thrd(BoostThreadFunc);                     
2     
3 _beginthread(CRunThreadFunc,0,NULL);                             //不用
4 _beginthreadex(NULL, 0, ThreadFunEx, NULL, 0, NULL);             
5 pThread=AfxBeginThread(ThreadFunc,NULL,THREAD_PRIORITY_NORMAL); 


BOOST线程创建的方式种类比较多,可以通过函数对象,Boost::bind成员函数等多种方式创建,由于工作没太多时间,就不总结了。                   

一般MFC使用AfxBeginThread比较安全。

*******************************注意***********************************************

如果出现无法解析的外部符号 "public: virtual struct CRuntimeClass * __thiscall CAddDeviceApp::GetRuntimeClass(void)const "

等错误:

以下原因是会引起上述错误的:   1,在.h文件中写了DECLARE_DYNAMIC,而在.cpp文件中没有写IMPLEMENT_DYNAMIC    2,在.h文件中写了DECLARE_DYNCREATE ,但在.cpp文件中没有写上IMPLEMENT_DYNCREATE

在编写自定义类时,你必须知道,如果在类定义中包含了DECLARE_DYNAMIC,那你必须在类声明中包含IMPLEMENT_DYNAMIC;如果在类定义中包含了DECLARE_DYNCREATE,你必须在类声明中包含IMPLEMENT_DYNCREATE

.h类中定义

DECLARE_DYNCREATE(CMyWinThread)

DECLARE_MESSAGE_MAP()

.cpp类中定义

IMPLEMENT_DYNAMIC(CMyWinThread,CWinThread)   BEGIN_MESSAGE_MAP(CMyWinThread, CWinThread)    END_MESSAGE_MAP()

 

只需要记住:消息映射在.h文件和.cpp文件中是一一对应的。

****************************************************************

如果运行时出现内存不足情况:

把创建线程的代码改为:

CAddDeviceApp * pThread=NULL;
pThread= new CAddDeviceApp();
pThread->CreateThread();



CAddDeviceApp类的定义:

class CAddDeviceApp : public CWinThread
{
     DECLARE_DYNCREATE(CAddDeviceApp)
public:
     CAddDeviceApp();
public:
     virtual BOOL InitInstance();
     virtual int ExitInstance();
//void ShowNormalMsg(LPCTSTR lpszText);
public:

    virtual ~CAddDeviceApp();
    DECLARE_MESSAGE_MAP()


};


实现的方法:

CAddDeviceApp::CAddDeviceApp()
{
}
CAddDeviceApp::~CAddDeviceApp()
{
}
IMPLEMENT_DYNAMIC(CAddDeviceApp,CWinThread)   
BEGIN_MESSAGE_MAP(CAddDeviceApp, CWinThread)    
END_MESSAGE_MAP()


BOOL CAddDeviceApp::InitInstance()
{
INT_PTR nRes;
CNewDiaDlg stat_T;//模态对话框的类
m_pMainWnd = &stat_T;
nRes = stat_T.DoModal();

if (IDD_DIALOG1 == nRes)
return TRUE;


}
int CAddDeviceApp::ExitInstance()
{
return CWinThread::ExitInstance();
}

结束线程关闭模态对话框的用法:

HWND hPopuphWnd =::GetForegroundWindow();  //获取当前正在运行窗口的句柄
::PostMessage(hPopuphWnd,WM_CLOSE,0,0);

注意:使用线程弹出的模态对话框作为当前正在运行的窗口,想要正确的关闭,必须先关闭模态对话框

再运行主线程的窗口。

这个错误是由于在编译过程中,找不到与名称为"public: virtual struct qmetaobject const"相关的外部符号所导致的。 在C++中,当我们声明和定义一个类的成员函数时,编译器需要找到对应的函数定义来生成可执行代码。如果找不到对应的定义,就会出现"无法解析外部符号"错误。 根据错误信息中提到的"qmetaobject",推测可能是引用了Qt框架中的某个类或函数,但链接器无法找到相关的定义。要解决这个错误,我们可以参考以下几个步骤: 1. 确认编译环境是否已正确配置Qt框架。在使用Qt框架开发程序时,需要先配置好编译器和IDE。确保已正确安装Qt框架,并在项目设置中配置好相关的库文件和头文件路径。 2. 检查代码中是否正确包含了相关的头文件。如果使用了Qt中的类或函数,需要在代码中包含相应的头文件。例如,如果使用了QMetaObject类,需要在代码中包含#include <QMetaObject>语句。 3. 检查代码中是否正确链接了Qt库。在项目设置中,需要将Qt相关的库文件链接到项目中。这可以通过在项目配置中添加Qt库的名称或路径来实现。 4. 确保Qt库的版本与项目设置兼容。如果Qt库的版本不匹配,可能会导致链接错误。需要确定项目使用的Qt库版本,并将其配置为项目设置中的Qt版本。 通过检查以上几个因素,应该能够解决"error lnk2001: 无法解析外部符号"错误。如果问题仍然存在,可以查看详细的错误日志以获取更多信息,或者参考Qt官方文档和社区解决方案来解决该问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值