利用Java编写一个Java虚拟机

 更新时间:2023年07月30日 08:54:48   作者:Haidnor  
这篇文章主要为大家详细介绍了如何使用 Java17 编写的 Java 虚拟机,文中的示例代码讲解详细,具有一定的学习价值,感兴趣的可以了解下

github 项目链接

https://github.com/FranzHaidnor/haidnorJVM

haidnorJVM

使用 Java17 编写的 Java 虚拟机

意义

  • 纸上得来终觉浅,绝知此事要躬行。只学习 JVM 机制和理论,很多时候任然觉得缺乏那种大彻大悟之感
  • 使用简单的方式实现 JVM,用于学习理解 JVM 运行原理

主要技术选型

实现功能

  • 实现了 99% 的 JVM 字节码指令。参照 JVM 字节码规范实现 The Java Virtual Machine Instruction Set
  • 支持算数运算符 (+,-,*,^,%,++,--)
  • 支持关系运算符 (==,!=,>,<,>=,<=)
  • 支持位运算符 (&,|,^,~,<<,>>,>>>)
  • 支持赋值运算符 (=,+=,-=,*=,%=,<<=,>>=,&=,^=,|=)
  • 支持 instanceof 运算符
  • 支持循环结构代码 (while,do...while,for,foreach)
  • 支持条件结构代码 (if,if...else,if...else if)
  • 支出创建自定义类
  • 支持创建对象、访问对象
  • 支持抽象类
  • 支持多态继承、接口
  • 支持访问静态方法
  • 支持访问对象方法
  • 支持 JDK 中自带的 Java 类
  • 支持反射
  • 支持异常
  • 枚举 (开发中...)
  • switch 语法 (开发中...)
  • lambda 表达式 (开发中...)

局限性

不支持多线程

不支持多维数组

暂无双亲委派机制实现

无垃圾收集器实现。垃圾回收依靠宿主 JVM

快速体验

你需要准备什么

  • 集成开发环境 (IDE)。你可以选择包括 IntelliJ IDEA、Visual Studio Code 或 Eclipse 等等
  • JDK 17。并配置 JAVA_HOME
  • JDK 8。haidnorJVM 的主要目标是运行 Java8 本版的字节码文件。(但 haidnorJVM 没有强制要求字节码文件是 Java8 版本)
  • Maven

配置 haidnorJVM

配置日志输出级别

在 resources\simplelogger.properties 文件中修改日志输出级别,一般使用 debuginfo

  • 配置 info 级别将不会看到任何 haidnorJVM 内部运行信息
  • 配置 debug 级别下运行将会非常友好的输出 JVM 正在执行的栈信息
public class Demo5 {
    public static void main(String[] args) {
        String str = method1("hello world");
        method1(str);
    }
    public static String method1(String s) {
        return method2(s);
    }
    public static String method2(String s) {
        return method3(s);
    }
    public static String method3(String s) {
        System.out.println(s);
        return "你好 世界";
    }
}

每一个  结构图形,都表示一个 JVM 线程栈中的栈帧

配置 rt.jar 路径

修改 haidnorJVM.properties 文件中的内容。配置 rt.jar 的绝对路径,例如rt.jar=D:/Program Files/Java/jdk1.8.0_361/jre/lib/rt.jar

运行单元测试用例

在 IDE 中打开项目中 test 目录下的 haidnor.jvm.test.TestJVM.java 文件。 这是 haidnorJVM 的主要测试类, 里面的测试方法可以解析加载运行 .class 字节码文件。

public class TestJVM {
   /**
    *  haidnorJVM 会加载 HelloWorld.java 在 target 目录下的编译后的字节码文件,然后运行其中的 `main(String[] args)` 方法。
    *  你可以使用打断点的方式看到 haidnorJVM 是如何解释运行 Java 字节码的。
    *  值得注意的是,这种方式编译运行的字节码文件是基于 java17 版本的。
    */
   @Test
   public void test() {
      runMainClass(HelloWorld.class);
   }
}

运行 .class 文件

  • 使用 maven 命令将 haidnorJVM 编译打包,得到 haidnorJVM.jar 文件
  • 编写一个简单的程序,例如以下代码
public class HelloWorld {
   public static void main(String[] args) {
     System.out.println("HelloWorld");
   }
}
  • 编译代码,得到 HelloWorld.class 文件 (推荐使用 JDK8 进行编译)
  • 使用 haidnorJVM 运行程序。执行命令 java -jar haidnorJVM.jar -class R:\HelloWorld.class。注意! 需要 class 文件的绝对路径

