我知道我可以在Linux using prctl()
中设置线程名称(gdb和htop中可见的名称)。但是对于其他操作系统,这很可能不会起作用。另外,我可以尝试using pthread_setname_np()
,这在POSIX系统中更有用,但仍然缺乏完全兼容性。
所以我希望有一些更便携的方式,也许QThread
提供了我找不到的东西。有没有这样的方式?
答案 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上的发布版本中不可用。