SpringBoot项目部署之启动脚本

news2025/4/18 20:19:02

一、启动脚本方案

1. 基础启动方式

1.1 直接运行JAR

java -jar your-app.jar --spring.profiles.active=prod
  • 优点:简单直接,适合快速测试
  • 缺点:终端关闭即终止进程

1.2 后台运行

nohup java -jar your-app.jar > app.log 2>&1 &
  • 参数说明
    • nohup:忽略挂断信号,后台运行
    • > app.log:标准输出重定向
    • 2>&1:错误输出重定向
    • &:后台运行

2. Shell脚本封装

start.sh

#!/bin/bash

# 使用参数传入 JAR 文件名和应用名称
JAR_NAME=wkl-business.jar
APP_NAME=wkl-business

# 启动命令(保留原逻辑)
nohup java -jar -Dfile.encoding=UTF-8 $JAR_NAME $APP_NAME &
echo "Started $APP_NAME with command: java -jar $JAR_NAME $APP_NAME"


二、关闭脚本方案

1. 基础关闭方式

# 通过进程ID关闭
kill $(lsof -t -i:8080)

# 强制终止
kill -9 <PID>

2. Shell脚本封装

stop.sh

#!/bin/bash

# 使用参数传入应用名称
APP_NAME=wkl-business

# 原停止逻辑(仅提取 PID 方式)
pid=$(ps ax | grep -i "$APP_NAME" | grep java | grep -v grep | awk '{print $1}')
if [ -z "$pid" ]; then
  echo "No $APP_NAME running."
  exit -1
fi

echo "The $APP_NAME ($pid) is running..."
kill $pid
echo "Send shutdown request to $APP_NAME ($pid) OK"

三、完整示例脚本1

  • jar包同级目录下创建logs文件夹
  • 注:记得给脚本增加权限:chmod 777 start.sh、chmod 777 stop.sh
  • 脚本如下:

start.sh

#!/bin/bash
BASEPATH=$(cd "$(dirname "$0")"; pwd)
APP_NAME="demo.jar"
JAR_PATH=$BASEPATH/$APP_NAME
LOG_DIR=$BASEPATH/logs/${APP_NAME%.*}/nohup.out  # 日志路径示例
PID_FILE=$BASEPATH/$APP_NAME.pid
JAVA_OPTS="-Xms1g -Xmx2g -Dspring.profiles.active=prod"

# 创建日志目录
mkdir -p $(dirname $LOG_DIR)

# 检查运行状态
if [ -f "$PID_FILE" ]; then
  PID=$(cat "$PID_FILE")
  if ps -p $PID > /dev/null 2>&1; then
    echo "Application is already running with PID: $PID. Exiting..."
    exit 1
  else
    echo "PID file exists but process not found. Cleaning up..."
    rm -f "$PID_FILE"
  fi
fi

# 启动应用
nohup java $JAVA_OPTS -jar "$JAR_PATH" >> "$LOG_DIR" 2>&1 &

# 记录进程ID
echo $! > "$PID_FILE"
echo "Application started with PID: $(cat "$PID_FILE")"

# 查看日志
tail -100f "$LOG_DIR"

stop.sh

#!/bin/bash
BASEPATH=$(cd "$(dirname "$0")"; pwd)
APP_NAME="demo.jar"
PID_FILE="$BASEPATH/$APP_NAME.pid"

if [ -f "$PID_FILE" ]; then
  PID=$(cat "$PID_FILE")
  
  # 验证进程是否存在且为 Java 进程
  if ps -p "$PID" > /dev/null 2>&1 && [[ "$(ps -p "$PID" -o comm=)" == *"java"* ]]; then
    echo "Stopping application with PID: $PID..."
    kill -15 "$PID"
    
    # 等待进程终止(最多30秒)
    for i in {1..30}; do
      if ! ps -p "$PID" > /dev/null 2>&1; then
        break
      fi
      sleep 1
    done
    
    # 强制终止(如果仍存在)
    if ps -p "$PID" > /dev/null 2>&1; then
      echo "Force killing PID: $PID..."
      kill -9 "$PID"
      sleep 1
    fi
  else
    echo "PID $PID is invalid. Cleaning up..."
  fi
  
  # 清理 PID 文件
  rm -f "$PID_FILE"
else
  echo "PID file not found. Application is not running."
fi

echo "Application stopped"

四、完整示例脚本2

  • jar包同级目录下创建logs文件夹
  • jar包同级目录下创建config文件夹,包含application.yml 和logback.xml (非必须)文件,给777权限
  • 脚本如下:
# 该脚本为Linux下启动java程序的脚本
# 特别注意:
# 该脚本使用系统kill命令来强制终止指定的java程序进程。
# 所以在杀死进程前,可能会造成数据丢失或数据不完整。如果必须要考虑到这类情况,则需要改写此脚本,
# 
# 根据实际情况来修改以下配置信息 ##################################

# JAVA应用程序的名称
APP_NAME=xxxx系统
#获取脚本所在目录的绝对路径。
BASEPATH=$(cd "$(dirname "$0")"; pwd)
# jar包名称-记得更改
JAR_NAME=demo.jar
# PID 代表是PID文件(改进:使用绝对路径避免目录混乱)
JAR_PID="$BASEPATH/$JAR_NAME.pid"
# java虚拟机启动参数
JAVA_OPTS="-Xms512m -Xmx2048m -XX:MetaspaceSize=512m -XX:MaxMetaspaceSize=512m -XX:ParallelGCThreads=16 -XX:+PrintGCDateStamps -XX:+PrintGCDetails -XX:+HeapDumpOnOutOfMemoryError -Dfile.encoding=utf-8"
LOG_DIR="$BASEPATH/logs"

# 根据实际情况来修改以上配置信息 ##################################

# ######### Shell脚本中$0、$?、$!、$$、$*、$#、$@等的说明 #########
# $$ Shell本身的PID(ProcessID,即脚本运行的当前 进程ID号)
# $! Shell最后运行的后台Process的PID(后台运行的最后一个进程的 进程ID号)
# $? 最后运行的命令的结束代码(返回值)即执行上一个指令的返回值 (显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误)
# $- 显示shell使用的当前选项,与set命令功能相同
# $* 所有参数列表。如"$*"用「"」括起来的情况、以"$1 $2 … $n"的形式输出所有参数,此选项参数可超过9个。
# $@ 所有参数列表。如"$@"用「"」括起来的情况、以"$1" "$2" … "$n" 的形式输出所有参数。
# $# 添加到Shell的参数个数
# $0 Shell本身的文件名
# $1~$n 添加到Shell的各参数值。$1是第1参数、$2是第2参数…。
# ######### 说明结束 #########

# 检查程序是否处于运行状态(改进:使用pgrep精确匹配进程)
is_exist() {
  # 改进:使用pgrep替代ps/grep组合,避免误判
  pid=$(pgrep -f "$JAR_NAME" | grep -v "$$")  # 排除当前脚本进程
  if [ -n "$pid" ]; then
    return 0  # 存在返回0(符合Shell习惯)
  else
    return 1  # 不存在返回1
  fi
}

# 服务启动方法
start() {
  is_exist
  if [ $? -eq 0 ]; then
    echo "$APP_NAME is already running (PID: $(cat "$JAR_PID"))"
    return 1
  fi

  # 创建日志目录(原脚本未处理)
  

  # jar服务启动脚本(改进:验证启动是否成功)
  nohup java $JAVA_OPTS -jar -Dlogging.config=./config/logback.xml "$BASEPATH/$JAR_NAME" >> "$LOG_DIR/nohup.out" 2>&1 &
  pid=$!
  
  # 验证进程是否存活再记录PID
  if ps -p $pid > /dev/null 2>&1; then
    echo "$pid" > "$JAR_PID"
    echo "Start success, PID: $pid"
    tail -100f "$LOG_DIR/nohup.out"
  else
    echo "Failed to start $APP_NAME"
    exit 1
  fi
}

# 服务停止方法(改进:优雅终止+超时处理)
stop() {
  if [ ! -f "$JAR_PID" ]; then
    echo "PID file not found. Application is not running."
    return 1
  fi

  pid=$(cat "$JAR_PID")
  # 验证PID有效性
  if ! ps -p $pid > /dev/null 2>&1; then
    echo "Process $pid does not exist. Cleaning up."
    rm -f "$JAR_PID"
    return 1
  fi

  # 优雅终止(SIGTERM)
  echo "Stopping $APP_NAME (PID: $pid)..."
  kill $pid

  # 等待最多30秒
  for i in {1..30}; do
    if ! ps -p $pid > /dev/null 2>&1; then
      rm -f "$JAR_PID"
      echo "Stopped gracefully."
      return 0
    fi
    sleep 1
  done

  # 超时后强制终止(SIGKILL)
  echo "Force-stopping $APP_NAME..."
  kill -9 $pid
  sleep 1
  if ps -p $pid > /dev/null 2>&1; then
    echo "Failed to stop $APP_NAME."
    return 1
  else
    rm -f "$JAR_PID"
    echo "Force-stopped."
    return 0
  fi
}

