如何使用Java以编程方式安装所有Java JVM(非默认值)?

时间:2011-06-27 13:48:27

标签: java jvm

有谁知道如何以编程方式安装所有JVM(不是默认的)使用Java

例如,用户的计算机上安装了2个JVM:

JDK 5
JDK 6

我需要知道安装的所有版本,以便切换它正在使用的版本(默认情况下),然后以编程方式调用javac以使用特定的JDK版本编译一些源代码。

我一直在网上寻找一些信息,我发现:

但我找不到我要找的东西。

10 个答案:

答案 0 :(得分:13)

我最近处理的情况非常相似。以下代码几乎完全符合您的需求。它搜索java JRE和JDK,而不仅仅是JDK,但应该很容易根据您的需要进行编辑。注意:仅限Windows

/**
 * Java Finder by petrucio@stackoverflow(828681) is licensed under a Creative Commons Attribution 3.0 Unported License.
 * Needs WinRegistry.java. Get it at: https://stackoverflow.com/questions/62289/read-write-to-windows-registry-using-java
 *
 * JavaFinder - Windows-specific classes to search for all installed versions of java on this system
 * Author: petrucio@stackoverflow (828681)
 *****************************************************************************/

import java.util.*;
import java.io.*;

/**
 * Helper class to fetch the stdout and stderr outputs from started Runtime execs
 * Modified from http://www.javaworld.com/javaworld/jw-12-2000/jw-1229-traps.html?page=4
 *****************************************************************************/
class RuntimeStreamer extends Thread {
    InputStream is;
    String lines;

    RuntimeStreamer(InputStream is) {
        this.is = is;
        this.lines = "";
    }
    public String contents() {
        return this.lines;
    }

    public void run() {
        try {
            InputStreamReader isr = new InputStreamReader(is);
            BufferedReader    br  = new BufferedReader(isr);
            String line = null;
            while ( (line = br.readLine()) != null) {
                this.lines += line + "\n";
            }
        } catch (IOException ioe) {
            ioe.printStackTrace();  
        }
    }

    /**
     * Execute a command and wait for it to finish
     * @return The resulting stdout and stderr outputs concatenated
     ****************************************************************************/
    public static String execute(String[] cmdArray) {
        try {
            Runtime runtime = Runtime.getRuntime();
            Process proc = runtime.exec(cmdArray);
            RuntimeStreamer outputStreamer = new RuntimeStreamer(proc.getInputStream());
            RuntimeStreamer errorStreamer  = new RuntimeStreamer(proc.getErrorStream());
            outputStreamer.start();
            errorStreamer.start();
            proc.waitFor();
            return outputStreamer.contents() + errorStreamer.contents();
        } catch (Throwable t) {
            t.printStackTrace();
        }
        return null;
    }
    public static String execute(String cmd) {
        String[] cmdArray = { cmd };
        return RuntimeStreamer.execute(cmdArray);
    }
}

/**
 * Helper struct to hold information about one installed java version
 ****************************************************************************/
class JavaInfo {
    public String  path;        //! Full path to java.exe executable file
    public String  version;     //! Version string. "Unkown" if the java process returned non-standard version string
    public boolean is64bits;    //! true for 64-bit javas, false for 32

    /**
     * Calls 'javaPath -version' and parses the results
     * @param javaPath: path to a java.exe executable
     ****************************************************************************/
    public JavaInfo(String javaPath) {
        String versionInfo = RuntimeStreamer.execute( new String[] { javaPath, "-version" } );
        String[] tokens = versionInfo.split("\"");

        if (tokens.length < 2) this.version = "Unkown";
        else this.version = tokens[1];
        this.is64bits = versionInfo.toUpperCase().contains("64-BIT");
        this.path     = javaPath;
    }

    /**
     * @return Human-readable contents of this JavaInfo instance
     ****************************************************************************/
    public String toString() {
        return this.path + ":\n  Version: " + this.version + "\n  Bitness: " + (this.is64bits ? "64-bits" : "32-bits");
    }
}

/**
 * Windows-specific java versions finder
 *****************************************************************************/
public class JavaFinder {

    /**
     * @return: A list of javaExec paths found under this registry key (rooted at HKEY_LOCAL_MACHINE)
     * @param wow64  0 for standard registry access (32-bits for 32-bit app, 64-bits for 64-bits app)
     *               or WinRegistry.KEY_WOW64_32KEY to force access to 32-bit registry view,
     *               or WinRegistry.KEY_WOW64_64KEY to force access to 64-bit registry view
     * @param previous: Insert all entries from this list at the beggining of the results
     *************************************************************************/
    private static List<String> searchRegistry(String key, int wow64, List<String> previous) {
        List<String> result = previous;
        try {
            List<String> entries = WinRegistry.readStringSubKeys(WinRegistry.HKEY_LOCAL_MACHINE, key, wow64);
            for (int i = 0; entries != null && i < entries.size(); i++) {
                String val = WinRegistry.readString(WinRegistry.HKEY_LOCAL_MACHINE, key + "\\" + entries.get(i), "JavaHome", wow64);
                if (!result.contains(val + "\\bin\\java.exe")) {
                    result.add(val + "\\bin\\java.exe");
                }
            }
        } catch (Throwable t) {
            t.printStackTrace();
        }
        return result;
    }

