查看手写JAVA虚拟机系列可以进我的博客园主页查看。

  我们知道,我们编译.java并运行.class文件时,需要一些java命令,如最简单的helloworld程序。

  

  这里的程序最好不要加包名,因为加了包名的话编译和运行需要有所改动。

  看这里的命令。javac为编译命令,我们知道java的特点是一次编译,到处运行。这里的编译指的就是javac,对于java程序即.java文件,先要用javac编译成字节码。然后将字节码(.class文件)放到java虚拟机中运行,即上图中的java HelloWorld,java虚拟机把字节码翻译成对应机器上的机器指令,再由机器来执行具体的机器指令。也就是说java程序员是直接与java虚拟机交互,简介与机器交互。所以虚拟机完成的是java命令,也就是我们要完成的是java这个指令的功能

  那么我们把第一个目标定为,实现简单的命令行。即我们通过命令行可以输入一些内容,虚拟机读取之后可以给一定的反馈。

  GO语言中有两个和命令行相关的包,分别是os和flag(java中以类库即jar文件导入,go中直接以包的形式导入)。

  首先在GOPATH目录下的src里面新建一个jvmgo文件夹作为我们的工作空间目录,jvmgo里面再新建一个ch01为我们的第一个目标源码文件夹,添加cmd.go文件。

  

  在cmd.go里面输入如下代码(由于博客园的添加代码方式不支持go语言着色,所以采用C语言着色,高亮可能不太正确)

package main

import "flag"
import "fmt"
import "os" //定义Cmd结构体
type Cmd struct{
helpFlag bool
versionFlag bool
cpOption string
class string
args []string
} //解析命令行参数
func parseCmd() *Cmd {
cmd:=&Cmd{} //将printUsage函数传给flag.Usage
flag.Usage=printUsage
//设置各种解析的选项
flag.BoolVar(&cmd.helpFlag, "help", false, "print help message")
flag.BoolVar(&cmd.helpFlag, "?", false, "print help message")
flag.BoolVar(&cmd.versionFlag, "version", false, "print version and exit")
flag.StringVar(&cmd.cpOption, "classpath", "", "classpath")
flag.StringVar(&cmd.cpOption, "cp", "", "classpath")
//所有选项设置完成后调用flag.Parse解析所有选项,如果Parse失败,则调用flag.Usage打印帮助信息
flag.Parse() //调用flag.Args函数捕获未被解析的参数,第一个参数为主类名,后面的为传递给主类的参数
args:=flag.Args()
if len(args)>{
cmd.class=args[]
cmd.args=args[:]
} return cmd
} func printUsage() {
fmt.Printf("Usage:%s[-options] class [args...]\n",os.Args[])
}

  第一行为包名,main包,接着引入了三个包os,flag,fmt。os和flag都是处理命令行所需的包,fmt类似于C语言的printf和scanf等格式化IO。再往下定义了一个结构体Cmd,用来这个数据结构来格式化存储输入的命令行信息。helpFlag参数为命令行是否请求help,versionFlag参数为命令行是否请求version,cpOption为命令行传入的classpath即目标.class文件所在文件夹,class为命令行传入的.class文件名(不包括.class),args为命令行传入的其他参数。

  紧接着是一个parseCmd函数(go语言有函数和方法之分,方法调用需要receiver,函数调用则不需要 ),返回值为*Cmd,用来解析cmd传过来的参数。该函数里面先声明一个cmd并给这个cmd赋值一个新建的Cmd对象。go语言中的“:=”为声明并赋值,而"="为赋值。先把printUsage的函数赋值给flag.Usage,然后调用flag设置需要解析的选项,全部解析完毕,调用Parse函数解析所有选项。解析成功则结束,解析失败则调用printUsage打印到控制台。

  flag.Args可以捕获其他没有被解析的参数。上面解析成功之后,第一个参数就是主类名,剩下的就是传给主类的参数。

  工具类编写完成,下一个是主函数。先上主函数代码:

package main

import "fmt"

func main() {
//调用parseCmd解析命令行参数
cmd:=parseCmd() if cmd.versionFlag{
//输入了-version选项
fmt.Println("version 0.0.1")
}else if cmd.helpFlag||cmd.class==""{
//输入了-help选项
printUsage()
}else{
//启动jvm
stratJVM(cmd)
}
} func stratJVM(cmd *Cmd){
fmt.Printf("classpath:%s class:%s args:%v\n",
cmd.cpOption,cmd.class,cmd.args)
}

  跟java类似,在go里面main是一个特殊的包,go程序的入口就是main函数,但是不接受任何参数,也不能有返回值。main函数先调用parseCmd解析命令行参数,如果是-version则返回版本号,如果是-help则返回帮助信息,如果是其他则启动jvm,这里用一些输出信息“假装”启动了jvm,真正的jvm代码后面会加上。

  至此,对命令行的解析工作全部完成。先展示一下整个工作目录的结构,不然后面编译运行的时候会出错。

  

  我们的工作目录是D盘下的JVM里的goWorkSpace,再下面src,jvmgo,ch01,ch01里面包含的是我们的go文件。

  来测试一下,打开一命令行,输入go install jvmgo\ch01。这个命令是使用go.exe来install文件,这个文件存在于GOPATH下面的文件夹(jvmgo\ch01中),结果如图:

  

  然后在工作空间(GOPATH)的bin文件夹中就多出了一个ch01.exe。

  

  在此处打开命令行。可以进行一些操作:

  

  到这里,我们的命令行工具就完成了,虽然还没有涉及真正的虚拟机设计,但这也是虚拟机运行的重要一步,后面会逐渐介绍虚拟机的设计。

