SpringBoot项目——springboot配置Tomcat两个端口,https和http的方式 jar的打包和运行

目录

  • 引出
  • springboot配置Tomcat两个端口,https和http的方式
    • 1.生成SSL证书
    • 2.配置client.p12和https端口
    • 3.配置http的8080端口WebServerFactoryCustomizer接口
    • 4.启动项目
  • 项目应用:在某项目中有一个功能需要https协议
    • Tomcat启动https和http两个端口
    • 根据http或者https确定拦截后到哪个页面
    • 后端共享值,前端form表单获取
  • 问题:在https协议下,发送axios请求没反应
    • 解决方案一:用form表单
  • 项目的打包部署
    • 1.template might not exist or might not be accessible by any of the configured Template Resolvers

引出


1.springboot配置Tomcat两个端口,https和http的方式;
2.在https协议下,发送axios请求没反应,暂时用form表单解决;
3.运行jar包template might not exist报错及解决;

springboot配置Tomcat两个端口,https和http的方式

1.生成SSL证书

严格来说https不是一个独立协议,只是在http协议基础上增加了SSL/TLS加密层。所以我们需要先生成SSL证书,这里使用keytool生成jks。

keytool -genkey -alias client -keypass 12345678 -keyalg RSA -keysize 2048 -validity 365 -storetype PKCS12 -keystore ./client.p12 -storepass 12345678

在这里插入图片描述

在这里插入图片描述

2.配置client.p12和https端口

server:ssl:key-store: classpath:client.p12key-store-password: 12345678key-store-type: PKCS12key-alias: client# https的访问端口port: 8443

3.配置http的8080端口WebServerFactoryCustomizer接口

WebServerFactory接口的几个重要实现:

  • TomcatServletWebServerFactory:对应于tomcat

  • JettyServletWebServerFactory:对应jetty

  • UndertowServletWebServerFactory:对应undertow

  • NettyReactiveWebServerFactory:对应netty

Spring Boot默认使用http/1.1协议。所以我们增加额外的自定义https连接器。

package com.shanxi.gis.config;import org.apache.catalina.connector.Connector;
import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory;
import org.springframework.boot.web.server.WebServerFactory;
import org.springframework.boot.web.server.WebServerFactoryCustomizer;
import org.springframework.stereotype.Component;@Component
public class TomcatServerCustomer implements WebServerFactoryCustomizer<TomcatServletWebServerFactory> {@Overridepublic void customize(TomcatServletWebServerFactory factory) {final Connector httpConn = new Connector("HTTP/1.1");httpConn.setPort(8080);factory.addAdditionalTomcatConnectors(httpConn);}
}

4.启动项目

运行项目后可以看到启动了https的8843和http的8080两个端口

在这里插入图片描述

项目应用:在某项目中有一个功能需要https协议

Tomcat启动https和http两个端口

在这里插入图片描述

在这里插入图片描述

TomcatServerCustomer.java文件

package com.shanxi.gis.config;import org.apache.catalina.connector.Connector;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory;
import org.springframework.boot.web.server.WebServerFactory;
import org.springframework.boot.web.server.WebServerFactoryCustomizer;
import org.springframework.stereotype.Component;@Component
public class TomcatServerCustomer implements WebServerFactoryCustomizer<TomcatServletWebServerFactory> {@Value("${ServerHttpPort}")Integer httpHost;@Overridepublic void customize(TomcatServletWebServerFactory factory) {final Connector httpConn = new Connector("HTTP/1.1");httpConn.setPort(httpHost);factory.addAdditionalTomcatConnectors(httpConn);}
}

application.yml配置文件

server:ssl:key-store: classpath:client.p12key-store-password: 12345678key-store-type: PKCS12key-alias: client# https的访问端口port: 8443# 部署服务器的配置
ServerHttpsUrl: https://localhost:8443 # https的url
ServerHttpUrl: http://localhost:8080 # http的url
ServerHttpPort: 8080 # http的端口号
LoginPassword: Admin@1a2 # 登陆的密码spring:mvc:static-path-pattern: /**resources:static-locations: classpath:/static/thymeleaf:prefix: classpath:/templates/check-template-location: truecache: falsesuffix: .html #模板后缀encoding: UTF-8 #编码mode: HTML #模板servlet:content-type: text/html

