Maven系列——构建我的Android项目

本博客为个人原创,转载需在明显位置注明出处

在正式开始本篇文章之前,你需要做一些准备工作,他们是:

  1. android-maven-plugin

  2. android-sdk-deployer

关于android-maven-plugin:我们知道Maven只是一座工厂,真正工作的是plugin,由于Android不同于Java,需要构建aar或者apk包,这就是android-maven-plugin存在的意义。虽说让你准备,并不是说让你预先安装环境什么的,而是需要你先弄清楚android-maven-plugin是干嘛的,再读一读官方文档中的使用引导,待真正使用的时候才不会感觉太陌生而措手不及

关于android-sdk-deployer:在编译项目时需要在pom.xml中配置项目所有的依赖,例如junit、gson、fresco等等。对于一个Android项目,Android SDK的依赖是必须的。我们在Android Studio中编译项目依赖的是我们本机中下载好的SDK,但是用Maven构建时,所有的dependency都会从本地仓库中依赖,如果本地仓库没有再从远程仓库中下载,而这恰恰就是问题,因为Maven中央仓库中Android SDK的最高版本仅为4.1.1(不太清楚Google为什么没有继续上传新版本SDK),现在远远达不到我们编译代码的要求,所以我们就得自己想办法。android-sdk-deployer可以帮助我们将你本机安装的Android SDK整理打包,放入本地Maven仓库,这样我们就可以在项目中依赖任何版本的SDK了。

好了,有了上面两项准备,我们就可以正式开始构建自己的Android项目,为此我创建了一个名为Bomb的工程(点我查看),其中有两个module,一个是app,另一个是bombframe,我们的目标就是构建bombframe,将打好的包上传至远程仓库,并可以在另一个工程里面成功依赖。首先我们来看一下工程目录:

Paste_Image.png

咦,为什么有两个pom文件?还记得之前我说过,pom文件也是有继承关系的,在实际项目中,我们一个工程下会有多个module,他们肯定需要公共的插件,公共的依赖,这就是为什么有两个pom文件的原因。工程根目录下的pom文件为父pom,其中定义的是公共的plugin和dependency,而bombframe中的pom为子pom,定义的则是其自身的一些特殊配置,下面我们先从父pom开始介绍,由于配置较多,我分段展示讲解:

    <groupId>com.will4it</groupId>
    <artifactId>bomb-frame-parent</artifactId>
    <version>1.0.0-SNAPSHOT</version>
    <packaging>pom</packaging>
    <name>bomb-frame-parent</name>

    <modules>
        <module>bombframe</module>
    </modules>

父pom不真正提供构建,它有两个作用,一是将一些公共的配置集中存放,二是mvn命令执行的入口,所以packaging不是具体的打包类型,而是pom,你只需记住所有的父pom文件的packaging都是pom就行。另外就是modules,大家应该可以看的明白,这里面定义的是所有的子项目,将需要构建的所有module子项目配置上即可。(这里配置的子项目名是AS工程的module名,跟子pom文件里面的name没关系哈,至少一开始我是这么错误的认为的,结果走了弯路)

<properties>
    <android.maven.plugin.version>4.4.3</android.maven.plugin.version>
    <compiler.version>3.1</compiler.version>
    <javadoc.version>2.9.1</javadoc.version>
    <android.sdk.path>/Users/mowei/Documents/Software/adt-bundle-mac-x86_64-20140321/sdk
    </android.sdk.path>
    <android.sdk.version>21</android.sdk.version>
</properties>

properties我就不多说了,大家复制粘贴就行,将定义的值修改成适合你项目的就行

<dependencyManagement>
    <dependencies>
        <dependency>
            <groupId>android</groupId>
            <artifactId>android</artifactI
            <version>5.1.1_r2</version>
            <scope>provided</scope>
        </dependency>
    </dependencies>
</dependencyManagement>

