Android Studio 中合并jar包 - fatjar插件

jar.png

在开发中我们经常需要将某个模块以JAR包的形式输出,在打包的同时需要将它所依赖的多个JAR都整合到本JAR中,这样我们就可以直接使用“java -jar”命令执行它,而不用设置classpath或者部署所依赖的JAR,别的开发者也可以很方便的引用这个JAR。

在eclipse中有一个fatjar的插件可以很方便的完成这个需求,但是在android studio中我还没有发现这样的插件。想了下这类插件的原理也就是简单的文件读写,所以决定自己动手实现一个。

本文出处:http://www.jianshu.com/p/be46b3891d0e

JAR文件格式

动手写代码之前我们先了解下JAR文件格式。在软件领域,JAR文件(Java归档:Java ARchive)是一种软件包文件格式,通常用于聚合大量的Java类文件、相关的元数据(manifest文件)和资源(文本、图片等)文件到一个文件,便于分发的Java平台的应用软件或库。JAR文件是一种归档文件,以ZIP格式构建,以.jar为文件扩展名。用户可以使用JDK自带的jar命令创建或提取JAR文件。也可以使用其他zip压缩工具,不过压缩时zip文件头里的条目顺序很重要,因为Manifest资源配置文件常需放在首位。

jar_format.png

Manifest资源配置文件是JAR中包含的特殊文件,它被用来定义扩展或档案打包相关数据。Manifest文件中的条目定义这个JAR文件如何被使用。例如,类路径条目由其他JAR文件的绝对或相对路径的列表组成,用于指定在加载本JAR文件时同时加载的其他JAR文件。虽然旨在简化JAR的使用,但在实践中证明Manifest文件是非常脆弱的,因为入口点JAR在创建时依赖于所有相关的JAR的确切位置。一旦需要更改依赖库的位置,必需重建Manifest文件。本文实现的fatjarCreator工具,将所依赖的相关JAR都打包到本JAR中,可以避免这样的问题。

manifest.png

fatjarCreator工具的制作

因为jar文件实际是一个文件包,所以jar文件读写和普通的文件读写有一定的区别,我们需要先将jar包中的每个文件遍历出来再依次进行文件读写。幸运的是软件包 java.util.jar 提供了读写JAR 文件的类,如JarFile.class、JarOutputStream.class、Manifest.class、JarEntry.class等。这个合并工具本身也是用Java写的,完成后也是一个jar包,我将它命名为fatjarCreator.jar。核心逻辑如下:

    public void create(ArrayList<String>jarPaths, String output) {
        String jarPath = "";
        Manifest manifest = getManifest();   //获取outJar的manifest
        FileOutputStream fos = new FileOutputStream(output);
        JarOutputStream jos = new JarOutputStream(fos, manifest);  //根据输出路径和manifest创建输出流
        for (int i = 0; i < jarPaths.size(); i++) {
            jarPath = jarsPaths.get(i);
            JarFile jarFile = new JarFile(jarPath);
            Enumeration<?> entities = jarFile.entries();  
            while (entities.hasMoreElements()) {  
                JarEntry entry = (JarEntry) entities.nextElement();  //遍历jar文件中的每个文件节点

                //在写目录中的文件时,目录会自动创建
                //meta-inf使用我们上面获取的manifest,不使用之前的了meta-inf
                if (entry.isDirectory() || entry.getName().toLowerCase().startsWith("meta-inf")) {  
                    continue;  
                }  

                InputStream in = jarFile.getInputStream(entry);
                copyData2Jar(in, jos, entry.getName());
            }  

            jarFile.close();  
        }
    } 

    private void copyData2Jar(InputStream in, JarOutputStream jos, String newEntryName) {
        int bufferSize;  
        byte[] buffer = new byte[1024]; 
   
        jos.putNextEntry(new JarEntry(newEntryName));  //在outJar中的创建文件节点

        while ((bufferSize = in.read(buffer, 0, buffer.length)) != -1) {  
             jos.write(buffer, 0, bufferSize);  
        }  

        in.close();  
        jos.closeEntry();  
    }

    private Manifest getManifest() {  
        Manifest manifest = new Manifest();  
        Attributes attribute = manifest.getMainAttributes();  
        attribute.putValue("Manifest-Version", "1.0");  
        attribute.putValue("Created-By","fat jar plugin");  
        return manifest;  
    }  

同理我们也可以将一些资源(文本、图片等)文件打到JAR包中。这样合并JAR文件的方法我们就写好了,使用时只需要传入待合并的JAR文件的路径和输出文件的路径就可以了。通常需要合并的文件较多,如果使用命令行传入参数的方式,那么执行命令会很长,为了更方便的使用,我们可以用配置文件来传入文件路径。我这里使用了如下xml文件来作为配置文件,增删改查合并JAR文件时候只需要修改对应的配置文件即可。

<?xml version="1.0" encoding="UTF-8"?>
<files>
    <jars> 
       <jar path="D:\\codes\\libs\\base.jar"/>
       <jar path="D:\\codes\\libs\\cardboard.jar"/>
       <jar path="D:\\codes\\libs\\mAppTracker.jar"/>
       <jar path="D:\\codes\\libs\\mvvtracker.jar"/>
       <jar path="D:\\codes\\build\\intermediates\\bundles\\release\\classes.jar"/>
    </jars>
    <assets path="D:\\codes\\assets"/>
</files>

到这里我们就可以将实现的代码输出成一个可执行的fatjarCreator.jar。把它和上面个的配置文件fatjar_config.xml放在同一目录下,然后双击fatjarCreator.jar或者命令行执行“Java -jar fatjarCreator.jar”,我们都可以在我们设置的输出路径下看到合并后的JAR文件了。

配置gradle

我们的目的是想在Android Studio中更方便的合并JAR文件,现在合并的功能已经实现了,如何才能在AS中调用这个小工具呢?在module的build.gradle中创建一个名为createFatjar的task。然后我们可以在命令行中执行“gradle createFatjar”命令或者在AS右边的gradle栏中双击createFatjar来执行合并任务。需要把fatjarCreator.jar和fatjar_config.xml放在build.gradle所在目录。

task createFatjar (type: JavaExec) {
  javaexec { 
    main="-jar";        
    args=["fatjarCreator.jar"];
  }
}

代码

为了文章的简洁,文中的示例代码省略了很多,如入口方法,xml的解析等。想要获取完整项目代码的同学可以点击代码获取。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容