运行 .jar 文件

  • 使用 maven 命令将 haidnorJVM 编译打包,得到 haidnorJVM.jar 文件
  • 编写一个 java 项目编译打包成 .jar 文件,例如 demo.jar。要求 .jar 文件中的 META-INF/MANIFEST.MF 文件内有 Main-Class 属性 (含有 public static void main(String[] args) 方法的主类信息)
  • 使用 haidnorJVM 运行程序。执行命令 java -jar haidnorJVM.jar -class R:\demo.jar。注意! 需要 jar 文件的绝对路径

存在的问题

由于 haidnorJVM 目前运行 JDK 自带的类是使用反射解决的,因此 haidnorJVM 使用 JDK17 运行部分 JDK 自带的类时会存在一些问题,例如运行以下代码将会抛出异常

public class Demo {
    public static void main(String[] args) {
        List<Integer> list = List.of(1, 2, 3, 4, 5);
        list.add(6);
    }
}
java.lang.reflect.InaccessibleObjectException: Unable to make public boolean java.util.ImmutableCollections$AbstractImmutableCollection.add(java.lang.Object) accessible: module java.base does not "opens java.util" to unnamed module @18769467
	at java.base/java.lang.reflect.AccessibleObject.checkCanSetAccessible(AccessibleObject.java:354)
	at java.base/java.lang.reflect.AccessibleObject.checkCanSetAccessible(AccessibleObject.java:297)
	at java.base/java.lang.reflect.Method.checkCanSetAccessible(Method.java:199)
	at java.base/java.lang.reflect.Method.setAccessible(Method.java:193)

它表示尝试通过反射来访问一个方法或字段,但该方法或字段的可访问性限制导致无法访问。

这个限制通常是由于 Java 模块系统引起的。模块系统允许将代码划分为独立的模块,并控制模块之间的访问权限。以上异常的原因是 module java.base does not "opens java.util" to unnamed module,也就是说 java.base 模块没有向未命名模块开放 java.util 包

解决方法:启动 haidnorJVM 时添加 JVM 参数 --add-opens java.base/java.util=ALL-UNNAMED 绕过访问性限制

以上就是利用Java编写一个Java虚拟机的详细内容,更多关于Java虚拟机的资料请关注脚本之家其它相关文章!

相关文章

  • Java实现文件批量重命名具体实例

    Java实现文件批量重命名具体实例

    这篇文章主要介绍了Java实现文件批量重命名具体实例,需要的朋友可以参考下
    2014-02-02
  • spring 自动装配和aop的使用

    spring 自动装配和aop的使用

    这篇文章主要介绍了spring 自动装配和aop的使用,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-07-07
  • 使用springmvc临时不使用视图解析器的自动添加前后缀

    使用springmvc临时不使用视图解析器的自动添加前后缀

    这篇文章主要介绍了使用springmvc临时不使用视图解析器的自动添加前后缀,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-09-09
  • java多种幻灯片切换特效(经典)

    java多种幻灯片切换特效(经典)

    功能说明: 代码实现了多种幻灯片变换特效. 如:淡入淡出、缓慢覆盖、旋转覆盖等10多种变换效果。
    2013-03-03
  • Java入门之继承多态使用及说明

    Java入门之继承多态使用及说明

    文章概述Java继承与多态的核心概念,包括继承的共性抽取、子类父类关系、super关键字的使用、构造方法与代码块执行顺序,以及向上转型、动态绑定和重写机制,最后预告多态内容并附练习题
    2025-09-09
  • Java使用itext5实现PDF表格文档导出

    Java使用itext5实现PDF表格文档导出

    这篇文章主要介绍了Java使用itext5实现PDF表格文档导出,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-01-01
  • swing jtextArea滚动条和文字缩放效果

    swing jtextArea滚动条和文字缩放效果

    这篇文章主要为大家详细介绍了swing jtextArea滚动条和文字缩放效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-12-12
  • mybatis插入后返回主键id的3种方式图解

    mybatis插入后返回主键id的3种方式图解

    这篇文章主要给大家介绍了关于mybatis插入后返回主键id的3种方式,很多时候,在向数据库插入数据时,需要保留插入数据的,以便进行后续的操作或者将存入其他表作为外键,需要的朋友可以参考下
    2023-08-08
  • Java调用接口如何获取json数据解析后保存到数据库

    Java调用接口如何获取json数据解析后保存到数据库

    这篇文章主要介绍了Java调用接口如何获取json数据解析后保存到数据库问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-03-03
  • SpringBoot实现token登录的示例代码

    SpringBoot实现token登录的示例代码

    在进行登录验证时,我们需要session或cookie会话进行验证,当我们脱离浏览器用app等向服务端发请求就没有session和cookie机制,这时我们就需要使用token令牌进行登录验证,本文就详细的介绍一下,感兴趣的可以了解一下
    2022-03-03

最新评论