java当中的线程和操作系统的线程是什么关系?
关于操作系统的线程linux操作系统的线程控制原语
int pthread create(pthread t *thread, const pthread attr t *attr,void *(*start_routine) (void *), void *arg);
可以在linux系统下面通过man手册查看该函数的定义
根据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方法的源码的部分截图
可以看到这个方法最核心的就是调用了一个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
结果如图所示
结果是两个线程一直在交替执行,得到我们预期的结果。现在的问题就是我们如何通过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线程的模型和本质