Java:在运行时将类添加到 Jar 存档

Posted

技术标签:

【中文标题】Java:在运行时将类添加到 Jar 存档【英文标题】:Java: Add Class to Jar archive at runtime 【发布时间】:2010-06-24 14:08:09 【问题描述】:

我想在运行时将一些已编译的类(.class 文件)添加到当前 Jar 文件中的目录(包) 我怎样才能做到这一点?

谢谢

【问题讨论】:

为什么在运行时?为什么将其添加到 jar 文件而不是将类添加到类路径?我问这个是因为这个想法似乎有点奇怪。 我使用这个类作为程序的插件,程序从它的包中读取它的插件我还需要这个程序是一个单独的 jar 文件 【参考方案1】:

这是无法做到的 - 要更新 Jar 文件,您需要创建一个新文件并用新文件覆盖旧文件。

以下是有关如何执行此操作的示例:

import java.io.*;
import java.util.*;
import java.util.zip.*;
import java.util.jar.*;

public class JarUpdate 
   /**
    * main()
    */
   public static void main(String[] args) throws IOException 
      // Get the jar name and entry name from the command-line.

      String jarName = args[0];
      String fileName = args[1];

      // Create file descriptors for the jar and a temp jar.

      File jarFile = new File(jarName);
      File tempJarFile = new File(jarName + ".tmp");

      // Open the jar file.

      JarFile jar = new JarFile(jarFile);
      System.out.println(jarName + " opened.");

      // Initialize a flag that will indicate that the jar was updated.

      boolean jarUpdated = false;

      try 
         // Create a temp jar file with no manifest. (The manifest will
         // be copied when the entries are copied.)

         Manifest jarManifest = jar.getManifest();
         JarOutputStream tempJar =
            new JarOutputStream(new FileOutputStream(tempJarFile));

         // Allocate a buffer for reading entry data.

         byte[] buffer = new byte[1024];
         int bytesRead;

         try 
            // Open the given file.

            FileInputStream file = new FileInputStream(fileName);

            try 
               // Create a jar entry and add it to the temp jar.

               JarEntry entry = new JarEntry(fileName);
               tempJar.putNextEntry(entry);

               // Read the file and write it to the jar.

               while ((bytesRead = file.read(buffer)) != -1) 
                  tempJar.write(buffer, 0, bytesRead);
               

               System.out.println(entry.getName() + " added.");
            
            finally 
               file.close();
            

            // Loop through the jar entries and add them to the temp jar,
            // skipping the entry that was added to the temp jar already.

            for (Enumeration entries = jar.entries(); entries.hasMoreElements(); ) 
               // Get the next entry.

               JarEntry entry = (JarEntry) entries.nextElement();

               // If the entry has not been added already, add it.

               if (! entry.getName().equals(fileName)) 
                  // Get an input stream for the entry.

                  InputStream entryStream = jar.getInputStream(entry);

                  // Read the entry and write it to the temp jar.

                  tempJar.putNextEntry(entry);

                  while ((bytesRead = entryStream.read(buffer)) != -1) 
                     tempJar.write(buffer, 0, bytesRead);
                  
               
            

            jarUpdated = true;
         
         catch (Exception ex) 
            System.out.println(ex);

            // Add a stub entry here, so that the jar will close without an
            // exception.

            tempJar.putNextEntry(new JarEntry("stub"));
         
         finally 
            tempJar.close();
         
      
      finally 
         jar.close();
         System.out.println(jarName + " closed.");

         // If the jar was not updated, delete the temp jar file.

         if (! jarUpdated) 
            tempJarFile.delete();
         
      

      // If the jar was updated, delete the original jar file and rename the
      // temp jar file to the original name.

      if (jarUpdated) 
         jarFile.delete();
         tempJarFile.renameTo(jarFile);
         System.out.println(jarName + " updated.");
      
   

【讨论】:

但是 Jar 文件是 zip 格式的,可​​以将文件添加到 zip 存档中! @Snigger 如果您使用 WinZip 或 7Zip,是的,不是在 Java 中。 @Snigger:Java 的 zip 支持不如那些程序强大;重新实现或扩展它以支持向现有档案添加条目可能需要做很多工作。而且它很可能仍然无法执行您想要的操作,WinZip 或 7Zip 也无法执行,因为您要写入的文件将被一直从中读取类的 JVM 锁定。【参考方案2】:

我不太确定,但我认为不可能从 JAR 文件(我们称之为 foo.jar)加载类,然后修改加载类的同一个 JAR 文件,添加一个新的类并期望类加载器找到该类。

我会考虑重构应用程序本身并使其能够动态加载类(使用 URLClassLoader 或任何其他技术),而不是尝试强制执行您描述的单个 JAR 行为。

【讨论】:

以上是关于Java:在运行时将类添加到 Jar 存档的主要内容,如果未能解决你的问题,请参考以下文章

在进程运行时将符号链接更新到 jar 是不是安全?

在运行时将行和键添加到android中的自定义键盘

在运行作为 JAR 存档分发的项目时加载图像等资源

在运行时添加 JDBC jar 时没有为 HSQL db 找到合适的 JDBC 驱动程序?

java:请给一个简单的能双击就能运行的jar文件打包过程.

如何在运行时将onclick侦听器添加到游戏对象统一[重复]