文章插图
大家好,我是坤哥
网上看到一个很有意思的美团面试题:为什么线程崩溃崩溃不会导致 JVM 崩溃,这个问题我看了不少回答,但发现都没答到根上,所以决定答一答,相信大家看完肯定会有收获,本文分以下几节来探讨
- 线程崩溃,进程一定会崩溃吗
- 进程是如何崩溃的-信号机制简介
- 为什么在 JVM 中线程崩溃不会导致 JVM 进程崩溃
- openJDK 源码解析
文章插图
线程共享代码段,数据段,地址空间,文件
非法访问内存有以下几种情况,我们以 C 语言举例来看看
- 针对只读内存写入数据
- #include <stdio.h>
#include <stdlib.h>
int main() {
char *s = "hello world";// 向只读内存写入数据,崩溃
s[1] = 'H';
} - 访问了进程没有权限访问的地址空间(比如内核空间)
- #include <stdio.h>
#include <stdlib.h>
int main() {
int *p = (int *)0xC0000fff; // 针对进程的内核空间写入数据,崩溃
*p = 10;
} - 在 32 位虚拟地址空间中,p 指向的是内核空间,显然不具有写入权限,所以上述赋值操作会导致崩溃
- 访问了不存在的内存,比如
- #include <stdio.h>
#include <stdlib.h>
int main() {
int *a = NULL;
*a = 1;
}
进程是如何崩溃的-信号机制简介那么线程崩溃后,进程是如何崩溃的呢,这背后的机制到底是怎样的,答案是信号,大家想想要干掉一个正在运行的进程是不是经常用 kill -9 pid 这样的命令,这里的 kill 其实就是给指定 pid 发送终止信号的意思,其中的 9 就是信号,其实信号有很多类型的,在 linux 中可以通过 kill -l查看所有可用的信号
文章插图
当然了发 kill 信号必须具有一定的权限,否则任意进程都可以通过发信号来终止其他进程,那显然是不合理的,实际上 kill 执行的是系统调用,将控制权转移给了内核(操作系统),由内核来给指定的进程发送信号
那么发个信号进程怎么就崩溃了呢,这背后的原理到底是怎样的?
其背后的机制如下
- CPU 执行正常的进程指令
- 调用 kill 系统调用向进程发送信号
- 进程收到操作系统发的信号,CPU 暂停当前程序运行,并将控制权转交给操作系统
- 调用 kill 系统调用向进程发送信号(假设为 11,即 SIGSEGV,一般非法访问内存报的都是这个错误)
- 操作系统根据情况执行相应的信号处理程序(函数),一般执行完信号处理程序逻辑后会让进程退出
// 自定义信号处理函数示例#include <stdio.h>#include <signal.h>#include <stdlib.h>// 自定义信号处理函数,处理自定义逻辑后再调用 exit 退出void sigHandler(int sig) {printf("Signal %d catched!n", sig);exit(sig);}int main(void) {signal(SIGSEGV, sigHandler);int *p = (int *)0xC0000fff;*p = 10; // 针对不属于进程的内核空间写入数据,崩溃}// 以上结果输出: Signal 11 catched!
推荐阅读
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- 《我的前半生》贺涵为什么喜欢罗子君-,我的后半生罗子君和贺涵在一起了吗-
- 幻视为什么克制奥创,幻视是奥创吗-
- 一个单位的风气很重要,一个单位的风气为什么不正-
- 周星驰一下子老了,周星驰为什么变那么老-
- 为什么蝴蝶飞舞没有声音,而蜜蜂却嗡嗡响?
- 慈禧太后为什么不喜欢珍妃,慈禧是怎么把珍妃推入井中的-
- 二驴为什么说辛巴,辛巴哭着骂二驴-
- 为什么佛教起源于印度但没在印度流传下来,印度是佛教的发源地为什么印度没有佛教-
- 为什么看以前的照片觉得好看,照相看起来很老-
- 日本校服女装裙子为什么那么短,日本女生校服为什么这么短-