关于URI is hierarchical错误的解决方案

关于URI is hierarchical错误的解决方案

 

 

 

今天在工程导出成Jar包的时候碰见了该问题,工程中运行正确,Jar包中运行出错,分析了下主要在于文件读取的时候出了问题,最终用流进行读取的方式解决了该问题,调用了getResourceAsStream()方法,如果有朋友遇到了该问题,不妨试试。

-------------------------------------------------------------------------------

 

关于URI is hierarchical错误的解决方案: getResourceAsStream()
===============================================================
从类路径加载(一)
关键字: 类路径 ClassLoader
JAR文件是打包基于JAVA技术的解决方案的标准方法。它允许开发者将所有相关内容(.class、图片、声音、及所有支持的文件)打包到一个文件中。JAR格式支持压缩、认证、版本号及其他很多特性。
  
  从JAR文件中读取文件是很麻烦的事情,但是也不一定全是这样。本文将告诉你如何从JAR文件中读取文件,首先,得到JAR文件中的文件目录,然后在进行具体的某一个文件的操作。
  
  如果你对于常用的ZIP格式比较熟悉的话,JAR文件也就差不多。JAR文件提供一种将多个文件打包到一个文件中的方法,其中每一个文件可能独立地被压缩。JAR文件所增加的内容是manifest,它允许开发者可以提供附加的关于内容的信息。例如,manifest表明JAR文件中的哪个文件是用来运行一个程序的,或者库的版本号等
  
  J2SEDK提供了一个jar工具,你可以用它从控制台读写JAR文件。然而,如果你需要在程序中代码读写JAR文件,可能需要一点时间(本文只包含如何在程序中读写JAR文件)。好消息是你可以做到这一点,而且你不用担心解压的事,因为类库将帮助你完成这些。你所需要的类文件位于 java.util.jar包中。这里主要的类是JarFile,它是.jar文件自身的一个引用。较大文件中的每一个单独的文件是通过一个 JarEntry引用的。
  
  首先,通过把将JAR文件位置传给构造函数,创建一个JarFile的实例,位置可能是String或File的形式,如下:
    JarFile jarFile = new JarFile("thefile.jar");
  
  或者:
  
    File file = new File("thefile.jar");
    JarFile jarFile = new JarFile(file);
  
  
  你可能注意到当文件不在class path中时,JarFile类对于从JAR中读取文件文件是很有用的。
  当你想指定目标JAR文件时,JarFile类对于从JAR中读取文件同样也很有用。当然,如果JAR文件在class path中,从其中读取文件的方法比较简单,你可以用下面的方法:
  
   URL url = ClassLoader.getSystemResource(name);
  或者
  
   InputStream stream =
     ClassLoader.getSystemResourceAsStream(name);
  
  这种技术允许你从在class path中的JAR文件里读取文件。你不必指定JAR文件名。
  
  
  还有其他的构造函数,如认证支持,标志要删除的文件等。这些构造函数在这儿不做说明。
  
  当你有了该JAR文件的一个引用之后,你就可以读取其文件内容中的目录信息了。JarFile的entries方法返回所有entries的枚举集合 (Enumeration)。通过每一个entry,你可以从它的manifest文件得到它的属性,任何认证信息,以及其他任何该entry的信息,如它的名字或者大小等。
  
   Enumeration enum = jarFile.entries();
   while (enum.hasMoreElements()) {
    process(enum.nextElement());
   }
  
  正如前面提到的,每一个entry是一个JarEntry。该类有getName,getSize,getCompressedSize等方法。
  
  下面,让我们举例说明在程序中如何使用这些特性。下面的程序显示你指定的JAR文件中内容的名称,大小,压缩大小等(这有点类似于在使用jar命令时,指定"t"和"v"选项)。
  
   import java.io.*;
   import java.util.*;
   import java.util.jar.*;
  
   public class JarDir {
    public static void main (String args[]) throws IOException {
     if (args.length != 1) {
      System.out.println(
        "Please provide a JAR filename");
      System.exit(-1);
     }

     JarFile jarFile = new JarFile(args[0]);
     Enumeration enum = jarFile.entries();
     while (enum.hasMoreElements()) {
      process(enum.nextElement());
     }
    }
  
    private static void process(Object obj) {
     JarEntry entry = (JarEntry)obj;
     String name = entry.getName();
     long size = entry.getSize();
     long compressedSize = entry.getCompressedSize();
     System.out.println(
       name + "/t" + size + "/t" + compressedSize);
    }
   }
  
  如果你用J2SE1.4.1中的jce.jar作为参数运行上面的JarDir程序,你将看到类似于下面的输出(...部分为省略掉的输出):
  
  META-INF/MANIFEST.MF  5315  1910
  META-INF/4JCEJARS.SF  5368  1958
  META-INF/4JCEJARS.DSA  2207  1503
  META-INF/    0    2
  javax/ 0    0
  javax/crypto/  0    0
  javax/crypto/interfaces/    0    0
  javax/crypto/interfaces/DHKey.class   209   185
  javax/crypto/interfaces/DHPublicKey.class    265   215
  javax/crypto/interfaces/DHPrivateKey.class   267   215
  javax/crypto/interfaces/PBEKey.class  268   224
  javax/crypto/SecretKey.class  167   155
  ...
  
  注意输出最开始的META-INF行。这是manifest和安全认证信息。0字节的entries不是文件,但是有点相当于目录。
  
  为了从JAR文件中真正读取一个指定的文件,你必须到其entry的InputStream。这和JarEntry不一样。这是因为JarEntry 只是包含该entry的有关信息,但是并不实际包含该entry的内容。这和File和FileInputStream的区别有点儿相似。访问文件没有打开文件,它只是从目录中读取了该文件的信息。下面是如何得到entry的InputStream:
  
   InputStream input = jarFile.getInputStream(entry);
  
  当你有了输入流,你就可以像读取其他流一样读取它。在文本流中(text stream),记得使用读取器(Reader)从流中取得字符。对于面向字节的流,如图片文件,直接读取就行了。
  
  下面的程序演示如何从JAR文件中读取文件。指定JAR文件的名称,要读取的文件的名称(打包JAR文件中的某一个文件)作为参数来调用该程序。要读取的文件应该有一个文本类型的。
  
   import java.io.*;
   import java.util.jar.*;
  
   public class JarRead {
    public static void main (String args[])
      throws IOException {
     if (args.length != 2) {
      System.out.println(
       "Please provide a JAR filename and file to read");
      System.exit(-1);
     }
     JarFile jarFile = new JarFile(args[0]);
     JarEntry entry = jarFile.getJarEntry(args[1]);
     InputStream input = jarFile.getInputStream(entry);
     process(input);
     jarFile.close();
    }
  
    private static void process(InputStream input)
      throws IOException {
     InputStreamReader isr =
     new InputStreamReader(input);
     BufferedReader reader = new BufferedReader(isr);
     String line;
     while ((line = reader.readLine()) != null) {
      System.out.println(line);
     }
     reader.close();
    }
   }
  
  假设在myfiles.jar文件中有一个spider.txt文件,spider文件的内容如下:
  
   The itsy bitsy spider
   Ran up the water spout
   Down came the rain and
   Washed the spider out
  
  你可以通过下面的命令来显示该文本文件的内容:
  
   java JarRead myfiles.jar spider.txt 
=====================================================================
在Java的程序发布中,很多人会选择采用二进制的jar的格式进行发布,怎么样读取Jar里面的资源呢?
主要是采用ClassLoader的下面几个方法来实现:
public URL getResource(String name);
public InputStream getResourceAsStream(String name)
public static InputStream getSystemResourceAsStream(String name)
public static URL getSystemResource(String name)
后两个方法可以看出是静态的方法,这几个方法都可以从Jar中读取图片资源,但是对与动画的gif文件,笔者在尝试过程中发现,存在一些差异。
String gifName为Gif文件在Jar中的相对路径。
(1)使用了两个静态方法
BufferedImage image=ImageIO.read(ClassLoader.getSystemResourceAsStream(gifName));
或者
Image image=Toolkit.getDefaultToolkit().getImage(ClassLoader.getSystemResource(gifName));
这两种方式可以成功地读取gif文件,但是对于gif动画,显示出来地是静态的。
(2)使用其他两个方法
Image image=Toolkit.getDefaultToolkit().getImage(this.getClass.getClassLoader().getResource(gifName));
再这种方式下动画可以正常显示了。
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值