Maven入门看这一篇足够了(w字)
一、基本概念1) 概述1、什么是 MavenMaven 的正确发音是[ˈmevən],而不是“马瘟”以及其他什么瘟。Maven 在美国是一个口语化的词 语,代表专家、内行的意思。 一个对 Maven 比较正式的定义是这么说的:Maven 是一个项目管理工具,它包含了一个项目对象模 型 (POM:Project Object Model),一组标准集合,一个项目生命周期(Project Lifecy
😍一、 基本概念
1) 概述
🍀 什么是 Maven
Maven 的正确发音是[ˈmevən],而不是“马瘟”以及其他什么瘟。Maven 在美国是一个口语化的词 语,代表专家、内行的意思。 一个对 Maven 比较正式的定义是这么说的:Maven 是一个项目管理工具,它包含了一个项目对象模 型 (POM:Project Object Model),一组标准集合,一个项目生命周期(Project Lifecycle),一个依赖管 理系统(Dependency Management System),和用来运行定义在生命周期阶段(phase)中插件(plugin)目标 (goal)的逻辑。
🍀 Maven 能解决什么问题
可以用更通俗的方式来说明。我们知道,项目开发不仅仅是写写代码而已,期间会伴随着各种 必不可少的事情要做,下面列举几个感受一下: 1、我们需要引用各种 jar 包,尤其是比较大的工程,引用的 jar 包往往有几十个乃至上百个, 每用 到一种 jar 包,都需要手动引入工程目录,而且经常遇到各种让人抓狂的 jar 包冲突,版本冲突。 2、我们辛辛苦苦写好了 Java 文件,可是只懂 0 和 1 的白痴电脑却完全读不懂,需要将它编译成二 进制字节码。好歹现在这项工作可以由各种集成开发工具帮我们完成,Eclipse、IDEA 等都可以将代 码即时编译。当然,如果你嫌生命漫长,何不铺张,也可以用记事本来敲代码,然后用 javac 命令一 个个地去编译,逗电脑玩。 3、世界上没有不存在 bug 的代码,计算机喜欢 bug 就和人们总是喜欢美女帅哥一样。为了追求美为 了减少 bug,因此写完了代码,我们还要写一些单元测试,然后一个个的运行来检验代码质量。 4、再优雅的代码也是要出来卖的。我们后面还需要把代码与各种配置文件、资源整合到一起,定型 打包,如果是 web 项目,还需要将之发布到服务器,供人蹂躏。 试想,如果现在有一种工具,可以把你从上面的繁琐工作中解放出来,能帮你构建工程,管理 jar 包,编译代码,还能帮你自动运行单元测试,打包,生成报表,甚至能帮你部署项目,生成 Web 站 点,你会心动吗?Maven 就可以解决上面所提到的这些问题。
🍀 Maven 的优势举例
前面我们通过 Web 阶段项目,要能够将项目运行起来,就必须将该项目所依赖的一些 jar 包添加到 工程中,否则项目就不能运行。试想如果具有相同架构的项目有十个,那么我们就需要将这一份 jar 包复制到十个不同的工程中。我们一起来看一个 CRM项目的工程大小。 使用传统 Web 项目构建的 CRM 项目如下:

原因主要是因为上面的 WEB 程序要运行,我们必须将项目运行所需的 Jar 包复制到工程目录中,从 而导致了工程很大。 同样的项目,如果我们使用 Maven 工程来构建,会发现总体上工程的大小会少很多。如下图

小结:可以初步推断它里面一定没有 jar 包,继续思考,没有 jar 包的项目怎么可能运行呢?
2) 依赖管理概念
Maven 的一个核心特性就是依赖管理。当我们涉及到多模块的项目(包含成百个模块或者子项目),管理依赖就变成 一项困难的任务。Maven 展示出了它对处理这种情形的高度控制。 传统的 WEB 项目中,我们必须将工程所依赖的 jar 包复制到工程中,导致了工程的变得很大。那么 maven 工程是如何使得工程变得很少呢? 分析如下:


