一、
眼下Android的开发正如火如荼,开发方式大多数是Android SDK+Eclipse+ADT,大部分的操作都要依赖Eclipse(当然有高手直接敲命令的)。如果项目依赖其它库资源在不同的机器上难免又要设置一番,而且库资源及其依赖管理是相当混乱,针对这种情况,Maven是一个很好的解决方案,下面讲叙述如何将Android的工程与Maven整合。
首先介绍一下需要用到的工具:
Android SDK——这个不用说了,必须的
Eclipse——同上
Android Development Tool(ADT)——相信绝大部分的人都依赖这个插件来开发Android项目的
以上是搭建一个Android项目的基本需要,下面是整合Maven需要的一些工具(插件)
Maven Eclipse Plugin(m2e)——用过Maven的一般都熟悉这个,这个是一个Eclipse的插件,其具有如下功能:
- 识别工程内的pom.xml,并使用独有的编辑器打开pom.xml
- 将pom.xml内部声明的依赖库添加至“Maven Dependencies”的Library Container,这个是使我们能在EclipseIDE内使用pom.xml内声明的库的主要原因
- 在java builder内加入maven的builder,使从eclipse内执行时能够使用到maven相关功能
- 增加maven的各种独有的运行设置
- 安装好如上的插件,在Android的工程上点右键,弹出菜单Configure->Convert to Maven project
- 配置好项目信息后会生成pom.xml以及工程上的图标已经多出一个小M字
- 更改project的packaging类型为apk
- <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
- <modelVersion>4.0.0</modelVersion>
- <groupId>org.cst.ming.android.sample</groupId>
- <artifactId>sample</artifactId>
- <version>0.0.1-SNAPSHOT</version>
- <packaging>apk</packaging>
- ......
- 在pom内声明一个build的插件,在maven构建项目的时候就会在target内生成一个apk的文件
- <plugin>
- <groupId>com.jayway.maven.plugins.android.generation2</groupId>
- <artifactId>android-maven-plugin</artifactId>
- <version>3.0.0</version>
- <inherited>true</inherited>
- <extensions>true</extensions>
- <configuration>
- <sdk>
- <platform>7</platform>
- <path>${env.ANDROID_SDK_HOME}</path>
- </sdk>
- <deleteConflictingFiles>true</deleteConflictingFiles>
- <undeployBeforeDeploy>true</undeployBeforeDeploy>
- </configuration>
- </plugin>
- 这时候可以在pom.xml内添加各种的依赖,并且执行mvn clean package即可构建出一个apk文件在target
- 此时已经整合完毕,可以在IDE内通过ADT来启动项目,亦可以通过在pom.xml内配置android-maven-plugin通过mvn android:deploy将项目部署到指定的设备(虚拟机)上,详情可以查阅android-maven-plugin的官方文档
de.akquinet.android.archetypes:android-quickstart:1.0.6——用于快速构建一个android-maven项目
刚才正在配置一个项目的报告输出,在配置pom的时候按照maven2的做法在<reporting>节点下加上了pmd、checkstyle等plugin
- <reporting>
- <plugins>
- <plugin>
- <groupId>org.apache.maven.plugins</groupId>
- <artifactId>maven-checkstyle-plugin</artifactId>
- <version>${checkstyle.plugin.version}</version>
- </plugin>
- <plugin>
- <groupId>org.apache.maven.plugins</groupId>
- <artifactId>maven-pmd-plugin</artifactId>
- <version>${pmd.plugin.version}</version>
- </plugin>
- </reporting>
却发现在target/site目录下除了image及css文件夹之外,一个html也没有,而且预期的报告也没有生成,我就纳闷了
几经Google后,终于查到,原来maven-site-plugin在3.0版本采用了一种全新的配置形式,导致不能接受<reporting>节点下声明的报告插件,这一点在其官方网站上有介绍:http://maven.apache.org/plugins/maven-site-plugin/maven-3.html#Configuration_formats
解决办法就是,将<reporting>内的plugin声明,转换为如下形式:
- <plugin>
- <groupId>org.apache.maven.plugins</groupId>
- <artifactId>maven-site-plugin</artifactId>
- <version>3.0</version>
- <configuration>
- <reportPlugins>
- <plugin>
- <groupId>org.apache.maven.plugins</groupId>
- <artifactId>maven-checkstyle-plugin</artifactId>
- </plugin>
- <plugin>
- <groupId>org.apache.maven.plugins</groupId>
- <artifactId>maven-pmd-plugin</artifactId>
- </plugin>
- </reportPlugins>
- </configuration>
- </plugin>
还有一点就是,在3.0版本中,在maven-site-plugin内声明的插件是可以忽略version的,maven-site-plugin会按照如下顺序查找:
1. build.plugins
2. build.pluginsManagement
3.repository(在repository中查找可能会使用Snapshot的插件,和maven插件一样规则)
至此,改动已经完成,运行mvn clean site即可看见project-reports.html文件,打开所期望的报告页面了。
刚开始时候,很难明白maven中packaging、phase和goal之间的关系,后来搞清楚了,项目搭出一个样子的时候,又来一个新的问题——报告
maven在pom model 4.0定义中,特别为报告开辟了一个与build同级的专用节点——<reporting>,在reporting内可以声明一个plugins然后写上各种报告插件,(在Maven2.x中)运行mvn site即可生成一个很完整的项目报告。这个是怎么做到的呢,下面来一探究竟。
1. 首先,看mvn site这个命令做了什么?明显执行了site这个phase,但是phase是不能执行的,执行的是插件,那么就看看是什么插件绑定到site这个phase上。通过Effective POM看到,显然maven-site-plugin:site这个操作绑定到了site这个phase上
- <plugin>
- <artifactId>maven-site-plugin</artifactId>
- <version>2.3</version>
- <executions>
- <execution>
- <id>default-site</id>
- <phase>site</phase>
- <goals>
- <goal>site</goal>
- </goals>
- </execution>
- </executions>
- </plugin>
2. 其次,maven-site-plugin怎么知道reporting声明的插件呢?maven-site-plugin作为maven的核心插件,怎么说也有点特权吧。的确,在Effective POM内可以发现,其实maven做了点手脚,将reporting里面的内容转化成了maven-site-plugin下的一个配置项~_~
如果reporting里面什么都没有配置,maven就会继承最原始的配置(类似Java里面的Object一样),给maven-site-plugin加上一个默认的报告插件——maven-project-info-report-plugin,在Effective POM里面可以看到:
- <plugin>
- <artifactId>maven-site-plugin</artifactId>
- <version>2.3</version>
- ...
- <configuration>
- <outputDirectory>E:\Ming\Workspace\Eclipse\3.7\org.cst.javaee.project\target\site</outputDirectory>
- <reportPlugins>
- <reportPlugin>
- <groupId>org.apache.maven.plugins</groupId>
- <artifactId>maven-project-info-reports-plugin</artifactId>
- </reportPlugin>
- </reportPlugins>
- </configuration>
- </plugin>
**注意:这个是关键,在maven-site-plugin:3.0中reportPlugins内的项由reportPlugin转为plugin,这点转变使得maven原来所做的转化操作无效,因为maven-site-plugin:3.0不再解释reportPlugins.reportPlugin转而解释reportPlugins.plugin,所以在使用maven-site-plugin:3.0的时候必须在插件的configuration.reportPlugins.plugin内声明报告插件,而不能使用reporting。并且Maven3只能使用maven-site-plugin:3.0,所以一定要注意Maven2升级至Maven3时的变动!
3. 最后,maven-site-plugin怎样调用报告插件的?这个就太简单了,插件信息都知道了想怎么调用就怎么调用!那么goal呢,配置并没有显示声明goal喔?原来maven-site-plugin也有自己的限制,它所只支持某些报告插件(当然,可能有方法通过配置让它知道更多的报告插件,这个就没研究过了),所支持插件列表在官方文档上有列出:http://maven.apache.org/plugins/maven-site-plugin/maven-3.html#Plugins_Compatibility_Matrix_for_Maven_3
文章的最后,回应一下《maven-site-plugin笔记——2.3到3.0之间的变化》最后提的问题,为何新版配置没有生成index.html,解决办法就是在maven-site-plugin:3.0的configuration中添加maven-project-info-reports-plugin的声明即可
依赖冲突:
这个是dependency的GA一样但V不同,Maven自2.9(还是2.0.9,不清楚了)开始为了确保确定性,采用如下方法解决:
1. 依赖路径浅的优先:
假设,A->B->slf4j:1.6.2,A->C->E->slf4j:1.4.1,则slf4j:1.6.2优先
2. 声明优先,如果在dependencyManagement中声明的话会优先采用对应插件
3. 覆写优先,子POM内声明的优先于父POM中的依赖
*上述的依赖均假定为compile依赖
********************************************************************************************************
重复类的类加载:
同样是2.9还是2.0.9开始,Maven在classpath中下了一点功夫,由于各个容器对类加载规则的不一,所以以下仅讨论对Eclipse的类路径影响。
现在,由Maven插件向Eclipse提供的类路径是有顺序的,以当前工程为原点,加载顺序为:
1. 首先加载test-class,这个m2eclipse的jira上显示为fixed的一个issue:http://jira.codehaus.org/browse/MECLIPSE-318
2. 当前POM的依赖,顺序按照pom.xml内使用的顺序(由上至下,靠上优先加载)
3. 父POM使用的依赖会被最后加载
*上述观点均为资料收集后理解结果,并未经过测试!