# 服务运行状态查看方法
status() {
  is_exist
  if [ $? -eq 0 ]; then
    echo "$APP_NAME is running (PID: $(cat "$JAR_PID"))"
  else
    echo "$APP_NAME is not running"
  fi
}

# 重启服务方法
restart() {
  stop && start
}

# 帮助说明,用于提示输入参数信息
usage() {
    echo "Usage: sh $0 {start|stop|restart|status}"
    exit 1
}

###################################
# 读取脚本的第一个参数($1),进行判断
# 参数取值范围:{ start | stop | restart | status }
# 如参数不在指定范围之内,则打印帮助信息
###################################
case "$1" in
  start)
    start
    ;;
  stop)
    stop
    ;;
  restart)
    restart
    ;;
  status)
    status
    ;;
  *)
    usage
    ;;
esac
exit 0


五、注意事项

  1. 端口冲突处理

    # 检查端口占用
    lsof -i :8080
    
  2. 内存配置优化

    • 根据服务器配置调整-Xms-Xmx
    • 启用G1垃圾回收器:-XX:+UseG1GC
  3. 服务监控

    • 集成Prometheus监控
    • 配置邮件/SMS告警
  4. 滚动更新策略

    • 使用蓝绿部署
    • 配置Kubernetes滚动更新策略

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

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

相关文章

用Django和AJAX创建一个待办事项应用

用Django和AJAX创建一个待办事项应用 推荐超级课程: 本地离线DeepSeek AI方案部署实战教程【完全版】Docker快速入门到精通Kubernetes入门到大师通关课AWS云服务快速入门实战目录 用Django和AJAX创建一个待办事项应用让我们创建一个简单的 Django 项目,其中包含不同类型的 A…

JavaScript:游戏开发的利器

在近年来的科技迅速发展中&#xff0c;JavaScript 已逐渐成为游戏开发领域中最受欢迎的编程语言之一。它的跨平台特性、广泛的社区支持、丰富的库和框架使得开发者能够快速、有效地创建各种类型的游戏。本文将深入探讨 JavaScript 在游戏开发中的优势。 一、跨平台支持 JavaSc…

C语言今天开始了学习

好多年没有弄了&#xff0c;还是捡起来弄下吧 用的vscode 建议大家参考这个配置 c语言vscode配置 c语言这个语言简单&#xff0c;但是今天听到了一个消息说python 不知道怎么debug。人才真多啊

电商素材革命:影刀RPA魔法指令3.0驱动批量去水印,实现秒级素材净化

本文 去除水印实操视频展示电商图片水印处理的困境​影刀 RPA 魔法指令 3.0 强势登场​利用魔法指令3.0两步实现去除水印操作关于影刀RPA 去除水印实操视频展示 我们这里选择了4张小红书里面比较帅气的图片&#xff0c;但凡用过小红书的都知道&#xff0c;小红书右下角是会有小…

CVA6:支持 Linux 的 RISC-V CPU CORE-V

RISC-V 是一种开源的可扩展指令集架构 (ISA)&#xff0c;在过去几年中广受欢迎。RISC-V 的主要特性之一是它采用整体架构中性设计&#xff0c;支持浮点运算、加载存储架构、符号扩展加速和多路复用器简化。这使得 RISC-V 成为 FPGA 上软处理器的经济实惠的选择。自 RISC-V ISA …

轻奢宅家|约克VRF带你畅享舒适居家体验

下班回到家最期待什么&#xff1f;当然是一阵阵沁人心脾的舒适感扑面而来啦&#xff01;      想要从头到脚都舒服自在&#xff1f;答案就在眼前——就是它&#xff01;约克VRF中央空调&#xff01;      约克VRF中央空调独特的臻静降噪技术&#xff0c;让空调运行音轻…

uniapp微信小程序图片生成水印

整体思路&#xff1a; 用户通过uni.chooseImage选择图片后&#xff0c;获得图片文件的path和size。通过path调用uni.getImageInfo获取图片信息&#xff0c;也就是图片宽高。图片宽高等比缩放至指定大小&#xff0c;不然手机处理起来非常久&#xff0c;因为手机随便拍拍就很大。…

不用额外下载jar包,idea快速查看使用的组件源码

以nacos为例子&#xff0c;在idea中引入了nacos依赖&#xff0c;就可以查看源码了。 2. idea选择open&#xff08;不关闭项目直接选择file-open也可以&#xff09;, 在maven的仓库里找到对应的包&#xff0c;打开 2.idea中选择 jar包&#xff0c;选择 add as library 3.这样j…

网络通讯协议UDP转发TCP工具_UdpToTcpRelay_双向版

