前段时间看了一本书,说的是用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可执行文件;
--------------以上皆原创,给未来的自己留下一点学习的痕迹!--------