是否有一种可移植的方式为Qt提供线程名称?

时间:2013-11-08 08:52:30

标签: multithreading qt portability

我知道我可以在Linux using prctl()中设置线程名称(gdb和htop中可见的名称)。但是对于其他操作系统,这很可能不会起作用。另外,我可以尝试using pthread_setname_np(),这在POSIX系统中更有用,但仍然缺乏完全兼容性。

所以我希望有一些更便携的方式,也许QThread提供了我找不到的东西。有没有这样的方式?

2 个答案:

答案 0 :(得分:7)

QThread API中没有任何内容可以手动管理线程的系统名称,但是,从版本4.8.3开始,Qt会自动将线程的名称设置为线程的名称对象(QObject::objectName())。

这在QThread的实现中处理,如下所述。

qthread_unix.cpp中有类似的内容:

#if (defined(Q_OS_LINUX) || defined(Q_OS_MAC) || defined(Q_OS_QNX))
static void setCurrentThreadName(pthread_t threadId, const char *name)
{
#  if defined(Q_OS_LINUX) && !defined(QT_LINUXBASE)
    Q_UNUSED(threadId);
    prctl(PR_SET_NAME, (unsigned long)name, 0, 0, 0);
#  elif defined(Q_OS_MAC)
    Q_UNUSED(threadId);
    pthread_setname_np(name);
#  elif defined(Q_OS_QNX)
    pthread_setname_np(threadId, name);
#  endif
}
#endif

/* 
 * [...]
 */

QString objectName = thr->objectName();

if (Q_LIKELY(objectName.isEmpty()))
    setCurrentThreadName(thr->d_func()->thread_id, thr->metaObject()->className());
else
    setCurrentThreadName(thr->d_func()->thread_id, objectName.toLocal8Bit());

qthread_win.cpp中的等价物:

typedef struct tagTHREADNAME_INFO
{
    DWORD dwType;      // must be 0x1000
    LPCSTR szName;     // pointer to name (in user addr space)
    HANDLE dwThreadID; // thread ID (-1=caller thread)
    DWORD dwFlags;     // reserved for future use, must be zero
} THREADNAME_INFO;

void qt_set_thread_name(HANDLE threadId, LPCSTR threadName)
{
    THREADNAME_INFO info;
    info.dwType = 0x1000;
    info.szName = threadName;
    info.dwThreadID = threadId;
    info.dwFlags = 0;

    __try
    {
        RaiseException(0x406D1388, 0, sizeof(info)/sizeof(DWORD), (const ULONG_PTR*)&info);
    }
    __except (EXCEPTION_CONTINUE_EXECUTION)
    {
    }
}

/* 
 * [...]
 */

QByteArray objectName = thr->objectName().toLocal8Bit();
qt_set_thread_name((HANDLE)-1, objectName.isEmpty() ? thr->metaObject()->className() : objectName.constData());

请注意,在Windows上,如果设置了QT_NO_DEBUG,则上述代码将无法执行,因此它无法在发布模式下工作

答案 1 :(得分:6)

Qt documentation中,您可以找到:

  

选择线程的名称(由...标识)   例如,在Linux上的命令ps -L,你可以调用setObjectName()   在开始线程之前。如果你不调用setObjectName(),那么   给你的线程的名称将是运行时类型的类名   您的线程对象(例如,“RenderThread”)的情况   Mandelbrot示例,因为这是QThread子类的名称)。注意   目前,这在Windows上的发布版本中不可用。