通过分析发现:maven 工程中不直接将 jar 包导入到工程中,而是通过在 pom.xml 文件中添加所需 jar 包的坐标,这样就很好的避免了 jar 直接引入进来,在需要用到 jar 包的时候,只要查找 pom.xml 文 件,再通过 pom.xml 文件中的坐标,到一个专门用于”存放 jar 包的仓库”(maven 仓库)中根据坐标从 而找到这些 jar 包,再把这些 jar 包拿去运行。 那么问题来了
第一:”存放 jar 包的仓库”长什么样?
第二:通过读取 pom.xml 文件中的坐标,再到仓库中找到 jar 包,会不会很慢?从而导致这种方式 不可行!
第一个问题:存放 jar 包的仓库长什么样,这一点我们后期会分析仓库的分类,也会带大家去看我们 的本地的仓库长什么样。
第二个问题:通过 pom.xml 文件配置要引入的 jar 包的坐标,再读取坐标并到仓库中加载 jar 包,这 样我们就可以直接使用 jar 包了,为了解决这个过程中速度慢的问题,maven 中也有索引的概念,通 过建立索引,可以大大提高加载 jar 包的速度,使得我们认为 jar 包基本跟放在本地的工程文件中再 读取出来的速度是一样的。这个过程就好比我们查阅字典时,为了能够加快查找到内容,书前面的 目录就好比是索引,有了这个目录我们就可以方便找到内容了,一样的在 maven 仓库中有了索引我 们就可以认为可以快速找到 jar 包。
3) 一键构建概念
我们的项目,往往都要经历编译、测试、运行、打包、安装 ,部署等一系列过程。 什么是构建? 指的是项目从编译、测试、运行、打包、安装 ,部署整个过程都交给 maven 进行管理,这个 过程称为构建。 一键构建 指的是整个构建过程,使用 maven 一个命令可以轻松完成整个工作。
Maven 规范化构建流程如下:

我们一起来看 Hello-Maven 工程的一键运行的过程。通过 tomcat:run 的这个命令,我们发现现在的 工程编译,测试,运行都变得非常简单。mvn tomcat:run

在命令窗口的最后 访问链接即可打开项目

但是这个地方有一个问题 就是访问链接的时候 会显示 jsp未编译的500bug
org.apache.jasper.JasperException: org.apache.jasper.JasperException: Unable to load class for JSP
500报错原因:jdk版本是1.8的,不支持tomcat:run默认的tomcat6,改成tomcat7依赖包即可。
这个是参考的pom.xml 但是 我们这边使用的资料项目文件 pom不用修改 启动命令变成
mvn tomcat7:run 即可
https://www.cnblogs.com/nylgwn/p/14192091.html
效果图

😍 二、maven的安装和仓库种类
1) 安装
🍀 Maven 软件的下载
为了使用 Maven 管理工具,我们首先要到官网去下载它的安装软件。通过百度搜索“Maven“如下:

点击 Download 链接,就可以直接进入到 Maven 软件的下载页面:

目前最新版是 apache-maven-3.5.3 版本,我们当时使用的是 apache-maven-3.5.2 版本,大家也可以下 载最新版本。 Apache-maven-3.5.2 下载地址:(现在3.6版本已经普及 很多项目上也都在用3.6 两个版本有差别 跑项目的时候可以二者互换)
http://archive.apache.org/dist/maven/maven-3/ 下载后的版本如下:
![]()
🍀 Maven 软件的安装
Maven 下载后,将 Maven 解压到一个没有中文没有空格的路径下,比如 D:\software\maven 下面。 解压后目录结构如下:

bin:存放了 maven 的命令,比如我们前面用到的 mvn tomcat:run boot:存放了一些 maven 本身的引导程序,如类加载器等 conf:存放了 maven 的一些配置文件,如 setting.xml 文件 lib:存放了 maven 本身运行所需的一些 jar 包 至此我们的 maven 软件就可以使用了,前提是你的电脑上之前已经安装并配置好了 JDK。
🍀 JDK 的准备及统一
本次博客我们所使用工具软件的统一,JDK 使用 JDK8版本 1. JDK 环境:

🍀 环境变量
电脑上需安装 java 环境,安装 JDK1.7 + 版本 (将JAVA_HOME/bin 配置环境变量 path ),我们使 用的是 JDK8 相关版本 配置 MAVEN_HOME ,变量值就是你的 maven 安装 的路径(bin 目录之前一级目录)

上面配置了我们的 Maven 软件,注意这个目录就是之前你解压 maven 的压缩文件包在的的目录,最 好不要有中文和空格。 再次检查 JDK 的安装目录,如下图:

PATH变量设置

通过 mvn -v命令检查 maven 是否安装成功,看到 maven 的版本为 3.5.2 及 java 版本为 1.8 即为安装 成功。 找开 cmd 命令,输入 mvn –v命令,如下图:

