前段时间看了一本书,说的是用go语言实现java虚拟机,很有意思,于是就花了一段时间学习了一下go语言,虽然对go的底层理解不是很深,但是写代码还是可以的,就当做个读书笔记吧!

  链接在这里,另外还有一本《go程序设计语言》,有需要的直接一起拿走,链接:https://pan.baidu.com/s/152ZX7cLf5IcOzUk1C_Q8JQ  提取码:3ktm

  首先我们知道java能够跨平台的原因就是class字节码文件在不同的jvm中都可以运行,每个计算机都可以安装符合自己操作系统的jvm,然后class文件就可以通用了,而class文件被包装的形式有很多种,最常见的应该就是jar包,有兴趣的可以看看打开自己的jdk中随便的个jar包,里面其实就是一些class字节码文件;

  我们就简单一点,我们自己把一个java源码文件手动打包成一个jar包,下面的注释已经很清楚了

//javac HelloWorld.java将一个java源码文件编译成class文件
//java手动将class文件打成jar包命令:jar cvf hello.jar HelloWorld.class
//执行java -classpath hello.jar HelloWorld,表示执行hello中的jar包中HelloWorld中的main方法
public class HelloWorld {
public static void main(String[] args) {
System.out.println("Hello, world!");
}
}

 打成jar包之后,我们现在手中的文件有两个,一个是HelloWorld.class字节码文件,一个是hello.jar文件(jar包其实就是一个压缩包),我们分别用两条命令执行这两个文件,不知道大家有没有看出来什么?

  如果是字节码文件的话,就是直接用java xxx执行就行了;如果是jar包的形式,那么我们必须要指定jar包的全路径,以及jar包里面main方法所在的字节码文件,这样jvm才能找到入口,就会在jar包中找到该文件,然后运行;我们可以实现一下命令行的处理方式:

  不知道大家还记不记得查看jdk版本的命令,就是java -version;在go语言中,对这种命令行参数的处理有两种,一种是os.Args这个切片来处理,另外一种通过一个flag包,选用后者,flag包封装了很多操作;

  main.go代码如下:

package main

import "fmt"

func main() {
//同一个包下,私有方法也可以调用,如果是不同包,那就需要把parseCmd函数首字母大写
cmd := parseCmd()
//例如java -version,那么此时在parseCmd函数中解析version的值位true,然后赋值给versionFlag,就打印版本号
if cmd.versionFlag {
fmt.Println("version 1.0.0")
//例如输入的是java -help或者是java -classpath hello.jar ,没有加后面的类名,那么就调用printUsage函数
//打印提示信息
} else if cmd.helpFlag || cmd.class == "" {
printUsage()
//当参数都输入正确,那么就调用startJVM函数启动jvm,这里暂时就打印一句话,将输入的命令各部分参数都打印出来
} else {
startJVM(cmd)
}
} func startJVM(cmd *Cmd) {
fmt.Printf("classpath:%s class:%s args:%v\n", cmd.cpOption, cmd.class, cmd.args)
}

  cmd.go文件内容:

package main

import (
"flag"
"fmt"
"os"
) //简单的定义一个结构体,这里保存命令行中输入的参数
type Cmd struct {
helpFlag bool
versionFlag bool
cpOption string
class string
args []string
} func parseCmd() *Cmd {
cmd := &Cmd{}
//这里的意思就是如果解析失败的话,就调用printUsage函数
flag.Usage = printUsage
//解析命令行中输入,例如java -help,那么这里help就是true,然后把true赋值给cmd结构体中helpFlag保存起来
//最后一个是默认信息
flag.BoolVar(&cmd.helpFlag, "help", false, "print help message")
//例如java -version,那么version就是true,赋值给cmd中的versionFlag
flag.BoolVar(&cmd.versionFlag, "version", false, "print version and exit ")
//例如java -classpath hello.jar HelloWorld,这里classpath就是hello.jar,然后赋值给cpOption保存起来
flag.StringVar(&cmd.cpOption, "classpath", "", "classpath") //上面是定义解析规则,调用Parse函数才是真正开始解析
flag.Parse() //解析成功的话,那就继续获取后面的参数,注意这里的args是一个切片类型的
//例如java -classpath hello.jar HelloWorld arg1,arg2,这里的args[0]表示HelloWorld,args[1:]表示arg1和arg2,
//就是传给main方法形参字符串数组的参数
args := flag.Args()
if len(args) > 0 {
cmd.class = args[0]
cmd.args = args[1:]
}
return cmd
} //这里传进去的参数,解析错误的话就显示第一个参数的提示信息
func printUsage() {
fmt.Printf("Usage:%s [-options] class [args]\n", os.Args[0])
}

  

  其实很容易,然后我们可以测试一下,目录结构如下,其中main.go和cmd.go都在ch01中,我们进入到src目录下,打开终端,输入go install jvmgo\ch01,就会在workspace/bin下生成一个ch01.exe可执行文件;

