linux线程的栈大小 每个线程有自己单独的栈吗
很多朋友对于linux线程的栈大小和每个线程有自己单独的栈吗不太懂,今天就由小编来为大家分享,希望可以帮助到大家,下面一起来看看吧!
Linux系统默认堆栈大小分析linux默认堆栈大小
现代操作系统中,Linux系统成为一种新兴的操作系统,它一般各种计算机硬件平台上都得到了广泛的应用。Linux系统的出现给用户带来了各种便捷。其中,Linux系统默认的堆栈大小,可能会对Linux的稳定和性能造成影响。下面就来分析Linux系统中默认的堆栈大小。
首先,Linux系统上所有进程的默认堆栈大小为8M。由于进程会使用大量的API函数,而大多数API函数会在堆栈上保存一些重要的参数,因此堆栈的大小会影响进程运行的稳定性和性能。
其次,如果Linux系统的进程的堆栈大小太小的话,将会影响进程的运行。当进程调用API函数时,如果堆栈上的空间受限,则进程可能会发生段错误,导致程序的异常终止。
最后,Linux的堆栈大小可以通过ulimit-s来查看,也可通过/etc/security/limits.conf来进行调整,以满足不同进程动态申请更多堆栈空间的需求。比如:
//查看默认堆栈大小
$ ulimit-s
8192
//调整默认堆栈大小
$ vi/etc/security/limits.conf
*– stack 16384
以上,就是对Linux系统中默认堆栈大小的分析,虽然默认堆栈大小为8M,但是用户可以根据自己的需要调整堆栈大小,来满足进程运行的安全性和高效性的要求。
进程内核栈,用户栈及 Linux 进程栈和线程栈的区别
总结:线程栈的空间开辟在所属进程的堆区,线程与其所属的进程共享进程的用户空间,所以线程栈之间可以互访。线程栈的起始地址和大小存放在pthread_attr_t中,栈的大小并不是用来判断栈是否越界,而是用来初始化避免栈溢出的缓冲区的大小(或者说安全间隙的大小)
进程内核栈、用户栈
1.进程的堆栈
内核在创建进程的时候,在创建task_struct的同事,会为进程创建相应的堆栈。每个进程会有两个栈,一个用户栈,存在于用户空间,一个内核栈,存在于内核空间。当进程在用户空间运行时,cpu堆栈指针寄存器里面的内容是用户堆栈地址,使用用户栈;当进程在内核空间时,cpu堆栈指针寄存器里面的内容是内核栈空间地址,使用内核栈。
2.进程用户栈和内核栈的切换
当进程因为中断或者系统调用而陷入内核态之行时,进程所使用的堆栈也要从用户栈转到内核栈。
进程陷入内核态后,先把用户态堆栈的地址保存在内核栈之中,然后设置堆栈指针寄存器的内容为内核栈的地址,这样就完成了用户栈向内核栈的转换;当进程从内核态恢复到用户态之行时,在内核态之行的最后将保存在内核栈里面的用户栈的地址恢复到堆栈指针寄存器即可。这样就实现了内核栈和用户栈的互转。
那么,我们知道从内核转到用户态时用户栈的地址是在陷入内核的时候保存在内核栈里面的,但是在陷入内核的时候,我们是如何知道内核栈的地址的呢?
关键在进程从用户态转到内核态的时候,进程的内核栈总是空的。这是因为,当进程在用户态运行时,使用的是用户栈,当进程陷入到内核态时,内核栈保存进程在内核态运行的相关信心,但是一旦进程返回到用户态后,内核栈中保存的信息无效,会全部恢复,因此每次进程从用户态陷入内核的时候得到的内核栈都是空的(为什么?)。所以在进程陷入内核的时候,直接把内核栈的栈顶地址给堆栈指针寄存器就可以了。
3.内核栈的实现
内核栈在kernel-2.4和kernel-2.6里面的实现方式是不一样的。
在kernel-2.4内核里面,内核栈的实现是:
Union task_union{
Struct task_struct task;
Unsigned long stack[INIT_STACK_SIZE/sizeof(long)];
};
其中,INIT_STACK_SIZE的大小只能是8K。
内核为每个进程分配task_struct结构体的时候,实际上分配两个连续的物理页面,底部用作task_struct结构体,结构上面的用作堆栈。使用current()宏能够访问当前正在运行的进程描述符。
注意:这个时候task_struct结构是在内核栈里面的,内核栈的实际能用大小大概有7K。
内核栈在kernel-2.6里面的实现是(kernel-2.6.32):
Union thread_union{
Struct thread_info thread_info;
Unsigned long stack[THREAD_SIZE/sizeof(long)];
};
其中THREAD_SIZE的大小可以是4K,也可以是8K,thread_info占52bytes。
当内核栈为8K时,Thread_info在这块内存的起始地址,内核栈从堆栈末端向下增长。所以此时,kernel-2.6中的current宏是需要更改的。要通过thread_info结构体中的task_struct域来获得于thread_info相关联的task。更详细的参考相应的 current宏的实现。
struct thread_info{
struct task_struct*task;
struct exec_domain*exec_domain;
__u32 flags;
__u32 status;
__u32 cpu;
…..
};
注意:此时的task_struct结构体已经不在内核栈空间里面了。
linux下一个线程占用多少内存
对于您提出的这个问题,是不能够直接和准确的回答的;因为对于线程来说,应该对于任何的进程来说,分配的资源(包括内存)都是有限的,但是非固定的;如果您一定要说一些特殊情况,在线程池的管理中,线程池内的线程分配的内存就可以确定内存的大小,但是这任然是可变的。
所以,对于您提出这种方式的问题,建议您先熟悉以下几点:
什么是线程?
线程的工作是什么?
线程和进程的区别是什么?
线程的基本运用。