有没有一种可移植的方式来用 Qt 给线程名称?
Posted
技术标签:
【中文标题】有没有一种可移植的方式来用 Qt 给线程名称?【英文标题】:Is there a portable way to give thread name with Qt? 【发布时间】:2013-11-08 08:52:29 【问题描述】:我知道我可以在 Linux using prctl()
中设置线程名称(在 gdb 和 htop 中可见)。但是对于其他操作系统,这很可能无法正常工作。另外,我可以试试using pthread_setname_np()
,它在 POSIX 系统中更可用,但仍然缺乏完全的兼容性。
所以我想要一些更便携的方式,也许是QThread
提供的我还没有找到的东西。有没有这样的方法?
【问题讨论】:
您可以使用编译标志,具体取决于您正在编译的操作系统。 boost 没有办法,所以我怀疑 QT 会 【参考方案1】: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
,上述代码将不会被执行,因此它不会在Release模式下工作。
【讨论】:
这似乎只适用于足够新的 Qt 版本。我没有看到@Marek 在 4.8.1 中引用的行为。相反,我所有的线程的名称都与二进制名称一致。此外,grep
for setCurrentThreadName
在发行版提供的源代码树中找不到任何内容。无论如何,这似乎是Qt中第一次真正支持这个,所以我会接受你的回答。
@Ruslan 是的,如您所见here,该功能是为了调试目的而实现的。我真的很惊讶它在 Qt 文档中被引用,因为它只不过是一种调试解决方法。【参考方案2】:
在Qt documentation你可以找到:
选择您的线程将被赋予的名称(由 例如,Linux 上的命令 ps -L),您可以调用 setObjectName() 在开始线程之前。如果您不调用 setObjectName(),则 为您的线程指定的名称将是运行时类型的类名 您的线程对象(例如,在 Mandelbrot 示例,因为那是 QThread 子类的名称)。笔记 这目前不适用于 Windows 上的发布版本。
【讨论】:
以上是关于有没有一种可移植的方式来用 Qt 给线程名称?的主要内容,如果未能解决你的问题,请参考以下文章
是否有一种可移植的方式在 Makefile 中制作可移植的可选依赖项?