今天成功模拟了java线程中,start方法调用run方法的过程,特此记录。
环境:centos7,jdk1.8 ,gcc
- 首先构建好一个java类,用来模拟线程。
public class MyThread {
//JVM加载自定义的C库,此时还未创建该库
static {
System.loadLibrary("MyThreadNativeLibrary");
}
//模拟run方法
public void run() {
System.out.println("I am a mythread , I'm running!");
}
//模拟start方法,调用start0
public void start() {
start0();
}
//native方法start0,两件事 1:在操作系统创建一个线程 2:该线程会调用run方法
private native void start0();
public static void main(String[] args) {
MyThread myThread = new MyThread();
myThread.start();
}
}
2. 将MyThread.java上传到linux系统中,执行命令,得到编译好的class文件。
javac MyThread.java
3.java文件生成对应的h文件。
javah MyThread
4.编写被java调用的C文件——NewThread.c。
#include <pthread.h>
#include <stdio.h>
#include "MyThread.h"//记得导入刚刚编译的那个.h文件
pthread_t pid;
//这个方法要参考.h文件的15行代码,这里的参数得注意,你写死就行,不用明白为什么
JNIEXPORT void JNICALL Java_MyThread_start0
(JNIEnv *env, jobject c1){
jclass mythreadclass;
jobject mythreadobj;
jmethodID run1;
jmethodID init;
jint ret=0;
mythreadclass=(*env)->FindClass(env,"MyThread");
if(mythreadclass==NULL){
printf("class null \n");
}
init=(*env)->GetMethodID(env,mythreadclass,"<init>","()V");
mythreadobj=(*env)->NewObject(env,mythreadclass,init);
if(mythreadobj==NULL){
printf("obj null \n");
}
run1=(*env)->GetMethodID(env,mythreadclass,"run","()V");
if(run1==NULL){
printf("run null \n");
}
ret=(*env)->CallIntMethod(env,mythreadobj,run1,NULL);
printf("I am main Thread\n");
}
5.上传NewThread.c,,用以下命令将其编译成为一个动态链接库,同时会生成一个so文件,这样在静态代码块中,它会被laod到内存。
gcc -fPIC -I /usr/local/jdk1.8.0_191/include -I /usr/local/jdk1.8.0_191/include/linux -shared -o libMyThreadNativeLibrary.so NewThread.c
6.指定so文件到path。
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/thread
7.万事具备,下面可以运行了。
java MyThread
效果如图
可以看到java中的run方法已经成功执行了。
总结:在刚刚的过程中,我们通过定义的start0方法,成功调用了C文件的Java_MyThread_start0,在C中,以JNI反向映射的方式,JNI相关函数找到MyThread类文件,用构造方法创建MyThread对象,并调用其run方法。成功模拟了线程中,start方法的执行过程。
当然,JVM的实现与此不同,具体实现细节,会在其他文章中讨论。