Multithreading 有没有一种可移植的方法可以用Qt命名线程?
我知道我可以在Linux中设置线程名称(gdb和htop中可见的名称)。但对于另一个操作系统,这很可能不起作用。此外,我还可以尝试,这在POSIX系统中更为可用,但仍然缺乏完全的兼容性Multithreading 有没有一种可移植的方法可以用Qt命名线程?,multithreading,qt,portability,Multithreading,Qt,Portability,我知道我可以在Linux中设置线程名称(gdb和htop中可见的名称)。但对于另一个操作系统,这很可能不起作用。此外,我还可以尝试,这在POSIX系统中更为可用,但仍然缺乏完全的兼容性 所以我想有一些更可移植的方法,可能是QThread提供的一些我没有找到的东西。有这样的方法吗?在QThreadAPI中没有手动管理线程的系统名称,但是,自4.8.3版以来,Qt将自动将线程的名称设置为线程对象的名称(QObject::objectName()) 这在QThread的实现中处理,如下所述 你有这样的
所以我想有一些更可移植的方法,可能是
QThread
提供的一些我没有找到的东西。有这样的方法吗?在QThread
API中没有手动管理线程的系统名称,但是,自4.8.3版以来,Qt将自动将线程的名称设置为线程对象的名称(QObject::objectName()
)
这在QThread
的实现中处理,如下所述
你有这样的想法:
与之相当的是:
请注意,在Windows上,如果设置了QT\u NO\u DEBUG
,则不会执行上述代码,因此在发布模式下不会工作中,您可以找到:
选择线程将被赋予的名称(由
例如,在Linux上使用命令ps-L),您可以调用setObjectName()
在启动线程之前。如果不调用setObjectName(),则
给线程的名称将是运行时类型的类名
线程对象(例如,在
Mandelbrot示例,因为这是QThread子类的名称)。注
这在Windows上的版本中当前不可用
您可以使用编译标志,这取决于您为之编译的操作系统。boost没有办法,所以我怀疑QT是否会这样。这似乎只适用于足够新的QT版本。我没有看到@Marek在4.8.1中引用的行为。相反,我的所有线程的名称都与二进制文件的名称一致。另外,
setCurrentThreadName
的grep
在发行版提供的源代码树中找不到任何内容。无论如何,这似乎是Qt中第一个真正支持它的功能,所以我接受你的回答。@Ruslan是的,正如你所看到的,该功能是为了调试而实现的。事实上,我很惊讶Qt文档中引用了它,因为它只不过是一个调试工作区。
#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());
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());