我们发现 maven 的版本,及 jdk 的版本符合要求,这样我们的 maven 软件安装就成功了。
2) 仓库种类和彼此关系
maven 的工作需要从仓库下载一些 jar 包,如下图所示,本地的项目 A、项目 B 等都会通过 maven 软件从远程仓库(可以理解为互联网上的仓库)下载 jar 包并存在本地仓库,本地仓库 就是本地文 件夹,当第二次需要此 jar 包时则不再从远程仓库下载,因为本地仓库已经存在了,可以将本地仓库 理解为缓存,有了本地仓库就不用每次从远程仓库下载了。

本地仓库 :用来存储从远程仓库或中央仓库下载的插件和 jar 包,项目使用一些插件或 jar 包, 优先从本地仓库查找 默认本地仓库位置在 ${user.dir}/.m2/repository,${user.dir}表示 windows 用户目录。

远程仓库:如果本地需要插件或者 jar 包,本地仓库没有,默认去远程仓库下载。 远程仓库可以在互联网内也可以在局域网内。
中央仓库 :在 maven 软件中内置一个远程仓库地址 http://repo1.maven.org/maven2 ,它是中 央仓库,服务于整个互联网,它是由 Maven 团队自己维护,里面存储了非常全的 jar 包,它包 含了世界上大部分流行的开源项目构件。
🍀settings.xml文件设置
在 MAVE_HOME/conf/settings.xml 文件中配置本地仓库位置(maven 的安装目录下)

😍三、 maven标准目录结构和常用命令
1)目录结构
src/main/java —— 存放项目的.java 文件
src/main/resources —— 存放项目资源文件,如 spring, hibernate 配置文件
src/test/java —— 存放所有单元测试.java 文件,如 JUnit 测试类
src/test/resources —— 测试资源文件
target —— 项目输出位置,编译后的 class 文件会输出到此目录
pom.xml——maven 项目核心配置文件
注意:如果是普通的 java 项目,那么就没有 webapp 目录。
src/main/webapp 页面资源 js css 图片等等
2) 常用命令
clean
clean 是 maven 工程的清理命令,执行 clean 会删除 target 目录及内容。
compile
compile 是 maven 工程的编译命令,作用是将 src/main/java 下的文件编译为 class 文件输出到 target目录下。cmd 进入命令状态,执行 mvn compile,查看 target 目录,class 文件已生成,编译完成。
test
test 是 maven 工程的测试命令 mvn test,会执行 src/test/java 下的单元测试类。
cmd 执行 mvn test 执行 src/test/java 下单元测试类,运行 1 个测试用例,全部成功。
package
package 是 maven 工程的打包命令,对于 java 工程执行 package 打成 jar 包,对于 web 工程打成 war包。
install
install 是 maven 工程的安装命令,执行 install 将 maven 打成 jar 包或 war 包发布到本地仓库。
从运行结果中,可以看出:当后面的命令执行时,前面的操作过程也都会自动执行,这包括compile正式代码的编译 test 测试代码的编译 package的打包过程
😍四、 maven生命周期和概念模型图
1)生命周期

2)概念模型
📍 项目对象模型 (Project Object Model) 一个 maven 工程都有一个 pom.xml 文件,通过 pom.xml 文件定义项目的坐标、项目依赖、项目信息、 插件目标等。
📍 依赖管理系统(Dependency Management System) 通过 maven 的依赖管理对项目所依赖的 jar 包进行统一管理
📍 一个项目生命周期(Project Lifecycle) 使用 maven 完成项目的构建,项目构建包括:清理、编译、测试、部署等过程,maven 将这些 过程规范为一个生命周期,如下所示是生命周期的各各阶段:
📍 maven 通过执行一些简单命令即可实现上边生命周期的各各过程,比如执行 mvn compile 执行编译、 执行 mvn clean 执行清理。
📍 一组标准集合 maven 将整个项目管理过程定义一组标准,比如:通过 maven 构建工程有标准的目录结构,有 标准的生命周期阶段、依赖管理有标准的坐标定义等。
📍 插件(plugin)目标(goal) maven 管理项目生命周期过程都是基于插件完成的。
😍五、 使用骨架创建maven的java工程
idea 开发 maven 项目
在实战的环境中,我们都会使用流行的工具来开发项目。
🍀 idea 的 maven 配置
打开 File——Settings ——配置 maven
依据图片指示,选择本地 maven 安装目录,指定 maven 安装目录下 conf 文件夹中 settings 配置文件。

1)使用骨架创建maven工程