这里要强调一下,作为父pom,在定义公共dependency的时候,最外层需要加上一层dependencyManagement,没有为什么记住就行。另外这里只定义了一个dependency,就是我们的Android SDK,是通过上述android-sdk-deployer生成的。细心的童鞋应该可以发现groupId和version有点不太对劲,是的,我估计android-sdk-deployer的作者是为了跟Maven中央仓库的SDK做个区分。

    <build>
        <plugins>
            <plugin>
                <groupId>com.simpligility.maven.plugins</groupId>
                <artifactId>android-maven-plugin</artifactId>
                <version>${android.maven.plugin.version}</version>
                <extensions>true</extensions>
                <configuration>
                    <failOnNonStandardStructure>false</failOnNonStandardStructure>
                    <sdk>
                        <path>${android.sdk.path}</path>
                        <platform>${android.sdk.version}</platform>
                    </sdk>
                </configuration>
            </plugin>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <version>${compiler.version}</version>
                <configuration>
                    <source>1.6</source>
                    <target>1.6</target>
                    <encoding>utf-8</encoding>
                </configuration>
            </plugin>
        </plugins>
    </build>

接着就是构建这个工程需要的公共plugin,第一个就是让大家准备的android-maven-plugin,需要解释的是configuration标签,这里面是针对plugin做一些配置的更改。failOnNonStandardStructure的意思是在发现工程结构有错误的情况下不终止不报错,可以翻看一下Maven系列第一篇文章,一开始我就介绍了Maven构建对工程目录的要求,里面有一个resource目录,在我们Android Studio工程中默认是不存在的,但是Maven会严格检查工程结构,一旦不是标准结构就会终止构建并且报错,failOnNonStandardStructure就是为了跳过这一项。第二个是java compiler plugin,没有什么难理解的东西,就是配置JDK的版本和编码格式。

<distributionManagement>
    <repository>
        <id>releases</id>
        <url>你的远程release仓库地址</url>
    </repository>
    <snapshotRepository>
        <id>snapshots</id>
        <url>你的远程snapshot仓库地址</url       >
    </snapshotRepository>
</distributionManagement>

最后一段配置就是你的远程仓库的地址,光看配置没什么难理解的,之前给大家解释过snapshot和release的区别,就连仓库地址也是区分开的,这就是专业、标准。另外,上传到远程仓库是需要用户名和密码的,这个就需要/.m2/settings.xml文件来管理了,在我看来这个问题比较简单,咨询一下度娘你可以搞的定,如果你搞不定,可以给我评论,最后我统计一下人数,如果人数比较多,我会再写一篇文章专门讲解settings文件的配置使用。

以上就是父pom文件的配置,虽然有点长,但是分析下来还是不难的,只要你够耐心够细心,这些都不是问题。下面我们继续看子pom:

<groupId>com.will4it</groupId>
<artifactId>BombFrame</artifactId>
<version>1.0.0-SNAPSHOT</version>
<packaging>aar</packaging>
<name>BombFrame</name>

<parent>
    <groupId>com.will4it</groupId>
    <artifactId>bomb-frame-parent</artifactId>
    <version>1.0.0-SNAPSHOT</version>
    <relativePath>../pom.xml</relativePath>
</parent>

在子pom文件中,我们就可以定义packaging为对应的打包类型,这里我们需要构建的是aar包。另外,虽然pom也是继承关系,但是这种继承关系并不像Java代码那么智能,需要手动制定自己的parent是谁,还需要制定父pom文件的路径,这里大家注意一下即可

<properties>
    <keystore>bomb_keystore.jks</keystore>
    <key.store.password>bomb123456</key.store.password>
    <key.alias>bomb</key.alias>
    <key.alias.password>bomb123456</key.alias.password>
</properties>

依然只是properties的定义,这里基本上都是用于签名的keystore信息,下面jarsigner plugin里面需要

<dependencies>
    <dependency>
        <groupId>android</groupId>
        <artifactId>android</artifactId>
        <scope>provided</scope>
    </dependency>
</dependencies>

这里还需注意,虽然父pom文件中已经定义了Android SDK的依赖,但是子pom中还得重新指定一次,只是version不需要指定,对于dependency,父pom定义的作用主要是统一使用的version,所以子pom中还得再指定一次,如果version不指定,默认就是使用父pom中的版本,如果指定了就使用指定的版本。