UDP/TCP网络转发器程序说明书 1. 程序概述 本程序是一个高性能网络数据转发工具&#xff0c;支持UDP和TCP协议之间的双向数据转发&#xff0c;并具备以下核心功能&#xff1a; 协议转换&#xff1a;实现UDP↔TCP协议转换数据转换&#xff1a;支持十六进制/ASCII格式的数据转…

DIA——边缘检测

1.边缘 边缘是像素的突变位置。 2.常见边缘检测算法 通过找到一阶导数的极值点或者二阶导数的过零点来确定边缘像素的位置。边缘检测通常使用算子&#xff0c;即特定的卷积核。通过差分对离散的像素点求导&#xff0c;然后转化成卷积核进行卷积。使用卷积统一涵盖求导&…

【万象论坛】论坛系统测试报告

一、项目背景 1.1项目起因 在当今数字化浪潮下&#xff0c;互联网技术呈爆发式发展&#xff0c;新技术、新框架、新应用场景不断涌现。从大型企业的数字化转型到初创公司的技术创新&#xff0c;各个层面都离不开互联网技术的支撑。然而&#xff0c;技术人员在学习与工作过程中…

【AI工具】FastGPT:开启高效智能问答新征程

前言 在人工智能飞速发展的当下&#xff0c;各类 AI 工具如雨后春笋般涌现。FastGPT 作为一款基于大语言模型&#xff08;LLM&#xff09;的知识图谱问答系统&#xff0c;凭借其强大的数据处理和模型调校能力&#xff0c;为用户带来了便捷的使用体验。今天&#xff0c;就让我们…

华为数字芯片机考2025合集1已校正

单选 1&#xff0e;以下低功耗措施中&#xff0c;哪种不是降低电路翻转率的方法? A.在不进行算术运算的时候&#xff0c;使这些模块的输入保持不变&#xff0c;不让新的操作数进来 B.采用Gray 码或One‐hot 码作为状态机编码 C.减少电路中的glitch D.重新安排“if‐else”表达…

HackMyVM - todd记录

HackMyVM - toddhttps://mp.weixin.qq.com/s/E_-hepdfY-0veilL1fl2QA

【完整可用】使用openhtmltopdf生成PDF(带SVG)

文章目录 前言OpenHTMLToPDF 简介maven配置依赖字体文件demo代码其他资源放置截图防止maven编译字体文件 前言 AI和网上都是跑不起来或者版本过低的&#xff0c;还有各种BUG的。本文都是查阅官方文档得出的。如果你能跑起来请给个大大的赞&#xff01; OpenHTMLToPDF 简介 Ope…

CTF web入门之爆破

爆破 web21: 打开burp进行抓包 通过对密码进行解析。得知密码是由拼接而来 admin:1 选择要攻击的参数 攻击方式。 选择payload方式 。。添加参数 1&#xff0c;2&#xff0c;3。账号 分隔符 密码 选择加密方式。添加buse64.去掉url字符。不然buse64后&#xff0c;会在u…

7-openwrt-one通过web页面配置访客网络、无线中继等功能

前几个章节一直在介绍编译、分区之类的,都还没正常开始使用这个路由器的wifi。默认wifi是没有启动的,前面还是通过手动修改uci配置启动的,这个章节介绍下官方web页面的使用。特别是访客网络、无线中继 1、开启wifi,配置wifi基本信息 我们使用有线连接路由器,通过192.168.…

Android使用声网SDK实现音视频互动(RTC)功能

一、前期准备 1、注册声网账号 声网官网 2、创建项目 拿到AppID&#xff0c;主要证书 二、代码部分 先上一下官方提供的demo地址&#xff1a; Agora-RTC-QuickStart: 此仓库包含 Agora RTC Native SDK 的QuickStart示例项目。 - Gitee.comhttps://gitee.com/agoraio-comm…

FPGA_modelsim错误总结

1&#xff0c; 使用modelsim仿真DDR3报错Module ‘SIP_PHY_CONTROL‘ is not defined 在配置ddr3的时候vivado 速度太慢了&#xff0c;所以选用modelsim。我的是2018.3vivado&#xff0c;modelsim用了10.4 但是不行报错 然后看了帖子说 questasim可以下载了还是报错。 然后又…

了解 DeFi:去中心化金融的入门指南与未来展望

去中心化金融&#xff0c;或 DeFi&#xff0c;代表着全球金融体系运作方式的革命性转变。它是一个总称&#xff0c;指的是一个不断增长的去中心化应用程序&#xff08;dapp&#xff09;、协议和平台生态系统&#xff0c;这些生态系统构建在公共区块链网络上&#xff0c;无需传统…