我尝试从Google Go程序中启动JVM,如下所示:
package main
import "fmt"
import "os"
import "log"
func main() {
var name string
name="\\jrex64\\bin\\java.exe"
var path="-Xbootclasspath:jrex64\\lib\\rt.jar;"+
"jrex64\\lib\\jfxrt.jar;"+
"jrex64\\lib\\resources.jar;"+
"jrex64\\lib\\ext\\sunjce_provider.jar;"+
"jrex64\\lib\\ext\\zipfs.jar;"+
"jrex64\\lib\\ext\\sunmscapi.jar;"+
"jrex64\\lib\\ext\\sunec.jar;"+
"jrex64\\lib\\ext\\dnsns.jar;"+
"jrex64\\lib\\ext\\access-bridge-64.jar;"+
"jrex64\\lib\\security\\local_policy.jar;"+
"jrex64\\lib\\jce.jar;"+
"jrex64\\lib\\jfr.jar;"+
"jrex64\\lib\\jsse.jar;"+
"jrex64\\lib\\charsets.jar;"+
"jrex64\\lib\\";
var args[] string=make([]string,4)
args[0]="-verbose"
args[1]=path;
args[2]="-cp Ganesha_lib\\*"
args[3]="-jar Ganesha.jar"
var attr* os.ProcAttr
proc,err:=os.StartProcess(name,args,attr)
proc.Wait();
if err!=nil {
fmt.Println("an error occurred.\n")
log.Fatal(err)
}
}
这是我的第一个Go程序。得到以下错误让我完全不知所措:
panic:运行时错误:无效的内存地址或nil指针取消引用 [信号0xc0000005代码= 0x0 addr = 0x0 pc = 0x4278b5] goroutine 1 [正在运行]: os.startProcess(0x4aacb4,0x14,0xf840001eb0,0x500000005,0x0,...) C:/Users/ADMINI~1/AppData/Local/Temp/2/bindist767862039/go/src/pkg/os/exec_posix.go:28 + 0x152 os.StartProcess(0x4aacb4,0x14,0xf840001eb0,0x500000005,0x0,...) C:/Users/ADMINI~1/AppData/Local/Temp/2/bindist767862039/go/src/pkg/os/doc.go:24 + 0x5c main.main() D:/MyGoProject/src/main.go:60 + 0x23c goroutine 2 [系统调用]: 由runtime.main创建 C:/Users/ADMINI~1/AppData/Local/Temp/2/bindist767862039/go/src/pkg/runtime/proc.c:221 处理以退出代码2完成
我如何解释此错误代码?什么地方出了错?我怎样才能获得JVM startet - 它位于Go可执行文件的子目录中。
答案 0 :(得分:17)
通常建议您不要直接使用os.StartProcess。相反,使用os / exec,它具有更简单的界面。以下是我将如何启动java子进程并等待它完成。
http://play.golang.org/p/APlp9KK9wx
package main
import (
"fmt"
"log"
"os/exec"
"strings"
)
func main() {
var java = "\\jrex64\\bin\\java.exe"
var path = []string{
"jrex64\\lib\\rt.jar",
"jrex64\\lib\\jfxrt.jar",
"jrex64\\lib\\resources.jar",
"jrex64\\lib\\ext\\sunjce_provider.jar",
"jrex64\\lib\\ext\\zipfs.jar",
"jrex64\\lib\\ext\\sunmscapi.jar",
"jrex64\\lib\\ext\\sunec.jar",
"jrex64\\lib\\ext\\dnsns.jar",
"jrex64\\lib\\ext\\access-bridge-64.jar",
"jrex64\\lib\\security\\local_policy.jar",
"jrex64\\lib\\jce.jar",
"jrex64\\lib\\jfr.jar",
"jrex64\\lib\\jsse.jar",
"jrex64\\lib\\charsets.jar",
"jrex64\\lib\\",
}
pathflag := "-Xbootclasspath:" + strings.Join(path, ";")
cmd := exec.Command(java, "-verbose", pathflag, "-cp Ganesha_lib\\*", "-jar Ganesha.jar")
err := cmd.Run()
if err != nil {
fmt.Println("an error occurred.\n")
log.Fatal(err)
}
}
如果你很好奇,你之所以感到恐慌的原因是attr是一个零指针。相反,您可以完成attr := new(os.ProcAttr)
。
答案 1 :(得分:5)
之前的答案实际上并没有描述如何使用os.StartProcess()。这是一个例子:
cmdToRun := "someCommand"
args := []string{"arg1"}
procAttr := new(os.ProcAttr)
procAttr.Files = []*os.File{os.Stdin, os.Stdout, os.Stderr}
if process, err := os.StartProcess(cmdToRun, args, procAttr); err != nil {
fmt.Printf("ERROR Unable to run %s: %s", cmdToRun, err.Error())
} else {
fmt.Printf("%s running as pid %d", cmdToRun, process.Pid)
}
请注意,有必要初始化ProcAttr.Files字段 - 如果不这样做,您可能会在os包中深入获得索引超出范围的错误。如果要在自己的代码中提供新进程的输入或进程输出,则可以使用管道作为文件。您可能还想指定Dir(起始目录)和Env(环境变量)字段。
答案 2 :(得分:2)
下面:
var attr* os.ProcAttr
proc, err := os.StartProcess(name, args, attr)
attr
变量为零,如果在os.StartProcess
中取消引用,则会导致您看到的错误。