<build>
    <plugins>
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-jarsigner-plugin</artifactId>
            <version>1.4</version>
            <executions>
                <execution>
                    <id>signing</id>
                    <goals>
                        <goal>sign</goal>
                    </goals>
                    <phase>package</phase>
                    <inherited>true</inherited>
                    <configuration>
                        <includes>
                            <include>target/*.aar</include>
                        </includes>
                        <keystore>${keystore}</keystore>
                        <storepass>${key.store.password}</storepass>
                        <keypass>${key.alias.password}</keypass>
                        <alias>${key.alias}</alias>
                        <arguments>
                            <argument>-sigalg</argument>
                            <argument>MD5withRSA</argument>
                            <argument>-digestalg</argument>
                            <argument>SHA1</argument>
                        </arguments>
                    </configuration>
                </execution>
            </executions>
        </plugin>
    </plugins>
</build>

对于plugin就不需要重复指定了,除非是该module独有的plugin,就像这里的jarsigner plugin一样,在打包成功之后,我们需要对aar包进行签名,这里配置的都是一些签名相关信息。

ok,到这里,我们的配置就结束了,在真正执行命令之前,再给大家详细解释一下以下几个常用命令的含义:

mvn package:在需要打包的项目或者module的pom文件相同目录下,创建一个target文件夹,然后执行编译构建打包操作,最后将打好的包等所有相关文件都存入target文件夹中

mvn clean:参考package命令,clean的作用就是删除整个target文件夹

mvn install:在package的基础上,再将打好的包存入本地的Maven仓库

mvn deploy:在install的基础上,将打好的包和pom文件等上传至远程仓库中

清楚这些命令的作用之后,我们就一起来试试看:

~$mvn clean

~$mvn package

如果你的配置没有问题,你会看到:

mvn clean
mvn package

查看一下module根目录,多了一个target文件夹,里面装着Maven构建好的文件,要啥有啥:

target

接着继续敲命令:

~$mvn install

运行结果就不贴了哈,运行成功以后,咱们来看看打好的包有没有装进Maven本地仓库:

local repository

特别注意我标记的三个红框哈,从左到右他们分别是groupId,artifactId和version

最后上传到远程仓库去,还记得第一篇文章的最后让大家先注册好Maven中央仓库的账号嘛,这时候就派上用场了

~$mvn deploy

上传成功,我们另外新建一个AS工程,注意这里是新工程而不是新的module哈,然后在新工程gradle中compile 'com.will4it:bombframe:1.0.0-SNAPSHOT'来依赖看看:

gradle config

如图,在依赖的时候你需要做两项配置,一个是配置maven仓库的地址,就是图中repository中的url,为什么?因为gradle默认是从jcenter中寻找依赖包的,上面我们打的包是放在Maven中央仓库或者公司私服仓库的,所以你需要告诉gradle还需要访问哪些仓库地址,这样gradle才能帮你找到;另一个就是compile 'groupId:artifactId:version'了,不解释。来看看结果吧

gradle result

至此,Maven系列文章就结束了,希望我的分享能够帮助到你,一定要动手自己操作。时间有限能力一般,文章中可能有些问题我没说清楚或者理解错的,欢迎大家踊跃拍砖!

注:可能你还会有疑问,我们的代码是需要混淆的,Maven好像是支持javadoc的生成的?经过这四篇文章的学习练习,相信你应该有这个能力自己去摸索

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,988评论 25 707
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,647评论 18 139
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,799评论 6 342
  • 真宫寺是清睁开眼睛。 眼前是人类。 领头的人挑起了他的下巴,摘掉口罩,眼神里有淋漓尽致的本性。是区别于好品质的更伟...
    二慝阅读 6,383评论 2 6
  • 本来不想写了,此刻吃完汤圆,家里灯火通明,每个人的眼睛都透着异样兴奋的目光。 我是累了,他们没累。 突然想说说今天...
    瓶子時光阅读 293评论 2 2