IntelliJ IDEA SpringBoot 使用第三方Tomcat以及部署

Posted _Yufan

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了IntelliJ IDEA SpringBoot 使用第三方Tomcat以及部署相关的知识,希望对你有一定的参考价值。

花了半天时间终于成功,记录以备查阅。

一、第三方Tomcat部署

部署部分参考的是:把spring-boot项目部署到tomcat容器中

目标:把spring-boot项目按照平常的web项目一样发布到tomcat容器下

1. 修改打包形式

在pom.xml里设置 <packaging>war</packaging>

<groupId>com.study</groupId>
<artifactId>test</artifactId>
<version>0.0.1-SNAPSHOT</version>
<packaging>war</packaging>

2. 移除嵌入式tomcat插件

在pom.xml里找到spring-boot-starter-web依赖节点,在其中添加如下代码:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
    <!-- 移除嵌入式tomcat插件 -->
    <exclusions>
        <exclusion>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-tomcat</artifactId>
        </exclusion>
    </exclusions>
</dependency>

3. 添加servlet-api的依赖

下面两种方式都可以,任选其一

<dependency>
    <groupId>javax.servlet</groupId>
    <artifactId>javax.servlet-api</artifactId>
    <version>3.1.0</version>
    <scope>provided</scope>
</dependency>
<dependency>
    <groupId>org.apache.tomcat</groupId>
    <artifactId>tomcat-servlet-api</artifactId>
    <version>8.0.36</version>
    <scope>provided</scope>
</dependency>

4. 修改启动类,并重写初始化方法

我们平常用main方法启动的方式,都有一个App的启动类,代码如下:

@SpringBootApplication
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
}

直接让启动类继承SpringBootServletInitializer,并覆盖configure()方法:

@SpringBootApplication
public class Application extends SpringBootServletInitializer {
  
    @Override
    protected SpringApplicationBuilder configure(SpringApplicationBuilder builder) {
        // 注意这里要指向原先用main方法执行的Application启动类
        return builder.sources(Application.class);
    }
  
    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
}

5. 打包部署

在项目根目录下(即包含pom.xml的目录),在命令行里输入: 
mvn clean package即可, 等待打包完成,出现[INFO] BUILD SUCCESS即为打包成功。 
mvn clean package 命令可以按需要添加参数,
-DskipTests 不执行测试用例,但编译测试用例类生成相应的class文件至target/test-classes下。
-Dspring.profiles.active=dev 如果配置了多环境,可以设置打包到哪个环境。
然后把target目录下的war包放到tomcat的webapps目录下,启动tomcat,即可自动解压部署。 
最后在浏览器中输入

http://localhost:[端口号]/[打包项目名]/

发布成功

二、IntelliJ IDEA下使用第三方Tomcat运行项目

1. 完成部署中的1,2,3,4步骤

2. 配置第三方Tomcat

IDEA上方工具栏:Run->Edit Configurations

打开配置界面,左边"+"号->Tomcat Server->Local

完成后如下所示

然后在Deployment中点击"+"号,选择client:war,这样每次server启动的时候都会去打包一次war包(个人理解),然后去运行war包。

然后保存就可以了,然后启动项目就运行起来了。

三、部署中遇到的问题

1. log4j.properties中使用了环境变量

log4j.appender.dailyFile.File=${catalina.base}/logs/guandata/log.log4j

在启动第三方Tomcat时,Tomcat会去环境变量中寻找叫CATALINA_HOME、CATALINA_BASE的环境变量,并将它们加入到Tomcat到系统变量中,这样我们就能在项目中使用它们了。

2. 部署后访问静态资源,访问路径URL中多了项目名,导致静态资源找不到

举例:

原本在springboot内置Tomcat中运行项目,资源访问路径URL类似于:http://localhost:8080/css/bootstrap.min.css
但是打包部署到第三方Tomcat中,Tomcat默认会在路径中加上项目名,类似于:http://localhost:8080/myproject/css/bootstrap.min.css
这样就导致了访问不到静态资源 404

原因:

由于在前端引用资源时使用了绝对路径:

<link href="/css/bootstrap.min.css" rel="stylesheet">

这样默认在 http://localhost:8080/css/bootstrap.min.css 下面寻找,因此,我们需要在前面加上项目名,但是写死的话万一改项目名就很麻烦,当然是有办法解决的。

解决:

1. 在Java后端定义一个拦截器,用来获取 request.getContextPath() 然后传到前端,request.getContextPath() 得到的是项目的根路径,具体参考:关于request.getServletPath(),request.getContextPath()的总结

 1 @Component
 2 public class CommonIntercepter implements HandlerInterceptor {
 3 
 4     private final Logger logger = LoggerFactory.getLogger(this.getClass());
 5 
 6     @Override
 7     public boolean preHandle(HttpServletRequest request,
 8                              HttpServletResponse response, Object handler) throws Exception {
 9          return true;
10     }
11 
12     @Override
13     public void postHandle(HttpServletRequest request,
14                            HttpServletResponse response, Object handler,
15                            ModelAndView modelAndView) throws Exception {
16             request.setAttribute("ctx", request.getContextPath());
17             request.setAttribute("version", DateTimeUtils.currentTimeMillis());
18     }
19 
20     @Override
21     public void afterCompletion(HttpServletRequest request,
22                                 HttpServletResponse response, Object handler, Exception ex)
23             throws Exception {
24 
25     }
26 
27 
28 }

2. 前端在静态资源访问url前加上${ctx!},注意我用的是freemarker模版

<link href="${ctx!}/css/bootstrap.min.css" rel="stylesheet">
<link href="${ctx!}/css/font-awesome.min.css" rel="stylesheet">
<link href="${ctx!}/css/plugins/bootstrap-table/bootstrap-table.min.css" rel="stylesheet">
<link href="${ctx!}/css/animate.css" rel="stylesheet">
<link href="${ctx!}/css/style.css" rel="stylesheet">

这样就能正确访问到静态资源啦。

3. 要注意给Tomcat/bin路径下的脚本权限,否则可能会出现不可预料的后果

sudo chmod 777 /Library/Tomcat/apache-tomcat-9.0.10/bin/*.sh

以上是关于IntelliJ IDEA SpringBoot 使用第三方Tomcat以及部署的主要内容,如果未能解决你的问题,请参考以下文章

Intellij idea新建一个springboot项目

详解Intellij IDEA搭建SpringBoot

intellij idea 怎么找不到spring boot

intellij idea搭建springboot

IntelliJ IDEA搭建SpringBoot应用

intellij idea springboot无法读取配置文件的解决方法