    /**
     * @return: A list of JavaInfo with informations about all javas installed on this machine
     * Searches and returns results in this order:
     *   HKEY_LOCAL_MACHINE\SOFTWARE\JavaSoft\Java Runtime Environment (32-bits view)
     *   HKEY_LOCAL_MACHINE\SOFTWARE\JavaSoft\Java Runtime Environment (64-bits view)
     *   HKEY_LOCAL_MACHINE\SOFTWARE\JavaSoft\Java Development Kit     (32-bits view)
     *   HKEY_LOCAL_MACHINE\SOFTWARE\JavaSoft\Java Development Kit     (64-bits view)
     *   WINDIR\system32
     *   WINDIR\SysWOW64
     ****************************************************************************/
    public static List<JavaInfo> findJavas() {
        List<String> javaExecs = new ArrayList<String>();

        javaExecs = JavaFinder.searchRegistry("SOFTWARE\\JavaSoft\\Java Runtime Environment", WinRegistry.KEY_WOW64_32KEY, javaExecs);
        javaExecs = JavaFinder.searchRegistry("SOFTWARE\\JavaSoft\\Java Runtime Environment", WinRegistry.KEY_WOW64_64KEY, javaExecs);
        javaExecs = JavaFinder.searchRegistry("SOFTWARE\\JavaSoft\\Java Development Kit",     WinRegistry.KEY_WOW64_32KEY, javaExecs);
        javaExecs = JavaFinder.searchRegistry("SOFTWARE\\JavaSoft\\Java Development Kit",     WinRegistry.KEY_WOW64_64KEY, javaExecs);

        javaExecs.add(System.getenv("WINDIR") + "\\system32\\java.exe");
        javaExecs.add(System.getenv("WINDIR") + "\\SysWOW64\\java.exe");

        List<JavaInfo> result = new ArrayList<JavaInfo>();
        for (String javaPath: javaExecs) {
            if (!(new File(javaPath).exists())) continue;
            result.add(new JavaInfo(javaPath));
        }
        return result;
    }

    /**
     * @return: The path to a java.exe that has the same bitness as the OS
     * (or null if no matching java is found)
     ****************************************************************************/
    public static String getOSBitnessJava() {
        String arch      = System.getenv("PROCESSOR_ARCHITECTURE");
        String wow64Arch = System.getenv("PROCESSOR_ARCHITEW6432");
        boolean isOS64 = arch.endsWith("64") || (wow64Arch != null && wow64Arch.endsWith("64"));

        List<JavaInfo> javas = JavaFinder.findJavas();
        for (int i = 0; i < javas.size(); i++) {
            if (javas.get(i).is64bits == isOS64) return javas.get(i).path;
        }
        return null;
    }

    /**
     * Standalone testing - lists all Javas in the system
     ****************************************************************************/
    public static void main(String [] args) {
        List<JavaInfo> javas = JavaFinder.findJavas();
        for (int i = 0; i < javas.size(); i++) {
            System.out.println("\n" + javas.get(i));
        }
    }
}

您还需要更新的WinRegistry.java来读取Windows注册表的32位和64位部分的值:https://stackoverflow.com/a/11854901/828681

我通常不是java程序员,所以我的代码可能不遵循java约定。告诉我。

以下是从我的Win 7 64位计算机运行的示例:

>java JavaFinder

C:\Program Files (x86)\Java\jre6\bin\java.exe:
  Version: 1.6.0_31
  Bitness: 32-bits

C:\Program Files\Java\jre6\bin\java.exe:
  Version: 1.6.0_31
  Bitness: 64-bits

D:\Dev\Java\jdk1.6.0_31\bin\java.exe:
  Version: 1.6.0_31
  Bitness: 64-bits

C:\Windows\system32\java.exe:
  Version: 1.6.0_31
  Bitness: 64-bits

C:\Windows\SysWOW64\java.exe:
  Version: 1.6.0_31
  Bitness: 32-bits

答案 1 :(得分:3)

我快速检查了windows注册表,这个键似乎提供了系统上安装的各种Java版本

HKEY_LOCAL_MACHINE \ SOFTWARE \ JavaSoft \ Java运行时环境

在进一步的谷歌搜索中,确认这是正确的位置。 阅读这些文章了解详情...

Quickly retrieve available Java JVM on a workstation (Windows)

Java 2 Runtime Environment for Microsoft Windows

答案 2 :(得分:3)

