怎样制作自解压jar文件

本文详述如何创建自解压jar文件,通过编写Java解压程序和配置manifest文件,实现跨平台自解压功能。自解压jar文件允许在任意支持Java的平台上运行并解压其内容。步骤包括编写解压程序、创建manifest文件、打包jar文件,并提供了具体操作示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

这是一篇描述怎样制作自解压jar文件的文章,作者通过自己编写的一个自解压程序,并把这个自解压程序以及一个manifest文件一起加入原始的jar文件中,就制作出一个可以在各种支持java的平台上运行的自解压的jar   文件。   

自解压文件   
我们先来了解一下自解压文件,在window下可以用自解压制作工具如winzip   self-Extractor来制作自解压文件,这些工具会把一个zip文件与解压程序打包在一起而产生一个新的可执行文件。然后只要运行这个可执行文件,就可以把zip文件中的内容解开。那为什么要创建自解压jar文件呢,创建成自解压zip文件不就好了?我们应该注意到自解压jar文件可以在任意支持java的平台上解开并执行,例如,可以在linux下执行。创建jar自解压文件很简单,只需要一个特殊的JAR   manifest文件、一个基于java的解压程序(这个程序是原作者写的)、包含基本文件的jar   或者zip文件以及任何jsdk的jar应用程序   

manifest文件   
要生成可执行jar文件,需要在META-INF   目录下的manifest文件,文件名为:MANIFEST.MF   ,但在我们这里我们只需要在文件中指定在这个基于java   的解压程序中包含main()的类的名称:Main-Class:   ZipSelfExtractor   

我们已经把一个叫做jarmanifest的文件加入到这个技巧的源程序包中。   

解压程序   
你可以用各种方法来实现这个解压程序,在我们这里使用了一个简单直接的办法。首先,解压程序判断这个自解压jar文件的名称,有了这个文件名,解压程序使用解压标准,把文件解开。具体的可以查看在源码包中的ZipSelfExtractor.java文件。   

值得一提的是这里用了一个很巧妙的办法获取jar文件的文件名,虽然在命令行中出现的这个文件的名字,但它并没有作为参数传入类的main()中,因此,这里使用了以下的代码来获取文件名:   

private   String   getJarFileName   ()   
{   
myClassName   =   this.getClass().getName()   +   ".class ";   
URL   urlJar   =   
this.getClass().getClassLoader().getSystemResource(myClassName);   
String   urlStr   =   urlJar.toString();   
int   from   =   "jar:file: ".length();   
int   to   =   urlStr.indexOf( "!/ ");   
return   urlStr.substring(from,   to);   
}   

请注意:getSystemResource()   中使用了myClassName而不是ZipSelfExtractor.class作参数,这使得我们可以更改加压程序的名字而不需要修改代码。   

接下来,我们来分析获得这个jar文件的名字。首先,可以获取指向包含正在运行类的文件,urlStr   =   urlJar.toString();有了这个url,把jar文件名去掉,剩下的就是我们想要的,下面是这个url的格式:   
jar:file:/home/test/zipper.jar!/ZipSelfExtractor.class   

有了文件名,就可以开始解压,详细的解压算法请大家自己看源码。   

为了可以更方便实用,程序使用了图形界面,程序中使用了JFileChooser类可以选择要解压的目标目录。   

最后程序还确保不把这两个文件:manifest文件和extractor 's   .class(在我们这里是ZipSelfExtractor.class)文件也解出来,程序是用来解开原始的jar的内容,而这两个文件并属于jar原始内容。   

打包jar文件   
有了manifest文件与解压程序,我们就可以创建自解压jar文件了,以下是一个例子:   
1.创建一个zip文件Myzip.zip   
2.下载zipper.jar   
3.把文件解到当前目录,准备制作自解压jar文件   
java   -jar   zipper.jar   
4.把zipper.class拷贝成   ZipSelfExtractor.class   
5.把   myzip.zip   重命名为   myzip.jar   
6.把myzip.jar中的内容替换为jarmanifest和ZipSelfExtractor.class这两个文件   
jar   uvfm   myzip.jar   jarmanifest   ZipSelfExtractor.class   
7.执行java   -jar   myzip.jar就可以看到效果了,试试看   

后记   
一个自解压的jar文件能够很好的跨平台使用,自解压jar文件创建简单,只需要有jre1.2或或者更新的版本就可以实现了。   

附自解压程序的源代码:   

  
  
import java.io.*;
import java.net.*;
import javax.swing.*;
import java.util.zip.*;
import java.util.*;
import java.text.*;

public class ZipSelfExtractor extends JFrame {
    private static final long serialVersionUID = 1L;
    private String myClassName;
    static String MANIFEST = "META-INF/MANIFEST.MF";

    public static void main(String[] args) {
        ZipSelfExtractor zse = new ZipSelfExtractor();
        String jarFileName = zse.getJarFileName();
        zse.extract(jarFileName);
        System.exit(0);
    }

