有没有一种可移植的方式来用 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 给线程名称?的主要内容,如果未能解决你的问题,请参考以下文章

标准 C++ 中是不是有一种可移植的方式来检索主机名?

是否有一种可移植的方式来定义 INT_MAX?

是否有一种可移植的方式在 Makefile 中制作可移植的可选依赖项?

有没有一种可移植的方法来设置flock() 的超时时间?

是否有一种可移植的(C++ 标准)方法来计算前一个对齐的指针?

以编程方式确定JBoss 5.1中的JNDI根上下文/ ear名称