根据http或者https确定拦截后到哪个页面

LoginAuthorInterceptor.java文件

request.getScheme(); // for example, http, https, or ftp.

package com.shanxi.gis.interceptor;import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;/*** spring的拦截器,* 1.在容器中,@Component* 2.是spring的拦截器 implements HandlerInterceptor*/
@Component
public class LoginAuthorInterceptor implements HandlerInterceptor {@Value("${ServerHttpsUrl}")String httpsUrl;@Value("${ServerHttpUrl}")String httpUrl;@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {// 如果没有登陆,就去登陆页面,如果登陆了就放行HttpSession session = request.getSession();Object user = session.getAttribute("user");System.out.println(user);if ("admin".equals(user)){return true;}else {// 没有登陆,就去登陆页面String scheme = request.getScheme(); // for example, http, https, or ftp.// 如果是http就去,http的端口if ("http".equals(scheme)){response.sendRedirect(httpUrl+"/user/loginPage");}// 否则就去https的端口response.sendRedirect(httpsUrl+"/user/loginPage");return false;}}
}

后端共享值,前端form表单获取

login.html页面

<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head><meta charset="UTF-8"><title>登陆页面</title><link rel="stylesheet" href="/bootstrap/css/bootstrap.min.css"><script src="/js/jquery-3.5.1.js"></script><script src="/bootstrap/js/bootstrap.js"></script><script src="/js/axios.min.js"></script><script src="/js/vue.min-v2.5.16.js"></script><link rel="stylesheet" href="/css/login.css">
</head><body><div class="login-container" id="app"><h2>欢迎登录</h2>
<!--        "https://localhost:8443/user/login"--><form :action=url method="post"><label for="username">用户名:</label><input type="text" id="username" v-model="username" placeholder="请输入用户名" required name="username"><label for="password">密码:</label><input type="password" id="password" v-model="password" placeholder="请输入密码" required name="password"><input type="submit" value="登录" @click="loginBtn" class="btn btn-primary btn-block"></form></div><script>let app = new Vue({el:"#app",data:{username:"",password:"",url:"[[${httpsUrl}]]",},methods:{},created(){},})
</script>
</body>
</html>

后端共享值+跳转loginController.java