手写JAVA虚拟机(二)——实现java命令行的更多相关文章

  1. 深入理解Java虚拟机二之Java内存区域与内存溢出异常

    运行时数据区域 1.线程独有的内存区域 PROGRAM COUNTER REGISTER 程序计数器 程序计数器空间较小,是当前线程执行字节码的行号指示器,字节码解释器工作时就是通过改变这个计数器的值 ...

  2. 深入理解Java虚拟机(二)、Java对象的创建,内存布局和访问定位

    对象的创建: Object obj = new Object(); 常量池中是否有Ljava.lang.Object

  3. 【java虚拟机序列】java中的垃圾回收与内存分配策略

    在[java虚拟机系列]java虚拟机系列之JVM总述中我们已经详细讲解过java中的内存模型,了解了关于JVM中内存管理的基本知识,接下来本博客将带领大家了解java中的垃圾回收与内存分配策略. 垃 ...

  4. 转 Java虚拟机5:Java垃圾回收(GC)机制详解

    转 Java虚拟机5:Java垃圾回收(GC)机制详解 Java虚拟机5:Java垃圾回收(GC)机制详解 哪些内存需要回收? 哪些内存需要回收是垃圾回收机制第一个要考虑的问题,所谓“要回收的垃圾”无 ...

  5. Java虚拟机如何运行Java字节码

    一.Java的class文件的内容 1.首先编写一个简单的代码 public class StringDemo { public static void main(String[] args) { S ...

  6. 【Java虚拟机6】Java内存模型(Java篇)

    什么是Java内存模型 <Java虚拟机规范>中曾试图定义一种"Java内存模型"(Java Memory Model,JMM)来屏蔽各种硬件和操作系统的内存访问差异, ...

  7. java虚拟机学习-触摸java常量池(13-1)

    java虚拟机学习-深入理解JVM(1) java虚拟机学习-慢慢琢磨JVM(2) java虚拟机学习-慢慢琢磨JVM(2-1)ClassLoader的工作机制 java虚拟机学习-JVM内存管理:深 ...

  8. Java虚拟机2:Java内存区域

    1.几个计算机的概念 为以后写文章考虑,也为巩固自己的知识和一些基本概念,这里要理清楚几个计算机中的概念. 1.计算机存储单位 从小到大依次为位Bit.字节Byte.千字节KB.兆M.千兆GB.TB, ...

  9. opencv 手写选择题阅卷 (二)字符识别

    opencv 手写选择题阅卷 (二)字符识别 选择题基本上只需要识别ABCD和空五个内容,理论上应该识别率比较高的,识别代码参考了网上搜索的代码,因为参考的网址比较多,现在也弄不清是参考何处的代码了, ...

  10. <<深入Java虚拟机>>-第二章-Java内存区域-学习笔记

    Java运行时内存区域 Java虚拟机在运行Java程序的时候会将它所管理的内存区域划分为多个不同的区域.每个区域都有自己的用途,创建以及销毁的时间.有的随着虚拟机的启动而存在,有的则是依赖用户线程来 ...

随机推荐

  1. 利用封装、继承对Java代码进行优化

    注:本文实例分别可以在oldcastle(未优化的代码)和newcastle(优化后的代码)中查看,网址见文末 城堡游戏: 城堡中有多个房间,用户通过输入north, south, east, wes ...

  2. javascript递归函数

    递归函数:是指函数直接或间接调用函数本身,则称该函数为递归函数. 这句话理解起来并不难,从概念上出发,给出以下的例子: function foo(){ console.log("函数 foo ...

  3. 《网络》:设置三个密码:通过console口连接设备,进入特权模式,登录Telnet

    软件:Cisco Packet Tracer Instructor 软件下载链接在上一篇文章中. 内容:通过设置三个密码,熟悉采用Telnet方式配置交换机的方法. 细节说明:计算机的IP地址和交换机 ...

  4. Python内置函数(11)——complex

    英文文档: class complex([real[, imag]]) Return a complex number with the value real + imag*1j or convert ...

  5. jhipster生成项目无法使用restful请求,报access_denied 403错误

    写在前边: 我们的微服务是注册中心.uaa.gateway为基础,添加微服务应用,昨天下午在测试jhipster的增删改查,因为jhipster生成的代码都是restful的,好不容易找到网关配置的映 ...

  6. SpringBoot的重要特性

    一.Web特性 Spring Boot 提供了spring-boot-starter-web来为Web开发予以支持,spring-boot-starter-web为我们提供了嵌入的Tomcat以及Sp ...

  7. gradle入门(1-7)eclipse和gradle集成插件的安装和使用

    一.安装gradle插件:buildship 1.安装插件 gradle默认的本地缓存库在c盘user目录下的.gradle文件夹下,安装好gradle后,可以添加环境变量GRADLE_USER_HO ...

  8. zabbix配置微信报警

    首先我们先目睹下微信报警的效果 接下来我们正式开始操作. 一:注册企业微信. 打开企业微信注册:http://work.weixin.qq.com 根据以上提示填入相应的内容,然后注册即可. 二:登录 ...

  9. wrapper x64 版本发布到centos

    背景: 项目需要在spark任务提交服务器节点上自动提交任务到spark集群上.因此创建了一个固定时间监控任务项目,使用timer定时监控oracle数据库中是否有spark提交任务,如果有spark ...

  10. CSS基础:内联元素

    简介 内联元素由于涉及到文本字体,读写方向,汉字和字母差异等诸多方面的影响,因此其盒模型比块级元素更加复杂,对于内联非替换元素,比如一行文本,主要由以下几种框构成: "em 框", ...