    ZipSelfExtractor() {
    }

    private String getJarFileName() {
        myClassName = this.getClass().getName() + ".class ";
        URL urlJar = this.getClass().getClassLoader().getSystemResource(myClassName);
        String urlStr = urlJar.toString();
        int from = "jar:file: ".length();
        int to = urlStr.indexOf("!/ ");
        return urlStr.substring(from, to);
    }

    public void extract(String zipfile) {
        File currentArchive = new File(zipfile);

        JFileChooser fc = new JFileChooser();

        fc.setCurrentDirectory(new File(". "));
        fc.setDialogType(JFileChooser.OPEN_DIALOG);
        fc
                .setDialogTitle("Select   destination   directory   for   extracting   "
                        + currentArchive.getName());
        fc.setMultiSelectionEnabled(false);

        fc.setFileSelectionMode(JFileChooser.DIRECTORIES_ONLY);

        if (fc.showDialog(ZipSelfExtractor.this, "Select ") != JFileChooser.APPROVE_OPTION) {
            return; // only when user select valid dir, it can return
            // approve_option
        }

        File outputDir = fc.getSelectedFile();

        byte[] buf = new byte[1024];
        SimpleDateFormat formatter = new SimpleDateFormat(
                "MM/dd/yyyy   hh:mma ", Locale.getDefault());

        ProgressMonitor pm = null;

        boolean overwrite = false;

        ZipFile zf = null;
        FileOutputStream out = null;
        InputStream in = null;

        try {
            zf = new ZipFile(currentArchive);

            int size = zf.size();
            int extracted = 0;
            pm = new ProgressMonitor(getParent(), "Extracting   files... ",
                    "starting ", 0, size - 4);
            pm.setMillisToDecideToPopup(0);
            pm.setMillisToPopup(0);

            Enumeration entries = zf.entries();

            for (int i = 0; i < size; i++) {
                ZipEntry entry = (ZipEntry) entries.nextElement();
                if (entry.isDirectory())
                    continue;

                String pathname = entry.getName();
                if (myClassName.equals(pathname) || MANIFEST.equals(pathname.toUpperCase()))
                    continue;

                extracted++;
                pm.setProgress(i);
                pm.setNote(pathname);
                if (pm.isCanceled())
                    return;

                in = zf.getInputStream(entry);

                File outFile = new File(outputDir, pathname);
                Date archiveTime = new Date(entry.getTime());

                if (overwrite == false) {
                    if (outFile.exists()) {
                        Object[] options = { "Yes ", "Yes   To   All ", "No " };
                        Date existTime = new Date(outFile.lastModified());
                        Long archiveLen = new Long(entry.getSize());

                        String msg = "File name conflict: "
                                + "There is already a file with "
                                + "that name on the disk!\n "
                                + "\nFile name: "
                                + outFile.getName()
                                + "\nExisting file: "
                                + formatter.format(existTime)
                                + ", "
                                + outFile.length()
                                + "Bytes "
                                + "\nFile in archive: "
                                + formatter.format(archiveTime)
                                + ", "
                                + archiveLen
                                + "Bytes "
                                + "\n\nWould you like to overwrite the file? ";

                        int result = JOptionPane.showOptionDialog(
                                ZipSelfExtractor.this, msg, "Warning ",
                                JOptionPane.DEFAULT_OPTION,
                                JOptionPane.WARNING_MESSAGE, null, options,
                                options[0]);

                        if (result == 2) // No
                        {
                            continue;
                        } else if (result == 1) // YesToAll
                        {
                            overwrite = true;
                        }
                    }
                }

                File parent = new File(outFile.getParent());
                if (parent != null && !parent.exists()) {
                    parent.mkdirs();
                }

                out = new FileOutputStream(outFile);

                while (true) {
                    int nRead = in.read(buf, 0, buf.length);
                    if (nRead <= 0)
                        break;
                    out.write(buf, 0, nRead);
                }

                out.close();
                outFile.setLastModified(archiveTime.getTime());
            }

            pm.close();
            zf.close();
            getToolkit().beep();

            JOptionPane.showMessageDialog(ZipSelfExtractor.this, "Extracted   "
                    + extracted + "   file " + ((extracted > 1) ? "s " : " ")
                    + "   from   the\n " + zipfile
                    + "\narchive   into   the\n " + outputDir.getPath()
                    + "\ndirectory. ", "Zip   Self   Extractor ",
                    JOptionPane.INFORMATION_MESSAGE);

        } catch (Exception e) {
            System.out.println(e);
            if (zf != null) {
                try {
                    zf.close();
                } catch (IOException ioe) {
                    ;
                }
            }
            if (out != null) {
                try {
                    out.close();
                } catch (IOException ioe) {
                    ;
                }
            }
            if (in != null) {
                try {
                    in.close();
                } catch (IOException ioe) {
                    ;
                }
            }
        }
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值