从一个不在eclipse中工作的jar中读取文件

时间:2012-03-04 20:47:42

标签: java eclipse file-io jar

新手用户在这里,我第一次在这里发帖,所以我会尽力做到对不起,抱歉任何错误。

我一直在尝试从jar内的文件夹中读取文件,并将其路径添加到字符串数组中。我的jar文件夹结构如下所示: .class文件位于FileIO.jar \ com \ yumeprojects \ flowdemonstration \ 图像位于FileIO.jar \ com \ yumeprojects \ flowdemonstration \ images \

当我从eclipse外的编译jar文件运行程序时,结果是准确的,找到5个图像并将它们的路径添加到数组中。 这是输出:

E:\Yumeprojects 2011-2012\File IO tests>java -jar "FileIO0_2.jar"
(file:/E:/Yumeprojects%202011-2012/File%20IO%20tests/FileIO0_2.jar <no signer ce
rtificates>)
com/yumeprojects/flowdemonstration/test/bakgrund01.png
com/yumeprojects/flowdemonstration/test/bakgrund02.png
com/yumeprojects/flowdemonstration/test/bakgrund03.png
com/yumeprojects/flowdemonstration/test/bakgrund04.png
com/yumeprojects/flowdemonstration/test/bakgrund05.png
Number of images: 5

当我从日食中跑步时,我得到了这个:

(file:/E:/Yumeprojects%202011-2012/Project%20Mathematics/flow/bin/ <no signer certificates>)
Number of images: 0

代码看起来像这样,不是我的,我从这个网站复制了它。

package com.yumeprojects.flowdemonstration;

import java.io.File;
import java.io.IOException;
import java.net.URL;
import java.security.CodeSource;
import java.util.ArrayList;
import java.util.List;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;

public class Main {

   String[] fileNames;
   CodeSource src = Main.class.getProtectionDomain().getCodeSource();
   List<String> list = new ArrayList<String>();

   public static void main(String[] args){
      Main m = new Main();
      m.init();
   }

   public void init(){
      try {
         read();
      } catch (IOException e) {
         // TODO Auto-generated catch block
         e.printStackTrace();
      }
   }



   public void read() throws IOException{
      if(src != null){
         URL jar = src.getLocation();
         System.out.println(jar);

          ZipInputStream zip = new ZipInputStream(jar.openStream());
          System.out.println(zip);

          System.out.println(zip.getNextEntry());

          ZipEntry ze = null;
          ZipEntry ze2 = null;
          while((ze = zip.getNextEntry()) != null){
              String entryName = ze.getName();

              if(entryName.endsWith(".png") || entryName.endsWith(".PNG") ) {
                  list.add(entryName);
                  System.out.println(entryName);
              }
          }
      }
      fileNames = list.toArray(new String[list.size() ] );
      System.out.println("Number of images: " + fileNames.length);
    } 
}

为什么这段代码在eclipse中不起作用?是否需要在其中执行jar文件?如果确实如此,我如何让eclipse执行它以便我可以在eclipse中使用这个代码,并获得与从Eclipse外的jar启动它时获得的相同结果?

请帮忙

//库尔滕

EDIT 由于注释对解析代码不好,我会在这里解析它 Syso在日食中看起来像这样:

file:/E:/Yumeprojects%202011-2012/Project%20Mathematics/flow/bin/
java.util.zip.ZipInputStream@a62fc3
null
Number of images: 0

和外面:

E:\Yumeprojects 2011-2012\File IO tests>java -jar "FileIO0_3.jar"
file:/E:/Yumeprojects%202011-2012/File%20IO%20tests/FileIO0_3.jar
java.util.zip.ZipInputStream@649e4dc0
META-INF/MANIFEST.MF
com/yumeprojects/flowdemonstration/test/bakgrund01.png
com/yumeprojects/flowdemonstration/test/bakgrund02.png
com/yumeprojects/flowdemonstration/test/bakgrund03.png
com/yumeprojects/flowdemonstration/test/bakgrund04.png
com/yumeprojects/flowdemonstration/test/bakgrund05.png
Number of images: 5

E:\Yumeprojects 2011-2012\File IO tests>

第二次编辑:这是工作代码!它不漂亮,代码结构完全是地狱,但它做了它应该做的事情。感谢Guillaume Polet帮助我解决这个问题!

package com.yumeprojects.flowdemonstration;

import java.io.File;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URL;
import java.security.CodeSource;
import java.util.ArrayList;
import java.util.List;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;

public class Main {

   String[] fileNames;
   CodeSource src = Main.class.getProtectionDomain().getCodeSource();
   List<String> list = new ArrayList<String>();
   URL jar;
   URI jar2;
   File source;
   String[] images;
   static boolean inIDE;

   public static void main(String[] args){
      if(args.length > 0){
          for(int i = 0; i < args.length; i++){
              if(args[i].equalsIgnoreCase("inIDE")){
                  inIDE = true;
              }
          }
      }
      Main m = new Main();
      m.init();
   }

   public void init(){
      try {
         read();
      } catch (IOException e) {
         // TODO Auto-generated catch block
         e.printStackTrace();
      }
   }



   public void read() throws IOException{

       if(src != null){
         jar = src.getLocation();

         if(inIDE){
             try {
                jar2 = new URI(jar.toURI().toString() + "/com/yumeprojects/flowdemonstration/test");
            } catch (URISyntaxException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            System.out.println(jar2);
            source = new File(jar2);
            images = source.list();
            for(int i = 0; i < images.length; i++){
                 System.out.println(images[i]); 
            }
         }else{
             System.out.println(jar); 
             ZipInputStream zip = new ZipInputStream(jar.openStream());
             System.out.println(zip);
             System.out.println(zip.getNextEntry());

              ZipEntry ze = null;

              while((ze = zip.getNextEntry()) != null){
                  String entryName = ze.getName();

                  if(entryName.endsWith(".png") || entryName.endsWith(".PNG") ) {
                      list.add(entryName);
                      System.out.println(entryName);
                  }
              }
              fileNames = list.toArray(new String[list.size() ] );
              System.out.println("Number of images: " + fileNames.length);
         }




      }

    } 
}

1 个答案:

答案 0 :(得分:0)

错误就像我在之前的评论中所述:在Eclipse中,URL指向文件夹,同时从打包的jar执行,URL指向jar。在文件夹上创建ZipInputStream当然没有意义。

修复代码的一种方法是查找网址的结束方式:

如果以.jar结尾,那么您可以保留实际代码。

如果没有,则需要一个递归方法来查找文件。将您的URL转换为URI(toURI()),然后创建一个File对象。然后递归扫描该文件夹并像以前一样查找文件。

在这种情况下,这可能对您有用,但我认为它并不涵盖所有可能性,因此这绝对是最佳解决方案。

此外,我不喜欢这个想法,根据处于开发阶段或生产阶段的情况,执行相同任务的代码非常不同。您应该尝试找到解决问题的另一种方法。其他一些人可能会为您提供更好的解决方案。