Подтвердить что ты не робот

Как использовать JarOutputStream для создания JAR файла?

Как создать программный файл JAR с помощью java.util.jar.JarOutputStream? Файл JAR, созданный моей программой, выглядит корректно (он отлично извлечения), но когда я пытаюсь загрузить из него библиотеку, Java жалуется, что не может найти файлы, которые явно хранятся внутри него. Если я извлечу JAR файл и использую инструмент командной строки Sun jar, чтобы повторно сжать его, результирующая библиотека работает нормально. Короче говоря, что-то не так с моим файлом JAR.

Пожалуйста, объясните, как создать файл JAR программно, в комплекте с файлом манифеста.

4b9b3361

Ответ 1

Оказывается, что JarOutputStream имеет три недокументированных причуды:

  • Имена каталогов должны заканчиваться символом "/" .
  • Пути должны использовать "/" косые черты, а не "\"
  • Записи могут не начинаться с косой черты "/" .

Вот правильный способ создания Jar файла:

public void run() throws IOException
{
  Manifest manifest = new Manifest();
  manifest.getMainAttributes().put(Attributes.Name.MANIFEST_VERSION, "1.0");
  JarOutputStream target = new JarOutputStream(new FileOutputStream("output.jar"), manifest);
  add(new File("inputDirectory"), target);
  target.close();
}

private void add(File source, JarOutputStream target) throws IOException
{
  BufferedInputStream in = null;
  try
  {
    if (source.isDirectory())
    {
      String name = source.getPath().replace("\\", "/");
      if (!name.isEmpty())
      {
        if (!name.endsWith("/"))
          name += "/";
        JarEntry entry = new JarEntry(name);
        entry.setTime(source.lastModified());
        target.putNextEntry(entry);
        target.closeEntry();
      }
      for (File nestedFile: source.listFiles())
        add(nestedFile, target);
      return;
    }

    JarEntry entry = new JarEntry(source.getPath().replace("\\", "/"));
    entry.setTime(source.lastModified());
    target.putNextEntry(entry);
    in = new BufferedInputStream(new FileInputStream(source));

    byte[] buffer = new byte[1024];
    while (true)
    {
      int count = in.read(buffer);
      if (count == -1)
        break;
      target.write(buffer, 0, count);
    }
    target.closeEntry();
  }
  finally
  {
    if (in != null)
      in.close();
  }
}

Ответ 2

Там еще одна "причуда", на которую нужно обратить внимание: все имена JarEntry НЕ должны начинаться с "/".

Например: Имя записи jar для файла манифеста "META-INF/MANIFEST.MF", а не "/META-INF/MANIFEST.MF".

То же самое правило должно соблюдаться для всех записей jar.

Ответ 4

Вы можете сделать это с помощью этого кода:

public void write(File[] files, String comment) throws IOException {
    FileOutputStream fos = new FileOutputStream(PATH + FILE);
    JarOutputStream jos = new JarOutputStream(fos, manifest);
    BufferedOutputStream bos = new BufferedOutputStream(jos);
    jos.setComment(comment);
    for (File f : files) {
        print("Writing file: " + f.toString());
        BufferedReader br = new BufferedReader(new FileReader(f));
        jos.putNextEntry(new JarEntry(f.getName()));
        int c;
        while ((c = br.read()) != -1) {
            bos.write(c);
        }
        br.close();
        bos.flush();
    }
    bos.close();
//  JarOutputStream jor = new JarOutputStream(new FileOutputStream(PATH + FILE), manifest);

}

Переменная PATH: путь к файлу JAR

Переменная FILE: имя и формат