8

你能帮我找到ID进程给定的主(唯一)线程ID吗?

任务上下文:正在运行的进程(目前)没有窗口,但有一个(一些)线程。

通缉:WM_QUIT仅在主线程发布。

不想要的:使用TerminateProcessWM_QUIT在非主要线程上发布。

4

6 回答 6

7

获取主线程的线程 id 的一种更简单、更可靠的方法是让主线程使用::GetCurrentThreadId()共享全局变量记录自己的线程 id,可能在您的 WinMain 或“主线程”开头的某个地方:

MainThreadId_G = ::GetCurrentThreadId();

然后在您的其他线程中,您可以调用:::PostThreadMessage(MainThreadId_G, WM_QUIT, returncode, 0);

于 2014-05-22T02:40:47.387 回答
6
#ifndef MAKEULONGLONG
#define MAKEULONGLONG(ldw, hdw) ((ULONGLONG(hdw) << 32) | ((ldw) & 0xFFFFFFFF))
#endif

#ifndef MAXULONGLONG
#define MAXULONGLONG ((ULONGLONG)~((ULONGLONG)0))
#endif

bool CloseProcessMainThread(DWORD dwProcID)
{
  DWORD dwMainThreadID = 0;
  ULONGLONG ullMinCreateTime = MAXULONGLONG;

  HANDLE hThreadSnap = CreateToolhelp32Snapshot(TH32CS_SNAPTHREAD, 0);
  if (hThreadSnap != INVALID_HANDLE_VALUE) {
    THREADENTRY32 th32;
    th32.dwSize = sizeof(THREADENTRY32);
    BOOL bOK = TRUE;
    for (bOK = Thread32First(hThreadSnap, &th32); bOK;
         bOK = Thread32Next(hThreadSnap, &th32)) {
      if (th32.th32OwnerProcessID == dwProcID) {
        HANDLE hThread = OpenThread(THREAD_QUERY_INFORMATION,
                                    TRUE, th32.th32ThreadID);
        if (hThread) {
          FILETIME afTimes[4] = {0};
          if (GetThreadTimes(hThread,
                             &afTimes[0], &afTimes[1], &afTimes[2], &afTimes[3])) {
            ULONGLONG ullTest = MAKEULONGLONG(afTimes[0].dwLowDateTime,
                                              afTimes[0].dwHighDateTime);
            if (ullTest && ullTest < ullMinCreateTime) {
              ullMinCreateTime = ullTest;
              dwMainThreadID = th32.th32ThreadID; // let it be main... :)
            }
          }
          CloseHandle(hThread);
        }
      }
    }
#ifndef UNDER_CE
    CloseHandle(hThreadSnap);
#else
    CloseToolhelp32Snapshot(hThreadSnap);
#endif
  }

  if (dwMainThreadID) {
    PostThreadMessage(dwMainThreadID, WM_QUIT, 0, 0); // close your eyes...
  }

  return (0 != dwMainThreadID);
}
于 2013-03-24T10:07:02.190 回答
2

我已经检查了在 MFC 中是如何处理的,看起来 UI 线程是由构造函数确定的:

C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\atlmfc\src\mfc\appcore.cpp:

CWinApp::CWinApp(LPCTSTR lpszAppName)
{
...

    m_nThreadID = ::GetCurrentThreadId();

并且使用 MFC 调用AfxGetApp()->m_nThreadID您可以找出 UI 线程 ID。

但是 - 如果 .dll 不是从主线程加载的,则此方法不起作用 - 然后即使 MFC 的方法也不起作用 -AfxGetApp()->m_nThreadID将返回主线程以外的其他内容。

但通常您的 .dll 是从主线程加载的,但您的 .dll 不需要启用 mfc。我可以推荐这样的方法:

class GetMainThread
{
public:
    GetMainThread()
    {
        m_nThreadID = ::GetCurrentThreadId();
    }

    DWORD m_nThreadID;
}getMainThread;


DWORD getUIThread()
{
    DWORD id = 0;

    if( AfxGetApp() != NULL )
        id = AfxGetApp()->m_nThreadID;
    else 
        id = getMainThread.m_nThreadID;

    return id;
} //getUIThread

如果 .dll 由主 UI 线程加载,您将从构造函数调用(GetMainThread 类)中获得正确的线程 ID。

AfxGetApp()如果您不需要它们,请删除它们(在我的应用程序中,我需要那些)

于 2016-08-25T09:49:14.853 回答
1

在 cpp 文件的开头使用它,而不是在函数中:

DWORD mainThreadID = ::GetCurrentThreadId();

这将在您的 main 函数被执行之前初始化 mainThreadID,这保证是主线程。

于 2020-06-24T20:32:14.670 回答
0

You can't post a message to a specific thread. Messages are put in a queue, and are processed from main thread only.

If you want to close windows process that has UI, take a look at using sendmessage to send wm_close to another process

于 2013-03-24T10:05:26.400 回答
0

您是否正在寻找此功能:

DWORD WINAPI GetProcessIdOfThreadId(_In_ DWORD ThreadId)
{
    HANDLE Thread = OpenThread(THREAD_QUERY_LIMITED_INFORMATION, FALSE, ThreadId);
    DWORD process_id = GetProcessIdOfThread(Thread);
    CloseHandle(Thread);
    return process_id;
}
于 2018-06-25T16:31:15.397 回答