2)不使用骨架创建maven工程
直接点击next


创建速度比较快 而且目录结构全 干净 推荐不适用骨架进行创建maven工程
3)骨架创建maven的web工程
打开 idea,选择创建一个新工程

选择 idea 提供好的 maven 的web 工程模板


点击 Next 填写项目信息

点击 Next,此处不做改动。

点击 Next 选择项目所在目录

点击 Finish 后开始创建工程,耐心等待,直到出现如下界面



手动添加 src/main/java 目录,如下图右键 main 文件夹——New——Directory

创建一个新的文件夹命名为 java

点击 OK 后,在新的文件夹 java 上右键——Make Directory as——Sources Root

😍六、 maven工程servlet实例
如何自定义指定web目录(这种情况适用于使用Myclipse开发的项目 放在idea中运行 因为目录结构不太一致 所以需要手动指定一下) 创建jsp


实际开发以图二为标准

1) 创建一个 Servlet
src/java/main 创建了一个 Servlet,但报错

要解决问题,就是要将 servlet-api-xxx.jar 包放进来,作为 maven 工程应当添加 servlet 的坐标,从而 导入它的 jar
2) 在 pom.xml 文件添加坐标
直接打开 hello_maven 工程的 pom.xml 文件,再添加坐标

添加 jar 包的坐标时,还可以指定这个 jar 包将来的作用范围。
每个 maven 工程都需要定义本工程的坐标,坐标是 maven 对 jar 包的身份定义
🍀 坐标的来源方式
添加依赖需要指定依赖 jar 包的坐标,但是很多情况我们是不知道 jar 包的的坐标,可以通过如下方 式查询:
从互联网搜索
http://search.maven.org/
http://mvnrepository.com/
网站搜索示例:

3) 依赖范围
A 依赖 B,需要在 A 的 pom.xml 文件中添加 B 的坐标,添加坐标时需要指定依赖范围,依赖范围包 括
📍 compile:编译范围,指 A 在编译时依赖 B,此范围为默认依赖范围。编译范围的依赖会用在
编译、测试、运行,由于运行时需要所以编译范围的依赖会被打包。
📍 provided:provided 依赖只有在当 JDK 或者一个容器已提供该依赖之后才使用, provided 依
赖在编译和测试时需要,在运行时不需要,比如:servlet api 被 tomcat 容器提供。
📍 runtime:runtime 依赖在运行和测试系统的时候需要,但在编译的时候不需要。比如:jdbc
的驱动包。由于运行时需要所以 runtime 范围的依赖会被打包。
📍 test:test 范围依赖 在编译和运行时都不需要,它们只有在测试编译和测试运行阶段可用,
比如:junit。由于运行时不需要所以 test范围依赖不会被打包。📍 system:system 范围依赖与 📍 provided 类似,但是你必须显式的提供一个对于本地系统中 JAR
文件的路径,需要指定 systemPath 磁盘路径,system依赖不推荐使用。

在 maven-web 工程中测试各各 scop。
测试总结:
🚩 默认引入 的 jar 包 ------- compile 【默认范围 可以不写】(编译、测试、运行 都有效 )
🚩 servlet-api 、jsp-api ------- provided (编译、测试 有效, 运行时无效 防止和 tomcat 下 jar 冲突)
🚩 jdbc 驱动 jar 包 ---- runtime (测试、运行 有效 )
🚩 junit ----- test (测试有效)
依赖范围由强到弱的顺序是:compile>provided>runtime>test
🍀 项目中添加的坐标
编写 servlet
这里一开始会报错 需要在pom.xml中引入坐标
<dependencies>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
<version>2.5</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>javax.servlet.jsp</groupId>
<artifactId>jsp-api</artifactId>
<version>2.0</version>
<scope>provided</scope>
</dependency> <!--直接回车 idea有提示-->
<!--测试jar-->
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
</dependencies>

编写 jsp


这里有两种命令 一种是mvn tomcat:run 一种是mvn tomcat7:run
自己亲测 tomcat:run的命令并不能实现servlet跳转jsp页面 所以这里不在贴tomcat:run命令的相关图

出现这种问题是不能识别项目中的servlet 之所以识别不了 是因为pom.xml中的jar和tomcat的jar名字相同 冲突 但是pom.xml中的jar是当时为了解决servlet中继承HttpServlet类报错问题的 不能删除
这就导致 pom.xml中的jar能够使得编译通过 但是运行不行 不能通过 那么这个时候 我想让pom.xml中的jar只在编译的时候起作用 项目运行的时候不起作用
解决方式 增加作用域代码
<scope>provided</scope>

