通常我们把.class文件打包,一般手动把bin下的打成zip形式,然后该文件名为*.jar。大家有没有想过写个程序打jar包呢??最近我的工程需要这样做,我写了一段程序,用代码打jar包,打包的结果跟手动打jar包后的样子是一模一样的,就是文件大小不一样,手动要大一点,自动的要小一点。jar包里面什么东西都一样。

但是诡异的问题出现了,在我的工程中,在运行是要用到这个jar包,但是类加载器却加载不到jar包里面的.class文件,如果手动打包扔进去,就可以加载。我和郁闷,折磨我好几天了,希望借助这个平台,能够得到大家的帮助。谢谢。
前面java2000_net 已经提出了一些问题,感谢他,但是好像还不行。我的和java2000_net 代码如下:/** 
  *   功能:文件夹的压缩打包(jar) 
  *   
  *   @param   inputFileName 
  *                         被压缩或者打包的文件夹   outputFileName   压缩后或者打包的文件 
  */ package test;import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.util.jar.JarEntry;
import java.util.jar.JarOutputStream;public class Test {
  public static void jar(String inputFileName, String outputFileName) throws Exception {
    JarOutputStream out = new JarOutputStream(new FileOutputStream(outputFileName));
    File f = new File(inputFileName);// 这里应该以那个目录为根目录,jar的根应该是这个目录下的文件
    jar(out, f, "");
    out.close();
  }  private static void jar(JarOutputStream out, File f, String base) throws Exception {
    if (f.isDirectory()) {
      File[] fl = f.listFiles();
      base = base.length() == 0 ? "" : base + "/";
      for (int i = 0; i < fl.length; i++) {
        jar(out, fl[i], base + fl[i].getName());
      }
    } else {
      out.putNextEntry(new JarEntry(base));
      FileInputStream in = new FileInputStream(f);
      byte[] buffer = new byte[1024];
      int n = in.read(buffer);
      while (n != -1) {
        out.write(buffer, 0, n);
        n = in.read(buffer);
      }
      in.close();
    }
  }  public static void main(String[] args) {
    try {
      jar("e:\\workplace\\test\\bin", "d://test.jar");
    } catch (Exception e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
  }
}bin下就是包结构,以及.class文件

解决方案 »

  1.   

    打包成jar。。需要在jar里面生成一個配置文件的啊,應該是:
    META-INF文件夹和里面的MANIFEST.MF文件
      

  2.   

    MANIFEST.MF文件里面寫的是main方法所在類,以及要加載的包
      

  3.   

    MANIFEST.MF文件里面記錄的是,main方法所在類,以及需要加載的jar包
      

  4.   

    参考以下文章
    ---------------------------------------------------------------------------
    ar文件听说过吗,没有?或者陌生!好,没关系,这就是我们的第一站:打包发布。
    为什么会有这个玩意呢,首先,这是jar的全称:JavaTM Archive (JAR) file,是的,就是java存档文件。这有点类似zip文件,想一想它是干什么的用的呢,压缩!?没错就是要压缩,将我们原先零散的东西放到一下,重新组织,所有这些目的只有一个:方便!好了,不用管他是怎么压缩的,我们的重点是哪些是我们要压缩的(输入),还有压缩成了什么(输出),进而将它发布(部署)。
    那我们的输入(要压缩的东西)主要是class文件,还有辅助的资源(这其中可能有图片,jsp文件,html文件等等)。Jar技术在 jdk1.1版本中就已存在,在1.2中又有了增强。接下来说说jar的好处吧,这是官方的描述:安全,快速下载,压缩,猎取包,版本化包,可携。
    说了这么多,我们现在开始实施。
    先打开命令提示符(win2000或在运行筐里执行cmd命令,win98为DOS提示符),输入jar –help,然后回车(如果你盘上已经有了jdk1.1或以上版本),看到什么:
    用法:jar {ctxu}[vfm0Mi] [jar-文件] [manifest-文件] [-C 目录] 文件名 ...
    选项:
    -c 创建新的存档
    -t 列出存档内容的列表
    -x 展开存档中的命名的(或所有的〕文件
    -u 更新已存在的存档
    -v 生成详细输出到标准输出上
    -f 指定存档文件名
    -m 包含来自标明文件的标明信息
    -0 只存储方式;未用ZIP压缩格式
    -M 不产生所有项的清单(manifest〕文件
    -i 为指定的jar文件产生索引信息
    -C 改变到指定的目录,并且包含下列文件:
    如果一个文件名是一个目录,它将被递归处理。
    清单(manifest〕文件名和存档文件名都需要被指定,按'm' 和 'f'标志指定的相同顺序。示例1:将两个class文件存档到一个名为 'classes.jar' 的存档文件中:
    jar cvf classes.jar Foo.class Bar.class
    示例2:用一个存在的清单(manifest)文件 'mymanifest' 将 foo/ 目录下的所有
    文件存档到一个名为 'classes.jar' 的存档文件中:
    jar cvfm classes.jar mymanifest -C foo/ .来个小例子试试看:
    我们只有一个HelloWorld,如下:
    public class HelloWorld{
    public static void main(String[] args){
    System.out.println(“Hi, Hello World!”);
    }
    }
    我将这个java文件存到C盘跟目录下,ok,接下来,
    在先前打开的命令提示符下(跳转到C盘提示符下),我们输入javac HelloWorld.java,然后继续输入:jar cvf hello.jar HelloWorld.class,回车后去你的C盘看看,多了什么,没错 hello.jar 。
    基本的步骤我们现在都知道了,你可以自己去尝试一下随着jar后面的参数的不同,结果有什么变化。
    紧接着我们看看如何运行我们的jar包。
    在进入正题之前,你要先打开我们刚刚做好的jar包看看,多了什么呢,META-INF目录?再看看里面是什么,还有一个MANIFEST.MF文件是不是?用文本编辑器(我这里是UltraEdit)打开它看看:
    Manifest-Version: 1.0
    Created-By: 1.4.2 (Sun Microsystems Inc.)
    就是这样。这里我们对它进行修改,加一句:Main-Class: HelloWorld (在第三行)。这个就是我们之前写的那个类,也就是我们的入口类。也即,
    Manifest-Version: 1.0
    Created-By: 1.4.2 (Sun Microsystems Inc.)
    Main-Class: HelloWorld
    接下来,我们在命令提示符里执行:
    jar umf MANIFEST.MF app.jar
    这样我们使用了我们自己的MANIFEST.MF文件对原来默认的进行了更新。你不妨可以再进去看看是不是添上了Main-Class: HelloWorld这一句。
    Ok,这个最后的一步了,来验证我们做的一切,在命令提示符中输入:
    java -jar hello.jar(执行)
    出现了什么,――Hi, Hello World!
    我们再来看看jar文件在tomcat中发布,注意:在tomcat中我们就不能再用jar这种格式,而改war格式,它是专门用于web应用的,其实整个过程下来基本上和jar是类似的:
    先准备我们要打包的资源。
    找到存放tomcat的webapps目录,进到其中,新建一个文件夹,这里命名为hello,再进去新建WEB-INF文件夹,再进去新建classes 文件夹,此时我们也将我们唯一的servlet,HelloWorld.java放到这里,在与classes目录同级下建立一文件web.xml。 Ok,目前我们初步建立了一个简单的web应用。
    这是HelloWorld.java:
    import java.io.*;
    import javax.servlet.*;
    import javax.servlet.http.*;
    public class HelloWorld extends HttpServlet {
    public void doGet(HttpServletRequest req, HttpServletResponse res)
    throws ServletException, IOException {
    res.setContentType("text/html");
    PrintWriter out = res.getWriter();
    out.println("<HTML>");
    out.println("<HEAD><TITLE>Hello, World!</TITLE></HEAD>");
    out.println("<BODY>");
    out.println("Hello, World!");
    out.println("</BODY></HTML>");
    }
    }//end here!
    对它编译。下面是web.xml:
    <?xml version="1.0" encoding="ISO-8859-1"?>
    <!DOCTYPE web-app PUBLIC
    '-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN'
    'http://java.sun.com/j2ee/dtds/web-app_2_3.dtd'>
    <web-app>
    <servlet>
    <servlet-name>hello</servlet-name>
    <servlet-class>HelloWorld</servlet-class>
    </servlet>
    <servlet-mapping>
    <servlet-name>hello</servlet-name>
    <url-pattern>/HelloWorld</url-pattern>
    </servlet-mapping>
    </web-app>
    开始压缩,形成war档:
    在命令提示符下进到先前创制的hello目录下,执行 jar cvf hello.war * ,我们便得到hello.war。将它拷贝至webapps目录下,ok,来看最后一步,打开tomcat的目录conf中的server.xml,加入:
    <Context path="/hello" docBase="hello.war" debug="0"
    reloadable="true"/>
    大功告成!运行它,启动tomcat,后在浏览器中输入http://localhost:8080/hello/HelloWorld,有了吗?
    最后,如果你想用ant来完成以上的打包活动,下面就告诉你:
    对于jar来说。在build.xml中,
    <target name="jar">
    <jar destfile="${app_home}/hello.jar">
    <fileset dir="${dest}" includes="**"/>
    <!--fileset dir="${dest}" includes="**/action.properties"/-->
    </jar>
    </target>对于war,
    <war warfile="hello.war" webxml="./WEB-INF/web.xml">
    <fileset dir="html"/>
    <lib dir="lib/">
    <exclude name="oracle*.jar"/>
    </lib>
    <classes dir="build/servlets">
    <include name="**/*.class"/>
    </classes>
    </war>
    好了,就这么多,希望对你有点帮助。:)
    我上传了上面打过的两个包,hello.jar和hello.war。 『 点击下载 』
    『 点击下载 』
    第一rar文件对应的是hello.jar,下载后将其名改为hello.jar
    第二rar文件对应hello.war,下载后改为hello.war。
    这是由于上传不了jar格式和war格式的文件,你只好照我上面说的去做了 :)补充:
    ############
    jar基本操作:
    ############
    1. 创建jar文件
    jar cf jar-file input-file(s)
    c---want to Create a JAR file.
    f---want the output to go to a file rather than to stdout.
    eg: 1)jar cf myjar.jar query_maintain_insert.htm
    2)jar cvf myjar.jar query_maintain_insert.htm
    v---Produces verbose(详细的) output.
    3)jar cvf myjar.jar query_maintain_insert.htm mydirectory
    4)jar cv0f myjar.jar query_maintain_insert.htm mydirectory
    0---don't want the JAR file to be compressed.
    5)jar cmf MANIFEST.MF myjar.jar yahh.txt
    m---Used to include manifest information from an existing manifest file.
    6)jar cMf MANIFEST.MF myjar.jar yahh.txt
    M---the default manifest file should not be produced.
    7)jar cvf myjar.jar *
    *---create all contents in current directory.
    2. 察看jar文件
    jar tf jar-file
    t---want to view the Table of contents of the JAR file.
    eg: 1)jar vft yahh.jar
    v---Produces verbose(详细的) output.
    3. 提取jar文件
    jar xf jar-file [archived-file(s)]
    x---want to extract files from the JAR archive.
    eg: 1)jar xf yahh.jar yahh.txt(仅提取文件yahh.txt)
    2)jar xf yahh.jar alex/yahhalex.txt(仅提取目录alex下的文件yahhalex.txt)
    3)jar xf yahh.jar(提取该jar包中的所有文件或目录)
    4. 修改Manifest文件
    jar cmf manifest-addition jar-file input-file(s)
    m---Used to include manifest information from an existing manifest file.
    5. 更新jar文件
    jar uf jar-file input-file(s)
    u---want to update an existing JAR file. 
      

  5.   

    那我把META-INF/MANIFEST.MF文件也放到那里面,一起打成jar包,结构能出来,但是不行的,我以前就这样试过的
      

  6.   

    大家可以把这段代码拷过去一试啊,真的,随便找个工程的bin目录,在加个META-INF/MANIFEST.MF,用刚才这个代码打个jar包看看,保证郁闷
      

  7.   

    哇,这个问题我已经解决了,你还有一个帖子。看这里吧!你可能根本没注意我调整了哪里,这个版本我注明了!!package test; import java.io.File; 
    import java.io.FileInputStream; 
    import java.io.FileOutputStream; 
    import java.util.jar.JarEntry; 
    import java.util.jar.JarOutputStream; public class Test { 
      public static void jar(String inputFileName, String outputFileName) throws Exception { 
        JarOutputStream out = new JarOutputStream(new FileOutputStream(outputFileName)); 
        File f = new File(inputFileName); 
        jar(out, f, ""); 
        out.close(); 
      }   private static void jar(JarOutputStream out, File f, String base) throws Exception { 
        if (f.isDirectory()) { 
          File[] fl = f.listFiles(); 
          base = base.length() == 0 ? "" : base + "/"; // 注意,这里用左斜杠 
          for (int i = 0; i < fl.length; i++) { 
            jar(out, fl[ i], base + fl[ i].getName()); 
          } 
        } else { 
          out.putNextEntry(new JarEntry(base)); 
          FileInputStream in = new FileInputStream(f); 
          byte[] buffer = new byte[1024]; 
          int n = in.read(buffer); 
          while (n != -1) { 
            out.write(buffer, 0, n); 
            n = in.read(buffer); 
          } 
          in.close(); 
        } 
      }   public static void main(String[] args) { 
        try { 
          jar("e:\\workplace\\test", "d://test.jar"); 
        } catch (Exception e) { 
          // TODO Auto-generated catch block 
          e.printStackTrace(); 
        } 
      } 
    }
      

  8.   

    我晕,你已经贴了! SORRY! 我再次确认能否运行。我昨天测试能运行的!
      

  9.   

    MANIFEST.MF是一个特殊的文件,并不能手工把目录加上去的,需要使用
    java.util.jar.Manifest读取Manifest文件,再使用:
      JarOutputStream(OutputStream out, Manifest man)
    来构造的。
      

  10.   

    我再次确认,那个jar能运行,而且是用这个程序打包的!java -cp test.jar test.Test你是不是没有加上那个路径啊!
      

  11.   

    我替搂主声明一下,不是讨论能运行得 jar 问题,而是用这个方法打包的jar根本不能用, 但是他用winzip打开竟然没有问题。哈哈!
      

  12.   

    我运行了一下,也添加了META-INF和加载类库了,都是可以的啊。
      

  13.   

    是的,你把那个"/" 改成 "\" 就不行,要不你测试看看!base = base.length() == 0 ? "" : base + "\\";
      

  14.   

    谢谢啊,我发现问题所在了,是这个原因啊:是我多此一举啊,我在bin目录下手工添加这样的目录层次:META-INF/MANIFEST.MF
    (不能这样做啊)再进行打打包是不行的(手动打可以)MANIFEST.MF是一个特殊的文件,并不能手工把目录加上去的。如果你在写程序中需要一个这样的MANIFEST.MF,需要在上面代码中 
    JarOutputStream out = new JarOutputStream(new FileOutputStream(outputFileName),new Manifest())以上这个程序是可以用的
      

  15.   

    我在源代码根目录加了META-INF/MANIFEST.MF
    文件内容为空 
    用这个程序打包,生成的jar依然能用。
      

  16.   

    不是很明白楼主到底想要表示什么?
    为什么不用jar命令,而去偏偏用自己的方法去写呢?如果纯粹是为了学习java sdk也情有可原。
      

  17.   

    嗯,重载一下jar方法public static void jar(String inputFileName, String outputFileName,
            String manifestFile) throws Exception {
        Manifest manifest = new Manifest(new FileInputStream(manifestFile));
        JarOutputStream out = new JarOutputStream(new FileOutputStream(
                outputFileName), manifest);
        File f = new File(inputFileName);
        jar(out, f, "");
        out.close();
    }main调用时改为:jar("f:/aaa", "f:/test.jar", "f:/abc.mf");
    就可以了,abc.mf是清单文件应符合MANIFEST.MF文件规范,
    至少应有:Manifest-Version: 1.0
    也可以加上Main-Class: xxx.xxx.Xxx
    最后至少应用两个空行。希望这些能对您有用。
      

  18.   

    MF文件中冒号后面要有且只能有一个空格,注意一下。该死的CSDN,CSDN发帖时把它替换成三个空格了!
      

  19.   

    http://www.mdzlkj.cn/
    里面的经典代码有个连接有视频示范
      

  20.   

    用  myeclipse 自动打包