当前位置:首页 > Java技术 > 正文内容

从JAR和zip档案文件中提取Java资源

canca18年前 (2008-06-05)Java技术561
摘要
将一类 java 资源打包在一个 java archive (jar) 文件中是缩短下载时间、增强安全性和增强可管理性的极好方法。这篇技巧说明如何很容易地从 jar 文件中提取资源以供您自己使用。
多数 java 程序员都非常清楚使用 jar 文件将组成 java 解决方案的各种资源(即 .class 文件、声音和图像)打包的优点。(如果您不熟悉 jar 文件,请参阅后文的参考资源部分。)刚开始使用 jar 文件的人常问的一个问题是:“如何从 jar 文件中提取图像呢?”本文将回答这个问题,并会提供一个类,这个类使从 jar 文件中提取任何资源变得非常简单!

加载 gif 图像
假定我们有一个 jar 文件,其中包含我们的应用程序要使用的一组 .gif 图像。下面就是使用 jarresources 访问 jar 文 械耐枷裎募 姆椒ǎ?

     jarresources jar = new jarresources ("images.jar");
     image logo =
toolkit.getdefaulttoolkit().createimage (jar.getresource ("logo.gif");

这段代码说明我们可以创建一个 jarresources 对象,并将其初始化为包含我们要使用的资源的 jar 文件 -- images.jar。随后我们使用 jarresourcesgetresource() 方法将来自 logo.gif 文件的原始数据提供给 awt toolkit 的 createimage() 方法。

命名说明
jarresource 是一个非常简单的示例,它说明了如何使用 java 1.1 所提供的各种功能来处理 jar 和 zip 档案文件。

关于命名的简要说明。java 中的归档支持实际上是以流行的 zip 归档格式为起点的(请参阅 "java tip 21: use archive files to speed up applet loading")。因此,在最初实现处理档案文件的 java 支持时,所有类文件以及诸如此类的东西并未放在 java.util.zip 包中;这些类通常以 "zip" 开头。但在转向 java 1.1 时,功能已发生了变化,档案文件的名称也更具有 java 特征。因此,现在我们称之为 jar 文件的文件基本上是 zip 文件。

工作方式
jarresources 类的重要数据域用来跟踪和存储指定 jar 文件的内容:

public final class jarresources {

    public boolean debugon=false;

    private hashtable htsizes=new hashtable();
    private hashtable htjarcontents=new hashtable();

    private string jarfilename;

这样,该类的实例化设置 jar 文件的名称,然后转到 init() 方法完成全部实际工作。

    public jarresources(string jarfilename) {
       this.jarfilename=jarfilename;
       init();
    }

现在,init() 方法只将指定 jar 文件的整个内容加载到一个 hashtable(通过资源名访问)中。

这是一个相当有用的方法,下面我们对它作进一步的分析。zipfile 类为我们提供了对 jar/zip 档案头信息的基本访问方法。这类似于文件系统中的目录信息。下面我们列出 zipfile 中的所有条目,并用档案中每个资源的大小添充 htsizes hashtable:

    private void init() {
       try {
           zipfile zf=new zipfile(jarfilename);
           enumeration e=zf.entries();
           while (e.hasmoreelements()) {
               zipentry ze=(zipentry)e.nextelement();
               if (debugon) {
                  system.out.println(dumpzipentry(ze));
               }
               htsizes.put(ze.getname(),new integer((int)ze.getsize()));
           }
           zf.close();

接下来,我们使用 zipinputstream 类访问档案。zipinputstream 类完成了全部魔术,允许我们单独读取档案中的每个资源。我们从档案中读取组成每个资源的精确字节数,并将其存储htjarcontents hashtable 中,您可以通过资源名访问这些数据:

           fileinputstream fis=new fileinputstream(jarfilename);
           bufferedinputstream bis=new bufferedinputstream(fis);
           zipinputstream zis=new zipinputstream(bis);
           zipentry ze=null;
           while ((ze=zis.getnextentry())!=null) {
              if (ze.isdirectory()) {
                 continue;
              }
              if (debugon) {
                 system.out.println(
                    "ze.getname()="+ze.getname()+","+"getsize()="+ze.getsize()
                    );
              }
              int size=(int)ze.getsize();
              // -1 表示大小未知。
              if (size==-1) {
                 size=((integer)htsizes.get(ze.getname())).intvalue();
              }
              byte[] b=new byte[(int)size];
              int rb=0;
              int chunk=0;
              while (((int)size - rb) > 0) {
                  chunk=zis.read(b,rb,(int)size - rb);
                  if (chunk==-1) {
                     break;
                  }
                  rb+=chunk;
              }
              // 添加到内部资源 hashtable 中
              htjarcontents.put(ze.getname(),b);
              if (debugon) {
                 system.out.println(
                    ze.getname()+" rb="+rb+
                    ",size="+size+
                    ",csize="+ze.getcompressedsize()
                    );
              }
           }
        } catch (nullpointerexception e) {
           system.out.println("done.");
        } catch (filenotfoundexception e) {
           e.printstacktrace();
        } catch (ioexception e) {
           e.printstacktrace();
        }
    }

请注意,用来标识每个资源的名称是档案中资源的限定路径名,例如,不是包中的类名 -- 即 java.util.zip 包中的 zipentry 类将被命名为 "java/util/zip/zipentry",而不是 "java.util.zip.zipentry"。

代码的最后一个重要部分是简单的测试驱动程序。该测试驱动程序是一个简单的应用程序,它接收 jar/zip 档案名和资源名。它试图发现档案中的资源文件,然后将成功或失败的消息报告出来:

    public static void main(string[] args) throws ioexception {
        if (args.length!=2) {
           system.err.println(
              "usage: java jarresources "
              );
           system.exit(1);
        }
        jarresources jr=new jarresources(args[0]);
        byte[] buff=jr.getresource(args[1]);
        if (buff==null) {
           system.out.println("could not find "+args[1]+".");
        } else {
           system.out.println("found "+args[1]+ " (length="+buff.length+").");
        }
    }

} // jarresources 类结束。

您已了解了这个类。一个易于使用的类,它隐藏了使用打包在 jar 文件中的资源的全部棘手问题。

练习
现在您对从档案文件中提取资源已有了一定的认识,下面是可用来修改和扩展 jarresources 类的一些说明:



  • 不在构造期间一次性加载全部内容,而要延迟加载。对于大型 jar 文件,构造期间可能没有足够的内存加载全部文件。


  • 不只是提供类似 getresource() 这样的一般读方法,我们还可提供资源特定的读方法 -- 例如,用来返回 java image 对象的 getimage() 方法,用来返回 java class 对象的 getclass() 方法(在自定义的类加载程序的协助下),等等。如果 jar 文件足够小,则我们可以根据它们的扩展名(.gif、.class 等等)预先构建全部资源。


  • 某些方法应该提供关于给定 jar 文件本身(基本上是 zipfile 的包装)的信息,包括:jar/zip 的条目数;返回全部资源名的 enumerator;返回特定条目长度(和其他属性)的读方法;允许编制索引的读方法,这仅仅是举几个例子。


  • 可对 jarresources 进行扩展,以供 applet 使用。通过利用 applet 参数和 urlconnection 类,就可以从网络上下载 jar 内容,而不是将档案作为本地文件打开。此外,我们还可将该类扩展为一个自定义的 java 内容处理程序。

小结
如果您曾经渴望知道如何从 jar 文件中提取图像,那么您现在已学到了一种方法。有了本技巧提供的这个新类,您就不仅可以用 jar 文件处理图像,而且可以将提取魔术用于 jar 文件中的任何资源。


作者简介
arthur choi 目前是 ibm 的一位顾问程序员。他曾先后在几个公司任职,包括 samsung network laboratory 和 mitre。他参与过的项目包括客户机/服务器系统、分布式对象计算和网络管理。他在各种操作系统环境下用过多种语言。他于 1981 年开始用 fortran iv 和 cobol 进行编程。后来他转向 c 和 c++,最近两年他一直在用 java 工作。他最感兴趣的 java 应用是广域网中的数据仓库和因特网上的并行处理和分布式处理(使用基于代理的编程)。reach arthur 的电子邮件地址为 arthur.choi@javaworld.com。

john mitchell,先后做过雇员、咨询人员,现在是自己公司的老板,过去十年他所投资的领域包括前沿计算机软件的开发、对其他开发人员提供建议和培训。他的咨询范围包括 java 技术、编译器、解释器、基于 web 应用和因特网商务等。john 是 making sense of java: a guide for managers and the rest of us 一书的作者之一,并在编程杂志上发表了许多文章。除了为 javaworld 撰写 java tips 专栏之外,他还主持着 comp.lang.tcl.announce 和 comp.binaries.geos 新闻组。reach john 的电子邮件地址为 john.mitchell@javaworld.com。

扫描二维码推送至手机访问。

版权声明:本文由Ant.Master's Blog发布,如需转载请注明出处。

本文链接:https://iant.work/post/422.html

标签: Java技术
分享给朋友:

“从JAR和zip档案文件中提取Java资源” 的相关文章

Socket与ServerSocket的问题

//服务器端:import java.io.BufferedReader;import java.io.IOException;import java.io.InputStreamReader;import java.io.PrintStream;import java.net.ServerSock...

JSP与Servlet的对应关系

以前在QQzone写下的文章现在贴到这里来了... 最近比较忙啊!现在抽身写一篇文章。是关于JSP与Servlet的对应关系的。希望对大家有所帮助。其实我也是刚刚学的......-------Servlet--------------JSP----------1.ServletContext&nbs...

IM技术(1)

    做项目了,NetCL今天开工了,在这些日子里,我会将自己研究的内容写下来。做个记录,以下是我在网上搜到的。关于管理用户状态的解决方案,当然,我都有一个方案。不过对客户端的任务有点重吧,我方法是客户端从服务器端获到一个用户在线状态后,接着就与服务器无关了。好友离线...

JAVA内部类终极实例

最近心情不好,不想说太多东西了!电脑坏了,我现在又病了. class ClassFactory{ private final static String userName = "Hello,My name is CAnca."; public static Thread in =...

过滤网页HTML标记

JAVA过滤HTML中的所有标记。非常好用!! package canca.regex; import java.util.regex.Matcher;import java.util.regex.Pattern; public class HtmlFilter {  priva...

字符,字节和编码

字符,字节和编码

转自:http://www.regexlab.com/zh/encoding.htm------------------------------------------------------------- 级别:中级 摘要:本文介绍了字符与编码的发展过程,相关概念的正确理解。举例说明了一些实际应...

发表评论

访客

◎欢迎参与讨论,请在这里发表您的看法和观点。