按理说,都是某个进程下的线程, 应该进程id PID一样啊,但实际却都不一样
实际是被PID的名字给弄混了,线程进程都会有自己的ID,这个ID就叫做PID,PID是不特指进程ID,线程ID也可以叫做PID。
pthread库里的每一个线程都对应一个内核线程,都是有单独的pid。
The four threads will have the same PID but only when viewed from above. What you (as a user) call a PID is not what the kernel (looking from below) calls a PID.
In the kernel, each thread has it's own ID, called a PID (although it would possibly make more sense to call this a TID, or thread ID) and they also have a TGID (thread group ID) which is the PID of the thread that started the whole process.
Simplistically, when a new process is created, it appears as a thread where both the PID and TGID are the same (new) number.
When a thread starts another thread, that started thread gets its own PID (so the scheduler can schedule it independently) but it inherits the TGID from the original thread.
That way, the kernel can happily schedule threads independent of what process they belong to, while processes (thread group IDs) are reported to you.
关于线程继承关系图如下:
USER VIEW<-- PID 43 --> <----------------- PID 42 ----------------->+---------+| process |_| pid=42 |__/ | tgid=42 | \_ (new thread) __ (fork) _/ +---------+ \/ +---------+
+---------+ | process |
| process | | pid=44 |
| pid=43 | | tgid=42 |
| tgid=43 | +---------+
+---------+<-- PID 43 --> <--------- PID 42 --------> <--- PID 44 --->KERNEL VIEW
在这里你可以清晰的看到,创建一个新的进程会给一个新的PID和TGID,并且2个值相同,当创建一个新的线程的时候,会给你一个新的PID,并且TGID和之前开始的进程一致。
Linux通过进程查看线程的方法 1).htop
按t(显示进程线程嵌套关系)和H(显示线程) ,然后F4过滤进程名。2).ps -eLf | grep java
(快照,带线程命令,e是显示全部进程,L是显示线程,f全格式输出) 3).pstree -p <pid>
(显示进程树,不加pid显示所有) 4).top -Hp <pid>
(实时) 5).ps -T -p <pid>
(快照) 推荐程度按数字从小到大。
void * thread_start(void *arg)
{
printf("Process ID: %d, thread ID %d\n", getpid(), gettid());
}
#include <stdio.h>
#include <sys/types.h>
#include <sys/syscall.h>
#include <unistd.h>
void * thread_start(void *arg)
{
printf("[1] Process ID: %d, thread ID %d\n", getpid(), syscall(__NR_gettid));
printf("[2] Process ID: %d, thread ID %d\n", getpid(), syscall(SYS_gettid));
}