🍀 工程运行环境修改
增加tomcat7插件 servlet页面跳转插件
<build>
<plugins>
<plugin>
<groupId>org.apache.tomcat.maven</groupId>
<artifactId>tomcat7-maven-plugin</artifactId>
<version>2.2</version>
<configuration>
<port>8888</port>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.8.1</version>
<configuration>
<source>1.7</source>
<target>1.7</target>
<encoding>UTF-8</encoding>
</configuration>
</plugin>
</plugins>
</build>

启动的时候使用mvn tomcat7:run启动

🍀 动态模板
live cumtom xml

4) maven的Java工程取mysql数据库
🍀创建数据库
DROP TABLE IF EXISTS `user_info`;
CREATE TABLE `user_info` (
`id` int(11) NOT NULL AUTO_INCREMENT COMMENT '自增ID',
`uid` int(11) NOT NULL COMMENT '用户ID',
`nick_name` varchar(64) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '昵称',
`achievement` int(11) NULL DEFAULT NULL COMMENT '成就值',
`level` int(11) NULL DEFAULT NULL COMMENT '用户等级',
`job` varchar(32) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '职业方向',
`register_time` datetime(0) NULL DEFAULT NULL COMMENT '注册时间',
PRIMARY KEY (`id`) USING BTREE,
UNIQUE INDEX `uid`(`uid`) USING BTREE
) ENGINE = InnoDB AUTO_INCREMENT = 7 CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Dynamic;
INSERT INTO `user_info` VALUES (1, 1001, '牛客1号', 3100, 7, '算法', '2020-01-01 10:00:00');
INSERT INTO `user_info` VALUES (2, 1002, '牛客2号', 2100, 6, '算法', '2020-01-01 10:00:00');
INSERT INTO `user_info` VALUES (3, 1003, '牛客3号', 1500, 5, '算法', '2020-01-01 10:00:00');
INSERT INTO `user_info` VALUES (4, 1004, '牛客4号', 1100, 4, '算法', '2020-01-01 10:00:00');
INSERT INTO `user_info` VALUES (5, 1005, '牛客5号', 1600, 6, 'C++', '2020-01-01 10:00:00');
INSERT INTO `user_info` VALUES (6, 1006, '牛客6号', 3000, 6, 'C++', '2020-01-01 10:00:00');

🍀不使用骨架创建maven工程maven_mysql
🍀创建实体bean
🍀导入mysql-connect-java驱动包并设置作用域
<dependencies>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.6</version>
<scope>runtime</scope>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
</dependencies>
🍀实体类 getset方法tostring方法

🍀 dao接口

🍀 daoImpl实现类主要代码
public class UserInfoDaoImpl implements UserInfoDao {
@Override
public List<UserInfo> findAll() throws SQLException {
ArrayList<UserInfo> list = new ArrayList<>();
Connection conn = null;
PreparedStatement pstm = null;
ResultSet rs = null;
try {
//2 加载驱动类
Class.forName("com.mysql.jdbc.Driver");
//1 获取connect对象类
conn = DriverManager.getConnection("jdbc:mysql:///db_heima_day23", "root", "root");
//3 获取操作数据对象
pstm = conn.prepareStatement("select * from user_info");
//4 执行数据库查询操作
rs = pstm.executeQuery();
//把数据库结果集转成Java的List集合
while (rs.next()){
UserInfo userInfo = new UserInfo();
userInfo.setId(rs.getInt("id"));
userInfo.setUid(rs.getInt("uid"));
userInfo.setNick_name(rs.getString("nick_name"));
userInfo.setAchievement(rs.getInt("achievement"));
userInfo.setLevel(rs.getInt("level"));
userInfo.setJob(rs.getString("job"));
userInfo.setRegister_time(rs.getDate("register_time"));
list.add(userInfo);
}
} catch (Exception throwables) {
throwables.printStackTrace();
}finally {
conn.close();
pstm.close();
rs.close();
}
return list;
}
}
🍀测试类
public class UserInfoTest {
@Test
public void findAll() throws SQLException {
UserInfoDao dao = new UserInfoDaoImpl();
List<UserInfo> list = dao.findAll();
for (UserInfo userInfo : list) {
System.out.println("userInfo = " + userInfo);
}
}
}
🍀效果图

内容来自黑马视频资料 Thanks HeiMa!!!
更多推荐





所有评论(0)