Warning: file_get_contents(/data/phpspider/zhask/data//catemap/8/qt/6.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Multithreading 有没有一种可移植的方法可以用Qt命名线程?_Multithreading_Qt_Portability - Fatal编程技术网

Multithreading 有没有一种可移植的方法可以用Qt命名线程?

Multithreading 有没有一种可移植的方法可以用Qt命名线程?,multithreading,qt,portability,Multithreading,Qt,Portability,我知道我可以在Linux中设置线程名称(gdb和htop中可见的名称)。但对于另一个操作系统,这很可能不起作用。此外,我还可以尝试,这在POSIX系统中更为可用,但仍然缺乏完全的兼容性 所以我想有一些更可移植的方法,可能是QThread提供的一些我没有找到的东西。有这样的方法吗?在QThreadAPI中没有手动管理线程的系统名称,但是,自4.8.3版以来,Qt将自动将线程的名称设置为线程对象的名称(QObject::objectName()) 这在QThread的实现中处理,如下所述 你有这样的

我知道我可以在Linux中设置线程名称(gdb和htop中可见的名称)。但对于另一个操作系统,这很可能不起作用。此外,我还可以尝试,这在POSIX系统中更为可用,但仍然缺乏完全的兼容性


所以我想有一些更可移植的方法,可能是
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());