    @Value("${ServerHttpsUrl}")String httpsUrl;// 1.先到登陆页面@RequestMapping("/loginPage") // /user/loginPagepublic ModelAndView loginPage(){ModelAndView mv = new ModelAndView("user/login");mv.addObject("httpsUrl", httpsUrl + "/user/login");return mv;}

配置文件设置url

server:ssl:key-store: classpath:client.p12key-store-password: 12345678key-store-type: PKCS12key-alias: client# https的访问端口port: 8443# 部署服务器的配置
ServerHttpsUrl: https://localhost:8443 # https的url
ServerHttpUrl: http://localhost:8080 # http的url
ServerHttpPort: 8080 # http的端口号
LoginPassword: Admin@1a2 # 登陆的密码spring:mvc:static-path-pattern: /**resources:static-locations: classpath:/static/thymeleaf:prefix: classpath:/templates/check-template-location: truecache: falsesuffix: .html #模板后缀encoding: UTF-8 #编码mode: HTML #模板servlet:content-type: text/html

问题:在https协议下,发送axios请求没反应

问题如下:

在这里插入图片描述

解决方案一:用form表单

后端,用户名和密码正确后,重定向到index.html页面

// form表单下重定向到indexPage页面
response.sendRedirect(httpsUrl+“/user/indexPage”);

package com.shanxi.gis.controller;import com.shanxi.gis.entity.ResData;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.CrossOrigin;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.servlet.ModelAndView;import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;
import java.util.Objects;@Controller
@RequestMapping("/user")
@CrossOrigin // 允许跨域
public class LoginController {@Value("${ServerHttpsUrl}")String httpsUrl;@Value("${LoginPassword}")String loginPassword;// 1.先到登陆页面@RequestMapping("/loginPage") // /user/loginPagepublic ModelAndView loginPage(){ModelAndView mv = new ModelAndView("user/login");mv.addObject("httpsUrl", httpsUrl + "/user/login");return mv;}// 2.处理前端的axios请求@AutowiredHttpSession session; // TODO:保存用户名到session@RequestMapping("/login")@ResponseBodypublic ResData login(String username,String password, HttpServletResponse response) throws IOException {System.out.println(username +"//"+ password);if (Objects.equals(username, "") || username==null ||Objects.equals(password, "") || password==null){return new ResData(1001, "必填项为空", null);}if (!"admin".equals(username) || !loginPassword.equals(password)){return new ResData(1002, "用户名|密码错误", null);}session.setAttribute("user",username); // TODO:set进session// form表单下重定向到indexPage页面response.sendRedirect(httpsUrl+"/user/indexPage");return new ResData(200, "ok", null);}// 3.登陆成功到index页面@RequestMapping("/indexPage")public String loginIndex(){return "gis/index";}
}

前端发送form表单

<form :action=url method=“post”>

<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head><meta charset="UTF-8"><title>登陆页面</title><link rel="stylesheet" href="/bootstrap/css/bootstrap.min.css"><script src="/js/jquery-3.5.1.js"></script><script src="/bootstrap/js/bootstrap.js"></script><script src="/js/axios.min.js"></script><script src="/js/vue.min-v2.5.16.js"></script><link rel="stylesheet" href="/css/login.css">
</head><body><div class="login-container" id="app"><h2>欢迎登录</h2>
<!--        "https://localhost:8443/user/login"--><form :action=url method="post"><label for="username">用户名:</label><input type="text" id="username" v-model="username" placeholder="请输入用户名" required name="username"><label for="password">密码:</label><input type="password" id="password" v-model="password" placeholder="请输入密码" required name="password"><input type="submit" value="登录" @click="loginBtn" class="btn btn-primary btn-block"></form></div><script>let app = new Vue({el:"#app",data:{username:"",password:"",url:"[[${httpsUrl}]]",},methods:{loginBtn(){console.log("send----")let params = new URLSearchParams();params.append("username",this.username)params.append("password",this.password)// axios.post("/user/login",params)axios.post("/user/login",params).then(response=>{console.log("axios")if (response.data.code==200){// alert("登陆成功")location.href= "/user/indexPage"}else {alert(response.data.msg)}})}},created(){},})</script></body>
</html>

项目的打包部署

1.template might not exist or might not be accessible by any of the configured Template Resolvers

错误描述:

在idea中进行测试,所有功能都可以实现,尝试打包成jar包后运行,进入首页后没有显示用户信息页面,报500异常,后台显示Error resolving template [/user/info], template might not exist or might not be accessible by any of the configured Template Resolvers

报错信息:

在这里插入图片描述

2023-07-08 10:16:11.298 ERROR 28396 — [p-nio-80-exec-1] o.a.c.c.C.[.[.[/].[dispatcherServlet] : Servlet.service() for servlet [dispatcherServlet] in context with path [] threw exception [Request processing failed; nested exception is org.thymeleaf.exceptions.TemplateInputException: Error resolving template [/user/info], template might not exist or might not be accessible by any of the configured Template Resolvers] with root cause

org.thymeleaf.exceptions.TemplateInputException: Error resolving template [/user/info], template might not exist or might not be accessible by any of the configured Template Resolvers

在这里插入图片描述

解决方案一:

@RequestMapping("/infoPage")
public String infoPage(){return "/user/info";
}

跳转页面去掉第一个反斜杠,改为如下

    @RequestMapping("/infoPage")public String infoPage(){return "user/info";}

在这里插入图片描述

thymeleaf + Spring Boot 在开发环境正常,但用jar运行时报错 Error resolving template template might not exist or might not be accessible;

就可以了

解决方案二:

spring.thymeleaf.cache=false
spring.thymeleaf.prefix=classpath:/templates/
spring.thymeleaf.suffix=.html

改成

spring.thymeleaf.cache=false
spring.thymeleaf.prefix=classpath:/templates
spring.thymeleaf.suffix=.html
## spring相关的配置
spring:# 连接数据库datasource:type: com.alibaba.druid.pool.DruidDataSourcedriver-class-name: com.mysql.cj.jdbc.Driverurl: jdbc:mysql://127.0.0.1:3306/javaweb?useSSL=false&serverTimezone=Asia/Shanghai&allowPublicKeyRetrieval=trueusername: rootpassword: 123## 设置上传文件大小servlet:multipart:max-file-size: 10MB # 设置单个文件最大大小为10MB# 另一种解决方案thymeleaf:cache: falseprefix: classpath:/templatessuffix: .html

此时所有跳页面的都要加反斜杠

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/7457.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

工业物联网网关让PLC数据手机端监控和报警更加简单

在传统的工厂管理中&#xff0c;我们想要看到现场设备的实时数据&#xff0c;必须在控制室内通过工控机、电脑、触摸屏等这些上位机设备才能看到&#xff0c;同理PLC维护也需要工程师在现场进行编程调试工作&#xff0c;非常不方便。 随着工业物联网的发展&#xff0c;作为设备…

“学习嵌入式开发:明确目标,提升技能“

嵌入式领域涵盖广泛&#xff0c;不可能一次性掌握所有知识。因此&#xff0c;明确学习目标和方向非常重要。选择感兴趣且与职业发展相关的领域进行深入学习是明智之举。 嵌入式技术在不断发展&#xff0c;过去与现在存在差异。选择学习当前行业的主流技术和趋势是明智选择。掌…

抖音矩阵系统源码开发搭建部署分享

一、 功能开发设计 &#xff08;1&#xff09;数据概览&#xff1a;账号&#xff0c;视频top10数据统计 &#xff08;2&#xff09;AI视频创意&#xff1a;原创视频批量剪辑&#xff0c;阶乘算法&#xff0c;去重原理 &#xff08;3&#xff09;同城拓客&#xff1a;线下门店…

【Java基础教程】(四十六)IO篇 · 下:System类对IO的支持:错误输出、信息输出、系统输入,字符缓冲流、扫描流和对象序列化流~

Java基础教程之IO操作 下 &#x1f539;本节学习目标1️⃣ System类对 IO 的支持1.1 错误输出&#xff1a;System.err1.2 信息输出&#xff1a;System.out1.3 系统输入&#xff1a;System. in 2️⃣ 字符缓冲流&#xff1a;BufferedReader3️⃣ 扫描流&#xff1a;Scanner4️⃣…

两个数组的dp问题(2)--动态规划

一)交错字符串: 97. 交错字符串 - 力扣&#xff08;LeetCode&#xff09; 一)确定一个状态标识: 如果我选择s1的一段区间&#xff0c;再进行选择s2得一段区间那么s3这个字符串的长度就已经固定了 预处理:在s1字符串s2字符串和s3字符串前面加上一个虚拟字符&#xff0c;让下标从…

【MyBatis-Plus 进阶学习笔记】

MyBatis-Plus 进阶学习笔记记录 一、 MyBatis Plus 七大功能0. 数据准备1. 逻辑删除2. 自动填充2.1 优化1 自动填充 有的类没有更新和创建时间字段2.2 优化2 自己设置时间时填充自己设置的&#xff0c;不设置时自动填充 3. 乐观锁插件 注&#xff1a;wrapper不能服用4. 性能分析…

网安高级笔记1

html实体编码 HTML实体编码&#xff0c;格式 以&符号开头&#xff0c;以;分号结尾的 HTML 中的预留字符必须被替换为字符实体 在 HTML 中不能使用小于号&#xff08;<&#xff09;和大于号&#xff08;>&#xff09;&#xff0c;这是因为浏览器会误认为它们是…

HTML中的焦点管理

前言 焦点作为页面交互中的重要一环&#xff0c;涉及到的知识点也比较多&#xff0c;有必要做一个统一的总结。 HTML 中的可获取焦点的元素 具有 href 属性的 HTMLAnchorElement/HTMLAreaElement非禁用态的 HTMLInputElement/HTMLSelectElement/HTMLTextAreaElement/HTMLBut…

Docker——compose单机容器集群编排

Docker——compose单机容器集群编排 一、Docker-compose概述1.为何需要Docker-compose2.Docker-compose 的特征3.Docker-compose 的优势4.Docker-compose 的劣势5.Docker-compose 的生产环境 二、Docker Compose 环境安装三、YAML 文件格式及编写注意事项四、Docker Compose配置…

vscode使用g++编译.c文件或.cpp文件

vscode是一个跨平台、轻量级、插件非常丰厚的IDE&#xff0c;这里介绍在vscode里使用g来编译.cpp文件。g也叫GCC, 在Window中&#xff0c;是使用MinGW方式实现g的&#xff0c;它分为32位和64位2个版本&#xff0c;其中&#xff0c;MinGW-64是64位的&#xff0c;MinGW-32是32位的…

ConcurrentHashMap 相比于 HashMap 的优势

ConcurrentHashMap 使用每个链表头节点作为锁对象, 把一把大锁转换成多把小锁, 大大缩小了锁冲突的概率 HashTable 是给整个 Hash 表加锁, 因此只要有线程抢到了锁其他线程就得阻塞等待. ConcurrentHashMap 是对每个链表加锁, 因此只要不是对同一个链表进行修改就不会阻塞, 大…

【微信小程序】使用iView组件库的ActionSheet组件实现底部选择功能

效果1 效果2 要在微信小程序中使用iView组件库的ActionSheet组件&#xff0c;可以按照以下步骤进行&#xff1a; 首先&#xff0c;确保已经引入了iView组件库的样式和脚本文件。可以在app.wxss中引入iView的样式文件&#xff1a; import "/path/to/iview/weapp/dist/sty…

Ubuntu22.04部署K8s集群

Ubuntu22.04部署K8s集群 一、基础环境准备1.1 VMware Workstation Pro 17.01.2 Ubuntu22.04 二、系统环境配置2.1 设置Master与工作节点的机器名称及配置2.2 解析主机2.3 虚拟内存swap分区关闭2.4 开启IPv4转发2.5 设置时间同步2.6 开启防火墙的端口&#xff08;可选&#xff0…

linux下 UART串口相关

RS232的串口设备在linux 上会被识别为 /dev/ttyS* 或者 ttymxc* 一、串口简介 操作串口我们一般通过以下指令&#xff1a; 1、查看串口波特率等信息&#xff1a; stty -F /dev/ttyS0 -a #ttyS0为要查看的串口 2、设置串口参数&#xff1a; stty -F /dev/ttyS0 ispeed 115…

微信小游戏个人开发者上架:从注册到上线的详细步骤

微信小游戏个人开发者上架&#xff1a;从注册到上线的详细步骤 一&#xff0c;注册小程序账号1.1 微信公众平台1.2 填写信息1.3 绑定管理 二&#xff0c;打包步骤2.1 工具准备2.2 关于Unity版本2.3 打包详解 三&#xff0c;提包步骤3.1 填写用户隐私3.2 完善开发者自查3.3 游戏…

5.string变量-读取一行

C里面的读一行的用法。getline&#xff08;cin,addr&#xff09;; 从标准输入设备cin&#xff0c;读取一行字符串保存到字符串变量addr中 如果用户直接回车什么都不读取就没有任何数据输入 读一行直到遇到回车符&#xff0c;注意不包括回车符。 判断字符串是不是空的 addr.em…

Cron 选择器

// 定义一个名为 cron 的新组件 Vue.component(cron, {name: cron,props: [data],data() {return {second: {cronEvery: ,incrementStart: 3,incrementIncrement: 5,rangeStart: ,rangeEnd: ,specificSpecific: [],},minute: {cronEvery: ,incrementStart: 3,incrementIncremen…

2023年一建学霸笔记

考点:单方取消或辞去委托承担的民事责任女《民法典》规定&#xff0c;因解除合同造成对方损失的&#xff0c;除不可归责于该当事人的事由外&#xff0c;无偿委托合同的解除方应当赔偿因解除时间不当造成的直接损失&#xff0c;有偿委托合同的解除方应当赔偿对方的直接损失和合同…

简单理解TCP,UDP,HTTP

我们都知道TCP、UDP、HTTP内部有很复杂的过程&#xff0c;很多人没办法理解的那么深&#xff0c;只想知道这是个什么鬼。 1、TCP、UDP、HTTP 是什么? TCP/IP是个协议组&#xff0c;可分为三个层次&#xff1a;网络层、传输层和应用层。在网络层有IP协议、ICMP协议、ARP协议、…

关于云服务器ECS、宝塔的安装配置以及图床的使用

一、阿里云服务器的申请以及宝塔的安装 安装配置服务器的原理&#xff1a; step1&#xff1a;地址栏输入阿里云服务器官网地址 step2&#xff1a;在首页依次点击以下内容&#xff1a; step3&#xff1a;选择立即购买&#xff0c;并填写以下内容&#xff1a; step4&#xff1a…