go实现java虚拟机01的更多相关文章

  1. 带着新人看java虚拟机01

    1.前言(基于JDK1.7) 最近想把一些java基础的东西整理一下,但是又不知道从哪里开始!想了好久,还是从最基本的jvm开始吧!这一节就简单过一遍基础知识,后面慢慢深入... 水平有限,我自己也是 ...

  2. 深入理解java虚拟机-01 走进java

    第一章是对java的产生,历史的整体介绍 java的使用很广泛,安装jdk的时候会看到一句广告语runs in 10 billions machines.使用java的设备多达几十亿台 1.概述 优点 ...

  3. 深入理解Java虚拟机 #01# 自己编译JDK

    x 首先用书上的脚本尝试,失败. 之后根据源文件的 README 编译,抛出: root@linux:/opt/openjdk# sh ./get_source.sh ERROR: Need init ...

  4. Java虚拟机JVM学习01 流程概述

    Java虚拟机JVM学习01 流程概述 Java虚拟机与程序的生命周期 一个运行时的Java虚拟机(JVM)负责运行一个Java程序. 当启动一个Java程序时,一个虚拟机实例诞生:当程序关闭退出,这 ...

  5. 《深入理解Java虚拟机》笔记01 -- 运行时数据区

    运行时数据区示意图 1. 程序计数器 占用一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器.主要用来记录线程执行到哪条语句了,分支.循环.跳转.异常处理.线程恢复等功能都需要依赖这个 ...

  6. Java虚拟机一看就懂01

    Jvm内存结构 --- 线程隔离区域说明: 1.1.程序计数器 线程私有 是一块内存空间 唯一的一个在Java虚拟机规范中没有规定任何OOM情况的区域(不会OOM?) 1.2.Java虚拟机栈 线程私 ...

  7. java虚拟机开篇01

    一直以来对java 基础设施都啥都不知道啊,感觉有时候挺费力,挺吃劲的. 一下是一些很好的参考资料: http://blog.csdn.net/bingduanlbd/article/details/ ...

  8. 实战Java虚拟机之三“G1的新生代GC”

    今天开始实战Java虚拟机之三:“G1的新生代GC”. 总计有5个系列 实战Java虚拟机之一“堆溢出处理” 实战Java虚拟机之二“虚拟机的工作模式” 实战Java虚拟机之三“G1的新生代GC” 实 ...

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

    哪些内存需要回收? 哪些内存需要回收是垃圾回收机制第一个要考虑的问题,所谓“要回收的垃圾”无非就是那些不可能再被任何途径使用的对象.那么如何找到这些对象? 1.引用计数法 这个算法的实现是,给对象中添 ...

随机推荐

  1. [题解][Codeforces]Good Bye 2019 简要题解

    构造题好评,虽然这把崩了 原题解 A 题意 二人游戏,一个人有 \(k_1\) 张牌,另一个人 \(k_2\) 张,满足 \(2\le k_1+k_2=n\le 100\),每张牌上有一个数,保证所有 ...

  2. .net core webapi搭建(2)跨域

    Core WebAPI中的跨域处理 在使用WebAPI项目的时候基本上都会用到跨域处理 Core WebAPI的项目中自带了跨域Cors的处理,不需要单独添加程序包 如图所示 修改 Configure ...

  3. vuex 基本语法

    VUEX 的核心概念 1 .State (常用):2.Getters :3.Mutations(常用):4.Actions :5.Modules: 1.State是唯一的数据源,单一的状态树 cons ...

  4. AVLtree(C++实现)有统一的旋转操作

    在学习完AVLtree之后,我发现,左旋,右旋均可以采用统一的旋转方式来实现,所以把代码贴在下面 代码是完整的AVLTree实现 C++标准为C++11 在ubuntu 18.04下通过编译和调试 / ...

  5. nmap详解之基础概述

    概述 nmap是一个网络探测和安全扫描程序,系统管理者和个人可以使用这个软件扫描大型的网络,获取那台主机正在运行以及提供什么服务等信息.nmap支持很多扫描技术,例如:UDP.TCP connect( ...

  6. C# 解析JSON遇到以错误提示:应为状态“Element”。。遇到名称为“”、命名空间为“”的“Text”。

    话不多说:仔细看代码: ①json格式错误导致报错 {"TeachIQ":"  语言  0小时0分钟  未完成","Temperature" ...

  7. abp vnext2.0核心组件之.Net Core默认DI组件切换到AutoFac源码解析

    老版Abp对Castle的严重依赖在vnext中已经得到了解决,vnext中DI容器可以任意更换,为了实现这个功能,底层架构相较于老版abp,可以说是进行了高度重构.当然这得益于.Net Core的D ...

  8. java线程池及创建多少线程合适

    java线程池 1.以下是ThreadPoolExecutor参数完备构造方法: public ThreadPoolExecutor(int corePoolSize,int maximumPoolS ...

  9. 看片微信号+薇myy9199买片微信号+myy9199绝对靠谱号

    最新看片卖片微信号+myy9199,2020最新有效靠谱号,诚信有效,死链包换,2019年11月,我决定学习计算机编程,以java语言为主.我就读于传统工科专业,没怎么接触过计算机相关概念与课程,我知 ...

  10. 关于c/c++语言的EOF(C++实现闰年判断)

    EOF 是 End Of File 的缩写,在 C 语言标准库中的定义如下: #define EOF (-1) 迄今为止,关于 EOF 作用的观点各异.大多数程序员认为“文件中有一个 EOF 字符,用 ...