学习一下maven,整理一下比较混淆的概念和使用方法,记录笔记
1:Maven工程依赖
在Eclipse创建三个独立的Maven工程:web工程、service工程、dao工程,pom.xml分别如下:
web工程:依赖service
<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>com.paic.maven.cms</groupId>
<artifactId>web</artifactId>
<packaging>war</packaging>
<version>0.0.1-SNAPSHOT</version>
<name>web Maven Webapp</name>
<url>http://maven.apache.org</url>
<dependencies>
<dependency>
<groupId>com.paic.maven.cms</groupId>
<artifactId>service</artifactId>
<version>0.0.1-SNAPSHOT</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.1.0</version>
</dependency>
</dependencies>
<build>
<finalName>web</finalName>
</build>
</project>
service工程:依赖dao
<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/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.paic.maven.cms</groupId>
<artifactId>service</artifactId>
<version>0.0.1-SNAPSHOT</version>
<dependencies>
<dependency>
<groupId>com.paic.maven.cms</groupId>
<artifactId>dao</artifactId>
<version>0.0.1-SNAPSHOT</version>
</dependency>
</dependencies>
</project>
dao工程
<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/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.paic.maven.cms</groupId>
<artifactId>dao</artifactId>
<version>0.0.1-SNAPSHOT</version>
</project>
在classpath中体现出的结果就是:
service工程:
web工程:
可以看到web工程将service工程引用到了classpath下,并且通过service工程的依赖传递同时依赖了dao工程。效果很直观。
下面对web工程打包:maven install
报错信息如下:
[ERROR] Failed to execute goal on project web: Could not resolve dependencies for project com.paic.maven.cms:web:war:0.0.1-SNAPSHOT: Could not find artifact com.paic.maven.cms:service:jar:0.0.1-SNAPSHOT -> [Help 1]
[ERROR]
没有找到service的artifact,也就是没有找到service工程的jar
下面对service工程执行maven install,报错信息类似,没有找到dao的jar
没办法,依次对dao,service,web工程执行install操作
执行完毕之后
[INFO] Installing E:\maven_test_workspace\web\target\web.war to e:\repository\com\paic\maven\cms\web\0.0.1-SNAPSHOT\web-0.0.1-SNAPSHOT.war
[INFO] Installing E:\maven_test_workspace\web\pom.xml to e:\repository\com\paic\maven\cms\web\0.0.1-SNAPSHOT\web-0.0.1-SNAPSHOT.pom
可以看到在工程下,生成了war包,并且拷贝到了maven的repository下
service和dao文件夹下生成的就是其pom.xml中定义的jar,而web工程则是生成war
打开web工程里面生成的war包,lib如下:
依赖的jar包被自动打到lib下
这样虽然实现了依赖,但是还需要一个一个独立的按照依赖顺序分别打包,比较麻烦。这是第一个问题;
另外一个问题
修改dao工程的pom.xml,新增一个依赖
<dependencies>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.11</version>
</dependency>
</dependencies>
这里依赖了junit,如果要使用该jar的类,则dao,service,web都可以使用,因为它们存在依赖顺序。但是如果把junit的依赖放在service中,则dao就不能使用了,也是因为它们存在依赖顺序。
所以,依赖放在最底层,则其上面的工程就无条件引入,如果jar多次引用,会造成版本冲突,混淆;如果放在上层,则底层无法享受引用,比较麻烦:jar包依赖和工程依赖混在一起。。。
2:使用maven聚合
聚合的作用可以解决第一个问题:打包问题
简述一下就是拥有一个父maven工程,它包含了很多子模块,当对父maven工程打包时,maven会自动的将其聚合的子模块一起打包,不再需要一个一个install了
首先创建一个父maven工程,pom.xml如下:
<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/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.paic.maven.aggregate</groupId>
<artifactId>aggregate-parent</artifactId>
<version>0.0.1-SNAPSHOT</version>
<packaging>pom</packaging>
<modules>
<module>../aggregate-service</module>
<module>../aggregate-web</module>
</modules>
</project>
注意这里的打包packing必须为pom,且多了一个节点<modules>,其内容就代表了它要聚合的模块。下面就看看两个子模块的pom.xml
aggregate-web
<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/xsd/maven-4.0.0.xsd">
<groupId>com.paic.maven.aggregate</groupId>
<modelVersion>4.0.0</modelVersion>
<artifactId>aggregate-web</artifactId>
<version>0.0.1-SNAPSHOT</version>
<packaging>war</packaging>
<dependencies>
<dependency>
<groupId>com.paic.maven.aggregate</groupId>
<artifactId>aggregate-service</artifactId>
<version>0.0.1-SNAPSHOT</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.1.0</version>
</dependency>
</dependencies>
</project>
aggregate-service
<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/xsd/maven-4.0.0.xsd">
<groupId>com.paic.maven.aggregate</groupId>
<modelVersion>4.0.0</modelVersion>
<artifactId>aggregate-web</artifactId>
<version>0.0.1-SNAPSHOT</version>
<packaging>war</packaging>
<dependencies>
<dependency>
<groupId>com.paic.maven.aggregate</groupId>
<artifactId>aggregate-service</artifactId>
<version>0.0.1-SNAPSHOT</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.1.0</version>
</dependency>
</dependencies>
</project>
这样aggregate-web和aggregate-service都被聚合到了aggregate-parent下,并且aggregate-web依赖了aggregate-service,下面对aggregate-parent执行mvn install
执行成功,结果为:
并且打开aggregate-web中的war包,查看lib
不但把servlet的jar打入,并且也把aggregate-service的jar打入了,消除了最开始时必须按顺序对每个工程执行install的问题
3:使用maven继承
copy一些maven继承的概念和优点:
这比较符合的解决第二个问题,解决的问题简述:拥有一个父maven工程,它提供统一的jar依赖,子工程无须再次引入,并且父maven提供的依赖子类可以自主选择,不会强制依赖所有
为了简单起见,直接在上面的聚合例子中修改
修改父类aggregate-parent的pom.xml为:
<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/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.paic.maven.aggregate</groupId>
<artifactId>aggregate-parent</artifactId>
<version>0.0.1-SNAPSHOT</version>
<packaging>pom</packaging>
<modules>
<module>../aggregate-service</module>
<module>../aggregate-web</module>
</modules>
<!-- https://mvnrepository.com/artifact/org.mybatis/mybatis -->
<dependencies>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis</artifactId>
<version>3.4.1</version>
</dependency>
</dependencies>
<!-- dependencyManagement中的依赖不会被子项目依赖,其目的之一是提供各种依赖,供子项目按需自主选择 -->
<dependencyManagement>
<dependencies>
<!-- https://mvnrepository.com/artifact/junit/junit -->
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.11</version>
</dependency>
</dependencies>
</dependencyManagement>
</project>
新增了一些节点
aggregate-web的pom.xml
<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/xsd/maven-4.0.0.xsd">
<parent>
<groupId>com.paic.maven.aggregate</groupId>
<artifactId>aggregate-parent</artifactId>
<version>0.0.1-SNAPSHOT</version>
<relativePath>../aggregate-parent/pom.xml</relativePath>
</parent>
<modelVersion>4.0.0</modelVersion>
<artifactId>aggregate-web</artifactId>
<packaging>war</packaging>
<dependencies>
<dependency>
<groupId>com.paic.maven.aggregate</groupId>
<artifactId>aggregate-service</artifactId>
<version>0.0.1-SNAPSHOT</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.11</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.1.0</version>
</dependency>
</dependencies>
</project>
这里主要是新增了<parent>节点,目的就是要找到父pom,并且由于继承关系,<groupId>和<version>直接继承了父类,本身无须再次定义,只需要定义个性化的<artifactId>,还有<relativePath>
aggregate-service
<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/xsd/maven-4.0.0.xsd">
<parent>
<groupId>com.paic.maven.aggregate</groupId>
<artifactId>aggregate-parent</artifactId>
<version>0.0.1-SNAPSHOT</version>
<relativePath>../aggregate-parent/pom.xml</relativePath>
</parent>
<modelVersion>4.0.0</modelVersion>
<artifactId>aggregate-service</artifactId>
</project>
和上面类似
主要看一下aggregate-web的maven依赖:
可以看到web中并没有定义mybatis的依赖,但是它classpath中仍有其jar,这就是从parent中继承而来,同时包括了service的依赖;另一方面,web中还显式的引用了父类junit4.11,故jar也存在
看看aggregate-service的依赖:
和web类似,引入了mybatis,却没有junit,实现了自主选择依赖~!
打个包,三个均成功,还比较顺利·!