当前位置: 首页 > news >正文

Springboot JSP项目如何以war、jar方式运行

文章目录

  • 一,序
  • 二,样例代码
    • 1,代码结构
    • 2,完整代码备份
  • 三,准备工作
    • 1. pom.xml 引入组件
    • 2. application.yml 指定jsp配置
  • 四,war方式运行
    • 1. 修改pom.xml文件
    • 2. mvn执行打包
  • 五,jar方式运行
    • 1. 修改pom-jsp-jar.xml文件
    • 2. 修改 spring-boot-maven-plugin添加版本号
    • 3. 添加资源文件配置(`Jar运行必须`)
    • 4. mvn执行打包
  • 六,war、jar方式运行配置文件区别

一,序

Spring Boot 官方不推荐使用JSP来作为视图,但是仍有部分项目使用了JSP视图,Springboot JSP项目运行方式有war、Jar两种方式。

二,样例代码

1,代码结构

在这里插入图片描述

2,完整代码备份

如何使用下面的备份文件恢复成原始的项目代码,请移步查阅:神奇代码恢复工具

//goto Dockerfile
#基础镜像
FROM openjdk:8-jre-alpineRUN ln -sf /usr/share/zoneinfo/Asia/Shanghai /etc/localtime && echo 'Asia/Shanghai' >/etc/timezone#把你的项目war包引入到容器的root目录下
COPY target/*.war /app.warCMD ["--server.port=8080"]#项目的启动方式
#ENTRYPOINT ["java","-Xmx400m","-Xms400m","-Xmn150m","-Xss1024k","-jar","/app.war", "--spring.profiles.active=prod"]
ENTRYPOINT ["java","-jar","/app.war"]
//goto pom-jsp-jar.xml
<?xml version="1.0" encoding="UTF-8"?>
<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.fly</groupId><artifactId>docker-demo</artifactId><version>0.0.1</version><packaging>jar</packaging><parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.2.4.RELEASE</version><relativePath /></parent><properties><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding><maven.build.timestamp.format>yyyyMMdd-HH</maven.build.timestamp.format><docker.hub>registry.cn-shanghai.aliyuncs.com</docker.hub><java.version>1.8</java.version><skipTests>true</skipTests></properties><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><exclusions><exclusion><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-logging</artifactId></exclusion></exclusions></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-log4j2</artifactId></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-context-support</artifactId><scope>compile</scope></dependency><dependency><groupId>org.apache.commons</groupId><artifactId>commons-lang3</artifactId></dependency><dependency><groupId>com.google.guava</groupId><artifactId>guava</artifactId><version>20.0</version></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-devtools</artifactId><optional>true</optional></dependency><!-- JSTL for JSP --><dependency><groupId>javax.servlet</groupId><artifactId>jstl</artifactId></dependency><!-- For JSP compilation --><dependency><groupId>org.apache.tomcat.embed</groupId><artifactId>tomcat-embed-jasper</artifactId><scope>provided</scope></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><scope>provided</scope></dependency></dependencies><build><finalName>${project.artifactId}-${project.version}</finalName><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId><version>1.4.2.RELEASE</version></plugin><!-- 添加docker-maven插件 --><plugin><groupId>io.fabric8</groupId><artifactId>docker-maven-plugin</artifactId><version>0.41.0</version><executions><execution><phase>package</phase><goals><goal>build</goal><goal>push</goal><goal>remove</goal></goals></execution></executions><configuration><!-- 连接到带docker环境的linux服务器编译image --><!-- <dockerHost>http://192.168.182.10:2375</dockerHost> --><!-- Docker 推送镜像仓库地址 --><pushRegistry>${docker.hub}</pushRegistry><images><image><!--推送到私有镜像仓库,镜像名需要添加仓库地址 --><name>${docker.hub}/00fly/${project.artifactId}:${project.version}-UTC-${maven.build.timestamp}</name><!--定义镜像构建行为 --><build><dockerFileDir>${project.basedir}</dockerFileDir></build></image><image><name>${docker.hub}/00fly/${project.artifactId}:${project.version}</name><build><dockerFileDir>${project.basedir}</dockerFileDir></build></image></images></configuration></plugin></plugins><resources><resource><directory>src/main/java</directory><excludes><exclude>**/*.java</exclude></excludes></resource><resource><directory>src/main/resources</directory><includes><include>**/**</include></includes></resource><resource><directory>src/main/webapp</directory><includes><include>**/**</include></includes><targetPath>META-INF/resources</targetPath></resource></resources></build>
</project>
//goto pom.xml
<?xml version="1.0" encoding="UTF-8"?>
<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.fly</groupId><artifactId>docker-demo</artifactId><version>0.0.1</version><packaging>war</packaging><parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.2.4.RELEASE</version><relativePath /></parent><properties><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding><maven.build.timestamp.format>yyyyMMdd-HH</maven.build.timestamp.format><docker.hub>registry.cn-shanghai.aliyuncs.com</docker.hub><java.version>1.8</java.version><skipTests>true</skipTests></properties><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><exclusions><exclusion><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-logging</artifactId></exclusion></exclusions></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-log4j2</artifactId></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-context-support</artifactId><scope>compile</scope></dependency><dependency><groupId>org.apache.commons</groupId><artifactId>commons-lang3</artifactId></dependency><dependency><groupId>com.google.guava</groupId><artifactId>guava</artifactId><version>20.0</version></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-devtools</artifactId><optional>true</optional></dependency><!-- JSTL for JSP --><dependency><groupId>javax.servlet</groupId><artifactId>jstl</artifactId></dependency><!-- For JSP compilation --><dependency><groupId>org.apache.tomcat.embed</groupId><artifactId>tomcat-embed-jasper</artifactId><scope>provided</scope></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><scope>provided</scope></dependency></dependencies><build><finalName>${project.artifactId}-${project.version}</finalName><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId></plugin><!-- 添加docker-maven插件 --><plugin><groupId>io.fabric8</groupId><artifactId>docker-maven-plugin</artifactId><version>0.41.0</version><executions><execution><phase>package</phase><goals><goal>build</goal><goal>push</goal><goal>remove</goal></goals></execution></executions><configuration><!-- 连接到带docker环境的linux服务器编译image --><!-- <dockerHost>http://192.168.182.10:2375</dockerHost> --><!-- Docker 推送镜像仓库地址 --><pushRegistry>${docker.hub}</pushRegistry><images><image><!--推送到私有镜像仓库,镜像名需要添加仓库地址 --><name>${docker.hub}/00fly/${project.artifactId}:${project.version}-UTC-${maven.build.timestamp}</name><!--定义镜像构建行为 --><build><dockerFileDir>${project.basedir}</dockerFileDir></build></image><image><name>${docker.hub}/00fly/${project.artifactId}:${project.version}</name><build><dockerFileDir>${project.basedir}</dockerFileDir></build></image></images></configuration></plugin></plugins><resources><resource><directory>src/main/java</directory><excludes><exclude>**/*.java</exclude></excludes></resource><resource><directory>src/main/resources</directory><includes><include>**/**</include></includes></resource><resource><directory>src/main/webapp</directory><includes><include>**/**</include></includes><targetPath>META-INF/resources</targetPath></resource></resources></build>
</project>
//goto shell\docker\1-n\docker-compose.yml
version: '3'
services:demo-dk:image: demo-dk:1.0build:context: .dockerfile: Dockerfilecontainer_name: demo_dkdeploy:resources:limits:cpus: '0.50'memory: 300Mreservations:memory: 128Mports:- 8085:8085- 8086:8086- 8087:8087logging:driver: json-fileoptions:max-size: 5mmax-file: '1'
//goto shell\docker\1-n\Dockerfile
#基础镜像
FROM openjdk:8-jre-alpineRUN ln -sf /usr/share/zoneinfo/Asia/Shanghai /etc/localtime
RUN echo 'Asia/Shanghai' >/etc/timezone#把你的项目war包引入到容器的root目录下
COPY *.war /app.warRUN echo \
"#!/bin/sh\n"\
"nohup java -jar /app.war --server.port=8085 &\n"\
"nohup java -jar /app.war --server.port=8086 &"\
>> /start.shRUN chmod +x /start.sh
CMD nohup sh -c "/start.sh && java -jar /app.war --server.port=8087"
//goto shell\docker\1-n\restart-docker.sh
#!/bin/bash
docker-compose down && docker system prune -f && docker-compose build && docker-compose up -d
//goto shell\docker\1-n\start.sh
#!/bin/bash# 命令后加入 & ,保持程序后台持续运行
nohup java -jar /app.war --server.port=8085 & 
nohup java -jar /app.war --server.port=8086 &# 死循环,保持docker前台运行
while [[ true ]]; dosleep 1
done//goto shell\docker\1-n\start2.sh
#!/bin/bash# 命令后加入 & ,保持程序后台持续运行
nohup java -jar /app.war --server.port=8085 & 
java -jar /app.war --server.port=8086
//goto shell\docker\scale\docker-compose.yml
version: '3.8'
services:nginx:image: nginx:alpinecontainer_name: nginxdeploy:resources:limits:cpus: '0.80'memory: 50Mreservations:memory: 30Mports:- 80:80restart: on-failurevolumes:- ./nginx/nginx.conf:/etc/nginx/nginx.conf- ./nginx/conf.d:/etc/nginx/conf.dlogging:driver: 'json-file'options:max-size: '5m'max-file: '1'  depends_on:- demodemo:hostname: demoimage: registry.cn-shanghai.aliyuncs.com/00fly/docker-demo:0.0.1deploy:resources:limits:cpus: '0.80'memory: 200Mreservations:memory: 100Menvironment:- JAVA_OPTS=-server -Xms128m -Xmx128m -Djava.security.egd=file:/dev/./urandomrestart: on-failurelogging:driver: 'json-file'options:max-size: '5m'max-file: '1'
//goto shell\docker\scale\restart.sh
#!/bin/bash
docker-compose --compatibility restart
//goto shell\docker\scale\scale.sh
#!/bin/bash
docker-compose --compatibility up -d --scale nginx=1 --scale demo=3 
//goto shell\docker\scale\stop.sh
#!/bin/bash
docker-compose down
//goto shell\docker\single\docker-compose.yml
version: '3'
services:docker-demo:image: registry.cn-shanghai.aliyuncs.com/00fly/docker-demo:0.0.1container_name: docker-demodeploy:resources:limits:cpus: '0.80'memory: 300Mreservations:cpus: '0.05'memory: 100Mports:- 80:8080restart: on-failurelogging:driver: json-fileoptions:max-size: 5mmax-file: '1'
//goto shell\docker\single\restart.sh
#!/bin/bash
docker-compose down && docker system prune -f && docker-compose up -d
//goto shell\docker\single\stop.sh
#!/bin/bash
docker-compose down
//goto shell\init.sh
#!/bin/bash
dos2unix *
dos2unix */*
dos2unix */*/*chmod +x *.sh
chmod +x */*.sh
//goto shell\reload-jar.sh
#!/bin/bash
# get pidpname="springboot-cache.jar"
echo  -e  "jar-name=$pname\r\n"get_pid(){pid=`ps -ef | grep $pname | grep -v grep | awk '{print $2}'`echo "$pid"
}ps -ef|grep $pnamePID=$(get_pid)
if [ -z "${PID}" ] 
thenecho -e "\r\nJava Application already stop!"
elseecho -e '\r\nkill -9  '${PID} '\r\n'kill -9 ${PID}echo -e "Java Application is stop!"
firm -rf info.logecho -e "\r\nJava Application will startup!\r\n"
jar_path=`find .. -name $pname`#echo "jarfile=$jar_path"nohup java -jar $jar_path --server.port=8181 >>./info.log 2>&1 &ps -ef|grep $pname
//goto shell\reload-war.sh
#!/bin/bash
# get pid
get_pid(){pname="docker-demo.war"pid=`ps -ef | grep $pname | grep -v grep | awk '{print $2}'`echo "$pid"
}ps -ef|grep docker-demo.warPID=$(get_pid)
if [ -z "${PID}" ] 
thenecho -e "\r\nJava Application already stop!"
elseecho -e '\r\nkill -9  '${PID}kill -9 ${PID}echo -e "Java Application is stop!"
firm -rf info.logecho -e "\r\nJava Application will startup!"
nohup java -jar /work/demo/web/docker-demo.war -Xms64m -Xmx64m --server.port=8181 >>./info.log 2>&1 &
#sleep 2s
#nohup java -jar /work/demo/web/docker-demo.war -Xms64m -Xmx64m --server.port=8182 >>./info.log 2>&1 &ps -ef|grep docker-demo.warecho -e "please visit: https://test.00fly.online/user/"#tail -f info.log
//goto shell\start-service.sh
#!/bin/bash 
java -jar docker-demo.war -Xms64m -Xmx64m --server.port=8083 &
//goto src\main\java\com\fly\common\AnnotationHelper.java
package com.fly.common;import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;import org.springframework.context.ApplicationContext;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;import lombok.extern.slf4j.Slf4j;/*** * 注解处理工具类* * @author 00fly* @version [版本号, 2020-04-16]* @see [相关类/方法]* @since [产品/模块版本]*/
@Slf4j
public class AnnotationHelper
{private static List<String> cacheList = new ArrayList<>();private AnnotationHelper(){super();}/*** 得到类上面的注解信息* * @param scannerClass* @param allowInjectClass* @return*/private static Annotation getClassAnnotation(Class<?> scannerClass, Class<? extends Annotation> allowInjectClass){if (!scannerClass.isAnnotationPresent(allowInjectClass)){return null;}return scannerClass.getAnnotation(allowInjectClass);}/*** 使用Java反射得到注解的信息* * @param annotation* @param methodName* @return Exception*/private static Object getAnnotationInfo(Annotation annotation, String methodName)throws Exception{if (annotation == null){return null;}Method declaredMethod = annotation.getClass().getDeclaredMethod(methodName, null);return declaredMethod.invoke(annotation, null);}/*** 从上下文获取 Controller注解类的 RequestMapping注解url信息* * @param applicationContext* @return* @throws Exception* @see [类、类#方法、类#成员]*/public static List<String> getRequestMappingURL(ApplicationContext applicationContext)throws Exception{if (cacheList.isEmpty()){synchronized (AnnotationHelper.class){List<String> list = new ArrayList<>();Map<String, Object> map = applicationContext.getBeansWithAnnotation(Controller.class);for (String key : map.keySet()){Class<?> clazz = map.get(key).getClass();Annotation classAnnotation = getClassAnnotation(clazz, RequestMapping.class);Object object = getAnnotationInfo(classAnnotation, "value");if (object != null){String[] array = (String[])object;log.info("{} -> {}", clazz.getName(), object);for (String it : array){if (!it.contains("$")){list.add(it);}}}}cacheList = list;}}return cacheList;}
}
//goto src\main\java\com\fly\common\AuthorizationInterceptor.java
package com.fly.common;import java.net.InetAddress;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.ApplicationContext;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.handler.HandlerInterceptorAdapter;/*** * AuthorizationInterceptor* * @author 00fly* @version [版本号, 2019年7月21日]* @see [相关类/方法]* @since [产品/模块版本]*/
@Component
public class AuthorizationInterceptor extends HandlerInterceptorAdapter
{@AutowiredApplicationContext applicationContext;@Value("${server.port}")String port;@AutowiredHttpSession session;@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)throws Exception{if (session.getAttribute("urls") == null){session.setAttribute("urls", AnnotationHelper.getRequestMappingURL(applicationContext));session.setAttribute("port", port);session.setAttribute("ip", InetAddress.getLocalHost().getHostAddress());}return true;}
}
//goto src\main\java\com\fly\common\WebMvcConfig.java
package com.fly.common;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;/*** * WebMvcConfig* * @author 00fly* @version [版本号, 2019年7月21日]* @see [相关类/方法]* @since [产品/模块版本]*/
@Configuration
public class WebMvcConfig implements WebMvcConfigurer
{@Autowiredprivate AuthorizationInterceptor authorizationInterceptor;@Overridepublic void addInterceptors(InterceptorRegistry registry){registry.addInterceptor(authorizationInterceptor).addPathPatterns("/user/**");}
}
//goto src\main\java\com\fly\demo\model\User.java
package com.fly.demo.model;import java.io.Serializable;import lombok.Data;
import org.hibernate.validator.constraints.Range;import javax.validation.constraints.NotBlank;
import javax.validation.constraints.NotNull;@Data
public class User implements Serializable
{private static final long serialVersionUID = -4621975403168327735L;private Long userId;@NotBlank(message = "用户名不能为空")private String userName;@NotNull(message = "年龄不能为空")@Range(min = 10, max = 60, message = "年龄必须在{min}-{max}")private Integer age;private String desc;
}
//goto src\main\java\com\fly\demo\service\impl\UserServiceImpl.java
package com.fly.demo.service.impl;import java.util.List;import javax.annotation.PostConstruct;import org.apache.commons.lang3.RandomUtils;
import org.springframework.stereotype.Service;import com.fly.demo.model.User;
import com.fly.demo.service.UserService;
import com.google.common.collect.Lists;import lombok.extern.slf4j.Slf4j;@Slf4j
@Service
public class UserServiceImpl implements  UserService
{private List<User> users = Lists.newArrayList();private Long sequence = 1L;/*** 初始化* * @see [类、类#方法、类#成员]*/@PostConstructpublic void init(){log.info("PostConstruct users");for (int i = 0; i < 5; i++){User user = new User();user.setUserId(sequence);user.setUserName("user_" + sequence);user.setAge(RandomUtils.nextInt(10, 40));user.setDesc("This is a user " + sequence);users.add(user);sequence++;}}private void insert(User user){log.info("insert user : {}", user);user.setUserId(sequence + 1);users.add(user);sequence++;}private void update(User user){log.info("update user id = {}", user.getUserId());for (User it : users){if (it.getUserId().equals(user.getUserId())){it.setUserName(user.getUserName());it.setAge(user.getAge());it.setDesc(user.getDesc());return;}}}/*** 新增/根据id更新* * @param user* @see [类、类#方法、类#成员]*/@Overridepublic void saveOrUpdate(User user){if (user.getUserId() != null){update(user);return;}insert(user);}@Overridepublic void delete(Long userId){log.info("delete user id = {}", userId);for (User user : users){if (user.getUserId().equals(userId)){users.remove(user);return;}}}@Overridepublic User get(Long userId){log.info("get user id = {}", userId);for (User user : users){if (user.getUserId().equals(userId)){return user;}}return null;}@Overridepublic List<User> list(){log.info("list users");return users;}
}
//goto src\main\java\com\fly\demo\service\UserService.java
package com.fly.demo.service;import java.util.List;import com.fly.demo.model.User;public interface UserService
{void saveOrUpdate(User user);void delete(Long userId);User get(Long userId);List<User> list();}
//goto src\main\java\com\fly\demo\web\IndexController.java
package com.fly.demo.web;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.GetMapping;import com.fly.common.AnnotationHelper;/*** * IndexController* * @author 00fly* @version [版本号, 2020-04-16]* @see [相关类/方法]* @since [产品/模块版本]*/
@Controller
public class IndexController
{@AutowiredApplicationContext applicationContext;/*** 首页* * @param model* @return* @throws Exception* @see [类、类#方法、类#成员]*/@GetMapping("/")public String index(Model model)throws Exception{model.addAttribute("urls", AnnotationHelper.getRequestMappingURL(applicationContext));return "/index";}
}
//goto src\main\java\com\fly\demo\web\UserApi.java
package com.fly.demo.web;import javax.validation.Valid;import org.apache.commons.lang3.RandomStringUtils;
import org.apache.commons.lang3.RandomUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.validation.Errors;
import org.springframework.validation.ObjectError;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.ModelAttribute;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;import com.fly.demo.model.User;
import com.fly.demo.service.UserService;import lombok.extern.slf4j.Slf4j;@Slf4j
@Controller
@RequestMapping("/user")
public class UserApi
{@Autowiredprivate UserService userService;/*** 测试添加缓存,后续读取从缓存中取* * @param userId* @return* @see [类、类#方法、类#成员]*/@GetMapping("get")public User get(Long userId){return userService.get(userId);}/*** 新增/更新数据* * @param user* @return* @see [类、类#方法、类#成员]*/@PostMapping("/add")public String add(@Valid @ModelAttribute("item") User user, Errors errors, Model model){if (errors.hasErrors()){StringBuilder errorMsg = new StringBuilder();for (ObjectError error : errors.getAllErrors()){errorMsg.append(error.getDefaultMessage()).append(" ");}if (errorMsg.length() > 0){log.info("errors message={}", errorMsg);}model.addAttribute("items", userService.list());return "/user/show";}userService.saveOrUpdate(user);return "redirect:/user/list";}/*** 测试删除缓存*/@GetMapping("/delete/{id}")public String delete(@PathVariable Long id){userService.delete(id);return "redirect:/user/list";}@GetMapping({"/", "/list"})public String list(Model model){User user = new User();if (RandomUtils.nextInt(1, 10) > 1){user.setUserName(RandomStringUtils.randomNumeric(5));user.setAge(RandomUtils.nextInt(10, 70));user.setDesc(RandomStringUtils.randomNumeric(5));}model.addAttribute("item", user);model.addAttribute("items", userService.list());return "/user/show";}/*** 编辑数据* * @param id* @param model* @return* @see [类、类#方法、类#成员]*/@GetMapping("/update/{id}")public String update(@PathVariable Long id, Model model){model.addAttribute("item", userService.get(id));model.addAttribute("items", userService.list());return "/user/show";}
}
//goto src\main\java\com\fly\DemoBootApplication.java
package com.fly;import java.net.InetAddress;import org.apache.commons.lang3.SystemUtils;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.CommandLineRunner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication;
import org.springframework.context.annotation.Bean;import lombok.extern.slf4j.Slf4j;@Slf4j
@SpringBootApplication
public class DemoBootApplication
{@Value("${server.port}")Integer port;public static void main(String[] args){SpringApplication.run(DemoBootApplication.class, args);}@Bean@ConditionalOnWebApplicationCommandLineRunner openBrowser(){return args -> {if (SystemUtils.IS_OS_WINDOWS && port > 0){log.info("★★★★★★★★  now open Browser ★★★★★★★★ ");String ip = InetAddress.getLocalHost().getHostAddress();String url = "http://" + ip + ":" + port;Runtime.getRuntime().exec("cmd /c start /min " + url);}};}
}
//goto src\main\resources\application.yml
server:port: 8080
spring:mvc:view:prefix: /WEB-INF/viewssuffix: .jsp
//goto src\main\webapp\WEB-INF\views\index.jsp
<%@ page language="java" contentType="text/html; charset=utf-8" pageEncoding="utf-8"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/functions" prefix="fn"%>
<table><tr><th colspan="${fn:length(urls)+1}">Navigate</th></tr><tr><c:forEach var="item" items="${urls}"><td><a href="${pageContext.request.contextPath}${item}/">${item}</a></td></c:forEach></tr>
</table>
//goto src\main\webapp\WEB-INF\views\user\show.jsp
<%@ page contentType="text/html;charset=UTF-8"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/functions" prefix="fn"%>
<%@ taglib uri="http://www.springframework.org/tags" prefix="spring"%>
<%@ taglib uri="http://www.springframework.org/tags/form" prefix="form"%>
<meta charset="utf-8">
<html>
<head>
<title>00fly功能演示</title>
<style>
body {margin: 10;font-size: 62.5%;line-height: 1.5;
}.blue-button {background: #25A6E1;padding: 3px 20px;color: #fff;font-size: 12px;border-radius: 2px;-moz-border-radius: 2px;-webkit-border-radius: 4px;border: 1px solid #1A87B9
}table {width: 70%;
}th {background: SteelBlue;color: white;
}td, th {border: 1px solid gray;font-size: 12px;text-align: left;padding: 5px 10px;overflow:hidden; white-space:nowrap; text-overflow:ellipsis;max-width: 200px;
}
</style>
</head>
<script type="text/javascript">function formReset() {location.href="/user/list";}
</script>
<body><center><table><tr><th colspan="${fn:length(urls)+1}">Navigate</th></tr><tr><c:forEach var="item" items="${urls}"><td><a href="${pageContext.request.contextPath}${item}/">${item}</a></td></c:forEach></tr></table>    <form:form method="post" modelAttribute="item" action="${pageContext.request.contextPath}/user/add"><table><tr><th colspan="3">Add or Edit Item</th><form:hidden path="userId" /></tr><tr><td><form:label path="userName">userName:</form:label></td><td><form:input path="userName" size="30" maxlength="30"></form:input></td><td>String <form:errors path="userName" cssStyle="color:red" /></td></tr><tr><td><form:label path="age">age:</form:label></td><td><form:input path="age" size="30" maxlength="30"></form:input></td><td>Integer <form:errors path="age" cssStyle="color:red" /></td></tr><tr><td><form:label path="desc">desc:</form:label></td><td><form:input path="desc" size="30" maxlength="30"></form:input></td><td>String <form:errors path="desc" cssStyle="color:red" /></td></tr><tr><td colspan="3" style="text-align: center;"><input type="submit" class="blue-button" /> <input type="reset" class="blue-button" onclick="formReset()" /></td></tr></table></form:form><h3>Data List</h3><c:if test="${!empty items}"><table><tr><th width="5">序号</th><th width="5">Edit</th><th width="5">Delete</th><th width="100">userId</th><th width="150">userName</th><th width="150">age</th><th width="150">desc</th></tr><c:forEach items="${items}" var="it" varStatus="vs"><tr><td>${vs.count}</td><td><a href="<c:url value='/user/update/${it.userId}' />">Edit</a></td><td><a href="<c:url value='/user/delete/${it.userId}' />">Delete</a></td><td>${it.userId}</td><td>${it.userName}</td><td>${it.age}</td><td>${it.desc}</td></tr></c:forEach></table></c:if><h3>应用端口:<font color="red">${port}</font></h3><h3>Server IP<font color="red">${ip}</font></h3></center>
</body>
</html>