在Windows平台上,您可以shell出来查询是否存在已安装的JRE:

java -version:1.6 -version

如果安装了“java version”1.6.0_xx“”,该命令将返回。如果您没有安装它,它会说,“无法找到JRE会议规范1.6”

这似乎不适用于Linux,可能是因为Linux没有标准的安装Java的方法。

答案 3 :(得分:2)

基本上没有办法枚举All JVM,即使不是Java代码也是如此。考虑不同的平台,不同的JDK供应商,将JDK捆绑在其他产品中,从数据库到飞行模拟器和视频渲染引擎。 大多数JDK都是与另一个JDK的简单链接,例如v.1.2指向已安装的v.6。常见的是如何获得一个版本:java -version 但我们有一些观点:

  1. 对于Windows,请查看注册表HKEY_LOCAL_MACHINE \ Software \ JavaSoft \。还要考虑64位操作系统的Wow6432Node模式。检查主要供应商的产品,这些供应商在Java上创建软件。
  2. Mac OS X /System/Library/Frameworks/JavaVM.framework/Versions + Open JDK
  3. Linux和其他一些Unix--其中java,/ etc / alternatives / java rpm -qa | grep java等。

答案 4 :(得分:1)

安装非常模糊 - 即使JRE和JDK附带安装程序,实际上我们只需要将JRE或JDK的文件复制到一台机器上并且可以立即使用

因此,一般情况下,您必须在本地计算机上找到所有javajava.exe个可执行文件,并致电java -version以查看 1 < / sup>,如果名为java / java.exe的可执行文件确实是(JRE)的一部分。


刚才看到,你要求寻找JVM,并希望调用编译器。如果你正在寻找JDK,请使用上面的方法,但找到所有{{ 1}} / javac。他们也有javac.exe选项。


1 这种方法带来了没有风险 - 没有乐趣 - 请仔细看看肖恩的评论!如果您不能信任计算机(或其用户),那么您可能想要测试,如果可执行文件是脚本/批处理或二进制文件 - 即使二进制文件也可以擦除你的磁盘,即使是原始的javac可执行文件也可以用一些邪恶的代码代替......

答案 5 :(得分:1)

在Windows命令提示符下,您可以运行以下2个命令(或作为批处理文件一起运行)。这将在Windows添加/删除程序(注册表项)中查看,并报告在安装/卸载链接中找到哪些版本的JAVA,以及文件夹路径和其他一些内容。

wmic product where "name LIKE '%%java%%'" get * /format:textvaluelist > temp_javavers.txt 

notepad.exe temp_javavers.txt

答案 6 :(得分:0)

我不确定这是否能回答您的问题,但您可以使用系列版本控制功能控制applet或WebStart应用程序将在哪个主要JRE版本上运行。这允许您指定运行applet的主要Java版本。您应该能够从那里获得javac的位置。

http://www.oracle.com/technetwork/java/javase/family-clsid-140615.html

答案 7 :(得分:0)

  

..使用特定的JDK版本编译一些源代码。

使用适用于-source-target &amp;的-bootclasspath JavaCompilerCross-Compilation Options(用户可以在最新的JDK中使用)。 javac 即可。最后两个是JFileChooser {{3}}的一部分。

关于查找JDK,使用当前JRE的路径弹出{{1}}作为默认目录。如果用户无法从那里导航到JDK,那么他们应该编写代码是值得怀疑的。

答案 8 :(得分:0)

要查找安装的Java版本,比查找javac.exe更好的方法是检查注册表。 Java创建一个注册表项HKEY_LOCAL_MACHINE\Software\JavaSoft\Java Runtime Environment,其字符串CurrentVersion设置为版本号,例如1.5或1.6。

您可以使用该信息查找JVM:

  • HKEY_LOCAL_MACHINE \ Software \ JavaSoft \ Java Runtime Environment \ 1.5 \ JavaHome = C:\ Program Files \ Java \ j2re1.5
  • HKEY_LOCAL_MACHINE \ Software \ JavaSoft \ Java Runtime Environment \ 1.5 \ RuntimeLib = C:\ Program Files \ Java \ j2re1.4.2 \ bin \ client \ jvm.dll

您可以在此处查看更多内容:http://java.sun.com/j2se/1.4.2/runtime_win32.html

我不熟悉从 Java访问注册表,但您始终可以运行regedit的命令行界面并解析结果,这就是我所做的。

答案 9 :(得分:0)

如前所述,HKEY_LOCAL_MACHINE\SOFTWARE\JavaSoft\Java Runtime Environment应该列出可用的JVM,但我发现虽然这个注册表目前只在我的机器上列出了一个64位的1.6.0_31 JVM,但我也有一个32位的java安装在C:\Windows\SysWOW64下。

因此,注册表中列出的信息并未描绘全貌,如果要在64位Windows中运行32位JVM,还应尝试C:\Windows\SysWOW64