java的本质_Java线程本质

本文探讨了Java中的线程与操作系统线程的联系,通过Linux的pthread_create函数创建线程作为例子。在Java中,线程的启动通常通过调用start()方法,此方法最终会映射到操作系统级别的线程创建。通过分析Hotspot虚拟机源码,我们可以推测Java线程是基于操作系统线程的抽象。文章还介绍了如何使用JNI调用本地方法来模拟Java线程的启动过程,从而更深入理解Java线程的本质。
摘要由CSDN通过智能技术生成

java当中的线程和操作系统的线程是什么关系?

关于操作系统的线程linux操作系统的线程控制原语

int pthread create(pthread t *thread, const pthread attr t *attr,void *(*start_routine) (void *), void *arg);

可以在linux系统下面通过man手册查看该函数的定义

722f359ac53133774c2304b131b0c06c.png

根据man配置的信息可以得出pthread_create会创建一个线程,这个函数是linux系统的函数,可以用C或者C++直接调用,上面信息也告诉程序员这个函数在pthread.h,这个函数有四个参数

参数名字

参数定义

参数解释

pthread_t *thread

传出参数,调用之后会传出被创建

线程的id

定义 pthread_t pid;继而取地址

&pid

const pthread_attr_t

*attr

线程属性,关于线程属性是linux

的知识

一般传NULL,保持默认属性

void (start_routine)

(void *)

线程的启动后的主体函数

需要你定义一个函数,然后传函

数名即可

void *arg

主体函数的参数

没有可以传nulll

linux上启动一个线程的代码:

//头文件

#include <pthread.h>

#include<stdio.h>//定义一个变量,接受创建线程后的线程id

pthread_t pid;//定义线程的主体函数

void* thread entity(void*arg) {

printf("i am new Thread! from c");

}//main方法,程序入口,main和java的main一样会产生一个进程,继而产生一个main线程

intmain() {//调用操作系统的函数创建线程,注意四个参数

pthread create(&pid,NULL,thread entity,NULL);//usleep是睡眠的意思,那么这里的睡眠是让谁睡眠呢?//为什么需要睡眠?如果不睡眠会出现什么情况

usleep(100);

printf("main\n");

}

假设有了上面知识的铺垫,那么可以试想一下java的线程模型到底是什么情况呢?

在java代码里启动一个线程的代码

public classExample4Start {public static voidmain(String[] args) {

Thread thread= newThread(){

@Overridepublic voidrun() {

System.out.println("i am new Thread! from java ");

}

};

thread.start();

}

}

这里启动的线程和上面我们通过linux的pthread_create函数启动的线程有什么关系呢?只能去可以查看start()的源码了,看看java的start()到底干了什么事才能对比出来。start方法的源码的部分截图

b4244d17d7598ed36b3a7dd7c73f8757.png

可以看到这个方法最核心的就是调用了一个start0方法,而start0方法又是一个native方法,故而如果要搞明白start0我们需要查看Hotspot的源码,好吧那我们就来看一下Hotspot的源码吧,Hotspot的源码怎么看么?一般直接看openjdk的源码,openjdk的源码如何查看、编译调试?openjdk的编译我们后面会讨论,在没有openjdk的情况下,我们做一个大胆的猜测,java级别的线程其实就是操作系统级别的线程,什么意思呢?说白了我们大胆猜想star----->start0------------>ptherad_create

我们鉴于这个猜想来模拟实现一下java启动线程

public classEnjoyThread {public static voidmain(String[] args) {//自己定义的一个类

EnjoyThread enjoythread =newEnjoyThread();

enjoythread.start0();

}//本地方法

private native voidstart0();

}

这里我们让自己写的start0调用一个本地方法,在本地方法里面去启动一个系统线程,好吧我们写一个c程序来启动本地线程

本地方法的代码编写

#include <pthread.h>

#include<stdio.h>//定义变量接受线程id

pthread t pid;//线程的主体方法相当于 java当中的run

void* thread entity(void*arg) {//子线程死循环

while(1){//睡眠100毫秒

usleep(100);//打印

printf("I am new Thread\n");

}

}//c语言的主方法入口方法,相当于java的main

intmain() {//调用linux的系统的函数创建一个线程

pthread create(&pid,NULL,thread entity,NULL);//主线程死循环

while(1){//睡眠100毫秒

usleep(100);//打印

printf("I am main\n");

}

}

在linux上编译、运行上述C程序

编译这个程序

gcc thread.c -o thread.out -pthread

运行这个程序

./thread.out

结果如图所示

2aac925ced1e7dc754b3977a2dad4b27.png

结果是两个线程一直在交替执行,得到我们预期的结果。现在的问题就是我们如何通过start0调用这个c程序,这里就要用到JNI了

java利用JNI调用本地方法

packagecom.enjoy.concurrency;public classEnjoyThread {//装载库,保证JVM在启动的时候就会装载,故而一般是也给static

static{

System.loadLibrary("EnjoyThreadNative" );

}public static voidmain(String[] args) {

EnjoyThread enjoyThread=newEnjoyThread();

enjoyThread.start0();

}private native voidstart0();

}

这样完全还原java当中调用JVM源码启动线程的场景,继而可以系统的了解java线程的模型和本质

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值