三,准备工作

1. pom.xml 引入组件

		<!-- JSTL 支持,按需引入 --><dependency><groupId>javax.servlet</groupId><artifactId>jstl</artifactId></dependency><!-- JSP 编译支持,必须引入--><dependency><groupId>org.apache.tomcat.embed</groupId><artifactId>tomcat-embed-jasper</artifactId><scope>provided</scope></dependency>

2. application.yml 指定jsp配置

application.yml

spring:mvc:view:prefix: /WEB-INF/viewssuffix: .jsp

四,war方式运行

1. 修改pom.xml文件

	<packaging>war</packaging>

2. mvn执行打包

mvn clean package

执行后会在target目录生成war包,拷贝出来后运行

java -jar docker-demo-0.0.1.war

浏览器访问: http://127.0.0.1:8080/user/ 界面如下:
在这里插入图片描述

五,jar方式运行

复制 pom.xml重命名为 pom-jsp-jar.xml

1. 修改pom-jsp-jar.xml文件

	<packaging>jar</packaging>

2. 修改 spring-boot-maven-plugin添加版本号

注意只能使用1.4.2.RELEASE,其他版本不行

			<plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId><version>1.4.2.RELEASE</version></plugin>

3. 添加资源文件配置(Jar运行必须

		<resources><resource><directory>src/main/java</directory><excludes><exclude>**/*.java</exclude></excludes></resource><resource><directory>src/main/resources</directory><includes><include>**/**</include></includes></resource><resource><directory>src/main/webapp</directory><includes><include>**/**</include></includes><targetPath>META-INF/resources</targetPath></resource></resources>

4. mvn执行打包

mvn clean package -f pom-jsp-jar.xml

执行后会在target目录生成 jar包,拷贝出来后运行

java -jar docker-demo-0.0.1.jar

六,war、jar方式运行配置文件区别

在这里插入图片描述
在这里插入图片描述


有任何问题和建议,都可以向我提问讨论,大家一起进步,谢谢!

-over-

相关文章:

Springboot JSP项目如何以war、jar方式运行

文章目录 一&#xff0c;序二&#xff0c;样例代码1&#xff0c;代码结构2&#xff0c;完整代码备份 三&#xff0c;准备工作1. pom.xml 引入组件2. application.yml 指定jsp配置 四&#xff0c;war方式运行1. 修改pom.xml文件2. mvn执行打包 五&#xff0c;jar方式运行1. 修改…...

系统架构设计师(第二版)学习笔记----层次式架构设计理论与实践

【原文链接】系统架构设计师&#xff08;第二版&#xff09;学习笔记----层次式架构设计理论与实践 文章目录 一、层次式体系结构概述1.1 软件体系结构的作用1.2 常用的层次式架构图1.3 层次式体系可能存在的问题点 二、表现层框架设计2.1 MVC模式2.1.1 MVC三层模式2.1.2 MVC设…...

Python之字符串详解

目录 一、字符串1、转义字符与原始字符串2、使用%运算符进行格式化 一、字符串 在Python中&#xff0c;字符串属于不可变、有序序列&#xff0c;使用单引号、双引号、三单引号或三双引号作为定界符&#xff0c;并且不同的定界符之间可以互相嵌套。 ‘abc’、‘123’、‘中国’…...

《视觉SLAM十四讲》-- 概述与预备知识

文章目录 01 概述与预备知识1.1 SLAM 是什么1.1.1 基本概念1.1.2 视觉 SLAM 框架1.1.3 SLAM 问题的数学表述 1.2 实践&#xff1a;编程基基础1.3 课后习题 01 概述与预备知识 1.1 SLAM 是什么 1.1.1 基本概念 &#xff08;1&#xff09;SLAM 是 Simultaneous Localization a…...

Java8 Stream API全面解析——高效流式编程的秘诀

文章目录 什么是 Stream Api?快速入门流的操作创建流中间操作filter 过滤map 数据转换flatMap 合并流distinct 去重sorted 排序limit 限流skip 跳过peek 操作 终结操作forEach 遍历forEachOrdered 有序遍历count 统计数量min 最小值max 最大值reduce 聚合collect 收集anyMatch…...

分享一下微信小程序里怎么开店

如何在微信小程序中成功开店&#xff1a;从选品到运营的全方位指南 一、引言 随着微信小程序的日益普及&#xff0c;越来越多的人开始尝试在微信小程序中开设自己的店铺。微信小程序具有便捷、易用、即用即走等特点&#xff0c;使得开店门槛大大降低。本文将详细介绍如何在微…...

uniapp小程序刮刮乐抽奖

使用canvas画布画出刮刮乐要被刮的图片&#xff0c;使用移动清除画布。 当前代码封装为刮刮乐的组件&#xff1b; vue代码&#xff1a; <template><view class"page" v-if"merchantInfo.cdn_static"><image class"bg" :src&q…...

Qt 窗口无法移出屏幕

1 使用场景 设计一个缩进/展开widget的效果&#xff0c;抽屉效果。 看到实现的方法有定时器里move窗口&#xff0c;或是使用QPropertyAnimation。 setWindowFlags(Qt::Dialog | Qt::FramelessWindowHint |Qt::X11BypassWindowManagerHint&#xff09;&#xff1b; 记得在移…...

java毕业设计基于springboot+vue线上教学辅助系统

项目介绍 本论文主要论述了如何使用JAVA语言开发一个线上教学辅助系统 &#xff0c;本系统将严格按照软件开发流程进行各个阶段的工作&#xff0c;采用B/S架构&#xff0c;面向对象编程思想进行项目开发。在引言中&#xff0c;作者将论述线上教学辅助系统的当前背景以及系统开…...

开源 Wiki 软件 wiki.js

wiki.js简介 最强大、 可扩展的开源Wiki 软件。使用 Wiki.js 美观直观的界面让编写文档成为一种乐趣&#xff01;根据 AGPL-v3 许可证发布。 官方网站&#xff1a;https://js.wiki/ 项目地址&#xff1a;https://github.com/requarks/wiki 主要特性&#xff1a; 随处安装&a…...

STM32基本定时器中断

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、STM32定时器的结构&#xff1f;1. 51定时器的结构1.1如何实现定时1s的功能&#xff1f; 2. stm32定时器的结构2.1 通用定时器 二、使用步骤1.开启时钟2.初始…...

学习历程_基础_精通部分_达到手搓的程度

1. 计算机网络(更新版) 1.1 计算机网络-43题 1.2 2. 操作系统(更新版) 3. ACM算法(更新版) 4. 数据库&#xff08;更新版&#xff09; 5. 业务开发算法&#xff08;更新版&#xff09; 6. 分布式类&#xff08;更新版&#xff09; 7. 设计模式&#xff08;更新版&#xff…...

Redis中的List类型

目录 List类型的命令 lpush lpushx rpush lrange lpop rpop lindex linsert llen lrem ltrim lset 阻塞命令 阻塞命令的使用场景 1.针对一个非空的列表进行操作 2.针对一个空的列表进行操作 3.针对多个key进行操作. 内部编码 lisi类型的应用场景 存储(班级…...

3D模型格式转换工具HOOPS Exchange:如何将3D PDF转换为STEP格式?

3D CAD数据在制造、工程和设计等各个领域都扮演着重要的角色。为了促进不同软件应用程序之间的协作和互操作性&#xff0c;它通常以不同的格式进行交换。 HOOPS Exchange是一个强大的软件开发工具包&#xff0c;提供了处理和将3D CAD数据从一种格式转换为另一种格式的解决方案…...

DB-GPT介绍

DB-GPT介绍 引言DB-GPT项目简介DB-GPT架构关键特性私域问答&数据处理多数据源&可视化自动化微调Multi-Agents&Plugins多模型支持与管理隐私安全支持数据源 子模块DB-GPT-Hub微调参考文献 引言 随着数据量的不断增长和数据分析的需求日益增多&#xff0c;将自然语言…...

Java,面向对象,内部类

内部类的定义&#xff1a; 将一个类A定义在另一个类B里面&#xff0c;里面的那个类A就称为内部类&#xff08;InnerClass&#xff09;&#xff0c;类B则称为外部类&#xff08;OuterClass&#xff09;。 内部类的使用场景: 类A只在类B中使用&#xff0c;便可以使用内部类的方法…...

唯一ID如何生成,介绍一下目前技术领域最常使用的几种方法

纵使十面大山&#xff0c;又如何&#xff0c;无妨… 概述 唯一ID&#xff08;Unique Identifier&#xff09;是在计算机科学和信息技术领域中用于标识某个实体或数据的唯一标识符。生成唯一ID的方法可以根据具体需求和应用场景的不同而有所不同。以下是一些目前技术领域中常用…...

【翻译】XL-Sum: Large-Scale Multilingual Abstractive Summarization for 44 Languages

摘要 当代的关于抽象文本摘要的研究主要集中在高资源语言&#xff0c;比如英语&#xff0c;这主要是因为低/中资源语言的数据集有限。在这项工作中&#xff0c;我们提出了XL-Sum&#xff0c;这是一个包含100万篇专业注释的文章摘要对的综合多样数据集&#xff0c;从BBC中提取&…...

配置OpenCV

Open CV中包含很多图像处理的算法&#xff0c;因此学会正确使用Open CV也是人脸识别研究的一项重要工作。在 VS2017中应用Open CV&#xff0c;需要进行手动配置&#xff0c;下面给出在VS2017中配置Open CV的详细步骤。 1.下载并安装OpenCV3.4.1与VS2017的软件。 2.配置Open CV环…...

1-时间复杂度和空间复杂度

为了找到最适合当前问题而估量“算法”的评价s 时间复杂度空间复杂度执行效率&#xff1a;根据算法编写出的程序&#xff0c;执行时间越短&#xff0c;效率就越高占用的内存空间&#xff1a;不同算法编写出的程序&#xff0c;执行时占用的内存空间也不相同。如果实际场景中仅能…...

微信小程序之bind和catch

这两个呢&#xff0c;都是绑定事件用的&#xff0c;具体使用有些小区别。 官方文档&#xff1a; 事件冒泡处理不同 bind&#xff1a;绑定的事件会向上冒泡&#xff0c;即触发当前组件的事件后&#xff0c;还会继续触发父组件的相同事件。例如&#xff0c;有一个子视图绑定了b…...

java_网络服务相关_gateway_nacos_feign区别联系

1. spring-cloud-starter-gateway 作用&#xff1a;作为微服务架构的网关&#xff0c;统一入口&#xff0c;处理所有外部请求。 核心能力&#xff1a; 路由转发&#xff08;基于路径、服务名等&#xff09;过滤器&#xff08;鉴权、限流、日志、Header 处理&#xff09;支持负…...

基于FPGA的PID算法学习———实现PID比例控制算法

基于FPGA的PID算法学习 前言一、PID算法分析二、PID仿真分析1. PID代码2.PI代码3.P代码4.顶层5.测试文件6.仿真波形 总结 前言 学习内容&#xff1a;参考网站&#xff1a; PID算法控制 PID即&#xff1a;Proportional&#xff08;比例&#xff09;、Integral&#xff08;积分&…...

练习(含atoi的模拟实现,自定义类型等练习)

一、结构体大小的计算及位段 &#xff08;结构体大小计算及位段 详解请看&#xff1a;自定义类型&#xff1a;结构体进阶-CSDN博客&#xff09; 1.在32位系统环境&#xff0c;编译选项为4字节对齐&#xff0c;那么sizeof(A)和sizeof(B)是多少&#xff1f; #pragma pack(4)st…...

Linux相关概念和易错知识点(42)(TCP的连接管理、可靠性、面临复杂网络的处理)

目录 1.TCP的连接管理机制&#xff08;1&#xff09;三次握手①握手过程②对握手过程的理解 &#xff08;2&#xff09;四次挥手&#xff08;3&#xff09;握手和挥手的触发&#xff08;4&#xff09;状态切换①挥手过程中状态的切换②握手过程中状态的切换 2.TCP的可靠性&…...

2.Vue编写一个app

1.src中重要的组成 1.1main.ts // 引入createApp用于创建应用 import { createApp } from "vue"; // 引用App根组件 import App from ./App.vue;createApp(App).mount(#app)1.2 App.vue 其中要写三种标签 <template> <!--html--> </template>…...

【Oracle】分区表

个人主页&#xff1a;Guiat 归属专栏&#xff1a;Oracle 文章目录 1. 分区表基础概述1.1 分区表的概念与优势1.2 分区类型概览1.3 分区表的工作原理 2. 范围分区 (RANGE Partitioning)2.1 基础范围分区2.1.1 按日期范围分区2.1.2 按数值范围分区 2.2 间隔分区 (INTERVAL Partit…...

Element Plus 表单(el-form)中关于正整数输入的校验规则

目录 1 单个正整数输入1.1 模板1.2 校验规则 2 两个正整数输入&#xff08;联动&#xff09;2.1 模板2.2 校验规则2.3 CSS 1 单个正整数输入 1.1 模板 <el-formref"formRef":model"formData":rules"formRules"label-width"150px"…...

听写流程自动化实践,轻量级教育辅助

随着智能教育工具的发展&#xff0c;越来越多的传统学习方式正在被数字化、自动化所优化。听写作为语文、英语等学科中重要的基础训练形式&#xff0c;也迎来了更高效的解决方案。 这是一款轻量但功能强大的听写辅助工具。它是基于本地词库与可选在线语音引擎构建&#xff0c;…...

人工智能(大型语言模型 LLMs)对不同学科的影响以及由此产生的新学习方式

今天是关于AI如何在教学中增强学生的学习体验&#xff0c;我把重要信息标红了。人文学科的价值被低估了 ⬇️ 转型与必要性 人工智能正在深刻地改变教育&#xff0c;这并非炒作&#xff0c;而是已经发生的巨大变革。教育机构和教育者不能忽视它&#xff0c;试图简单地禁止学生使…...