您的位置 首页 > 数码极客

如何将java项目打包成jar格式

导读: 最近在看 Flink 源码的时候发现到一段实用的代码,该代码实现了 Java 动态编译以及生成 JAR 文件。将其进行改进后可以应用到我们的平台上,实现在平台页面上编写 java 代码语句,提交后由后台进行编译和打成 Jar 包再上传到指定的文件存储系统,从而代替之前在本地自己手动打 UDF 包的方式。下面我将对这段代码做一些简单分析,希望对各位有所帮助。

核心代码

public class TestUserClassLoaderJar { private static final String GENERATED_UDF_CLASS = "LowerUDF"; private static final String GENERATED_UDF_CODE = "public class " + GENERATED_UDF_CLASS + " extends extends org.a {\n" + " public String eval(String str) {\n" + " return ();\n" + " }\n" + "}\n"; /** * 将生成的 UDF class 打包到 JAR 中并且返回 JAR 所在的路径. */ public static file createJarFile(File tmpDir, String jarName) throws IOException { // 创建一个 java 文件 File javaFile = Pa(), GENERATED_UDF_CLASS + ".java").toFile(); javaFile.createNewFile(); // 将代码写入 java 文件中 FileU(javaFile, GENERATED_UDF_CODE); // 编译 java文件生成 class 文件 DiagnosticCollector<JavaFileObject> diagnostics = new DiagnosticCollector<>(); JavaCompiler compiler = ToolProvider.getSystemJavaCompiler(); StandardJavaFileManager fileManager = com(diagnostics, null, null); Iterable<? extends JavaFileObject> compilationUnit = (javaFile)); JavaCom task = com( null, fileManager, diagnostics, Collec(), null, compilationUnit); // 此处结果返回一个布尔值,可用于判断是否编译成功及是否执行下面的打包操作 (); // 将 class 文件打包到 Jar 中 File classFile = Pa(), GENERATED_UDF_CLASS + ".class").toFile(); File jarFile = Pa(), jarName).toFile(); JarOutputStream jos = new JarOutputStream(new FileOutputStream(jarFile)); JarEntry jarEntry = new JarEntry(GENERATED_UDF_CLASS + ".class"); jos.putNextEntry(jarEntry); byte[] classbytes = FileU()); jos.write(classBytes); jos.closeEntry(); jos.close(); return jarFile; } public static void main(String[] args) throws IOException { createJarFile(new File("G:\\jarSave\\"),";); } }

以上代码主要完成以下三步操作:

  • 创建一个 .java 文件,并将外部输入的 java 语句写入到该文件中
  • 对 java 文件进行编译并生成 class 文件
  • 将 class 文件打包到 JAR 中并返回 JAR 的路径

下图是动态编译的几个关键类的创建方式及作用:

JavaCompiler 的由来

在上面代码中通过 ToolProvider.getSystemJavaCompiler() 获取到 JavaCompiler。深入内部 findSystemToolClass() 方法发现其最终先是通过 Sy("java.home") 获取到 目录,再获取其上级目录中 lib 目录下的 (也就是 ),并进行动态加载 Jar 获取到 JavaCompiler。

findSystemToolClass 代码片段:

private Class<?> findSystemToolClass(String toolClassName) throws MalformedURLException, ClassNotFoundException { // try loading class directly, in case tool is on the bootclasspath try { return Cla(toolClassName, false, null); } catch (ClassNotFoundException e) { trace(FINE, e); // if tool not on bootclasspath, look in default tools location () ClassLoader cl = (refToolClassLoader == null ? null : re()); if (cl == null) { File file = new File(Sy("java.home")); if ().equalsIgnoreCase("jre")) file = (); 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 (!()) throw e; URL[] urls = { ().toURL() }; trace(FINE, urls[0].toString()); cl = URLCla(urls); refToolClassLoader = new WeakReference<ClassLoader>(cl); } return Cla(toolClassName, false, cl); } }

补充:FileUtils 工具类(已删减,只保留所需部分)

public class FileUtils { public static void writeFileUtf8(File file, String contents) throws IOException { writeFile(file, contents, "UTF-8"); } public static void writeFile(File file, String contents, String encoding) throws IOException { byte[] bytes = con(encoding); Files.write(), bytes, new OpenOption[]{S}); } private static byte[] read(InputStream source, int initialSize) throws IOException { int capacity = initialSize; byte[] buf = new byte[initialSize]; int nread = 0; while (true) { int n; while ((n = (buf, nread, Ma(capacity - nread, 4096))) > 0) { nread += n; } if (n < 0 || (n = ()) < 0) { return capacity == nread ? buf : Arrays.copyOf(buf, nread); } if (capacity <= 2147483639 - capacity) { capacity = Ma(capacity << 1, 4096); } else { if (capacity == 2147483639) { throw new OutOfMemoryError("Required array size too large"); } capacity = 2147483639; } buf = Arrays.copyOf(buf, capacity); buf[nread++] = (byte) n; } } public static byte[] readAllBytes(Path path) throws IOException { SeekableByteChannel channel = Files.newByteChannel(path); Throwable var2 = null; byte[] var7; try { InputStream in = C(channel); Throwable var4 = null; try { long size = c(); if (size > 2147483639L) { throw new OutOfMemoryError("Required array size too large"); } var7 = read(in, (int) size); } catch (Throwable var30) { var4 = var30; throw var30; } finally { if (in != null) { if (var4 != null) { try { in.close(); } catch (Throwable var29) { var4.addSuppressed(var29); } } else { in.close(); } } } } catch (Throwable var32) { var2 = var32; throw var32; } finally { if (channel != null) { if (var2 != null) { try { c(); } catch (Throwable var28) { var2.addSuppressed(var28); } } else { c(); } } } return var7; } }

最后

以上就是动态编译 Java 代码以及生成 Jar 文件的方式。

感谢您的阅读,如果喜欢本文欢迎关注和转发,本头条号将坚持持续分享IT技术知识。对于文章内容有其他想法或意见建议等,欢迎提出共同讨论共同进步。

责任编辑: 鲁达

1.内容基于多重复合算法人工智能语言模型创作,旨在以深度学习研究为目的传播信息知识,内容观点与本网站无关,反馈举报请
2.仅供读者参考,本网站未对该内容进行证实,对其原创性、真实性、完整性、及时性不作任何保证;
3.本站属于非营利性站点无毒无广告,请读者放心使用!

“如何将java项目打包成jar格式,java项目如何打包发布”边界阅读