起因:
我在做一个编译Java代码的功能,基本写的差不多了,我就想把它打包部署到我服务器上跑一跑,但是这不做不知道,一做果然就出了问题。我在IDEA上跑一点问题都没有,但是打包成Jar后,后台就显示空指针异常。
排坑:(这里解决办法仅供参考)
Maven打包是没问题的,而且Jar包也能正常跑,说明我的Maven设置和Spring依赖问题应该是没问题的,而经过我一番检查,空指针出现在:
//获得javacompile实例 JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
我都蒙了,因为这个是Java本身提供的方法,为啥获取不到JavaCompiler。于是我进一步查了这个的源代码:
public static JavaCompiler getSystemJavaCompiler() { return instance().getSystemTool(JavaCompiler.class, defaultJavaCompilerName); }
好像看不出啥,在往下:
privateT getSystemTool(Class clazz, String name) { Class extends T> c = getSystemToolClass(clazz, name); try { return c.asSubclass(clazz).newInstance(); } catch (Throwable e) { trace(WARNING, e); return null; } }
还是看不出啥,再进去一层:
privateClass extends T> getSystemToolClass(Class clazz, String name) { Reference > refClass = toolClasses.get(name); Class> c = (refClass == null ? null : refClass.get()); if (c == null) { try { c = findSystemToolClass(name); } catch (Throwable e) { return trace(WARNING, e); } toolClasses.put(name, new WeakReference >(c)); } return c.asSubclass(clazz); }
查了一个get方法,还是没啥头绪,再看findSystemToolClass()
private static final String[] defaultToolsLocation = { "lib", "tools.jar" };
private Class> findSystemToolClass(String toolClassName) throws MalformedURLException, ClassNotFoundException { // try loading class directly, in case tool is on the bootclasspath try { return Class.forName(toolClassName, false, null); } catch (ClassNotFoundException e) { trace(FINE, e); // if tool not on bootclasspath, look in default tools location (tools.jar) ClassLoader cl = (refToolClassLoader == null ? null : refToolClassLoader.get()); if (cl == null) { File file = new File(System.getProperty("java.home")); if (file.getName().equalsIgnoreCase("jre")) file = file.getParentFile(); for (String name : defaultToolsLocation) file = new File(file, name); // if tools not found, no point in trying a URLClassLoader // so rethrow the original exception. if (!file.exists()) throw e; URL[] urls = { file.toURI().toURL() }; trace(FINE, urls[0].toString()); cl = URLClassLoader.newInstance(urls); refToolClassLoader = new WeakReference(cl); } return Class.forName(toolClassName, false, cl); } }
看到加粗斜体的地方似乎有点感觉了,但是我还是没太想通问题在哪,我找了找百度,原来这么多人跟我遇到了一样的问题,最早的一篇博客还是09年写的。。
原来这里首先是Java通过System.getProperty()获得环境变量,然后在找lib下面的tools.jar,这个tool.jar里面就有JavaCompiler,随后我找了发现jre里面并没有这个包,但是jdk里面有。于是我把jdk目录下的tool.jar复制到了jre下,但是仍然不行,这时候我估计我可能是我的Java环境变量的设置问题,于是我检查了一遍,但是java命令和javac命令都没问题。。。而且我打印System。getProperty("java.home")也跟本身电脑设置的JAVA_HOME有点不同,有点玄学。。
最后我总结出两个解决办法:
一个就是网上的复制tool.jar的方法,也就是把jre目录下面缺的包补上。但是这种在我这没起作用
另一个是我自己的土办法,我手动调用jre/bin下面的java.exe 来运行这个jar包,当然提前也是需要把tool.jar复制进去,问题就没了。
这样就没有报空指针错误了。