企业级实战:将Java服务打包为Docker镜像的两种高效方法
企业级实战:将Java服务打包为Docker镜像的两种高效方法
摘要:本文针对Java服务容器化部署场景,提供 基于容器Commit 和 Dockerfile构建 两种镜像制作方案。重点解决动态库依赖、信号量配置、环境变量注入等企业级痛点问题,并提供全流程操作步骤与避坑指南。
一、需求背景与核心挑战
1.1 业务场景描述
- 服务构成:Java应用(
demo-1.0.jar) + JNI动态库(libdemo_jni.so) - 特殊依赖:
- 要求修改Linux内核信号量参数(
kernel.sem) - 需安装GCC运行时库(通过
gcc_rpm.tar) - 使用Supervisor管理进程(
supervisor-4.2.5.tar.gz)
- 要求修改Linux内核信号量参数(
1.2 技术难点分析
| 问题 | 解决方案 |
|---|---|
| 信号量参数容器启动后失效 | 在启动脚本中执行sysctl -p |
| JNI动态库路径加载 | 通过LD_LIBRARY_PATH环境变量指定 |
| 基础环境依赖复杂 | 预装GCC、setuptools等工具链 |
二、方案一:容器Commit快速构建镜像
2.1 操作流程图解
2.2 分步操作命令
步骤1:启动基础容器
docker run -it --name java-builder centos:7 /bin/bash
步骤2:传输文件到容器
# 宿主机执行(假设容器ID为d4)
docker cp libdemo_jni.so d4:/home/demo/lib/
docker cp {demo-1.0.jar,application.yml,jdk-8u152-linux-x64.tar.gz} d4:/home/demo/
步骤3:安装并配置JDK
tar -xvf jdk-8u152-linux-x64.tar.gz
vi /etc/profile # 添加以下内容# ===== 环境变量配置 =====
export JAVA_HOME=/home/demo/jdk1.8.0_152
export PATH=$JAVA_HOME/bin:$PATH
export LD_LIBRARY_PATH=/home/demo/lib
步骤4:配置内核信号量
echo "kernel.sem = 2000 125000 32 500" >> /etc/sysctl.conf
sysctl -p # 立即生效
步骤5:安装GCC环境
rpm -ivh *.rpm --nodeps --force # 强制安装依赖
步骤6:部署Supervisor
unzip setuptools-41.1.0.zip
cd setuptools-41.1.0 && python setup.py installtar -xvf supervisor-4.2.5.tar.gz
cd supervisor-4.2.5 && ./setup_supervisor.sh
步骤7:封装启动脚本
vi /home/demo/start.sh#!/bin/bash
sysctl -p # 关键步骤!确保信号量生效
/home/demo/supervisor-4.2.5/start_supervisor.sh
java -jar /home/demo/demo-1.0.jar
步骤8:生成镜像
docker commit java-builder demo:v1
三、方案二:Dockerfile标准化构建
3.1 Dockerfile全解析
# 使用基础镜像
FROM centos:7# 设置元数据
LABEL maintainer="devops-team@company.com"# 拷贝所有依赖文件
COPY jdk-8u152-linux-x64.tar.gz /home/demo/
COPY start.sh demo-1.0.jar application.yml /home/demo/
COPY libdemo_jni.so /home/demo/lib/
COPY gcc_rpm.tar /home/demo/# 环境变量配置
ENV JAVA_HOME=/usr/lib/jvm/jdk1.8.0_152
ENV PATH=$PATH:$JAVA_HOME/bin
ENV LD_LIBRARY_PATH=/home/demo/lib# 执行构建命令
RUN mkdir -p /home/demo/lib && \tar -xvf /home/demo/jdk-8u152-linux-x64.tar.gz -C /usr/lib/jvm/ && \echo "kernel.sem = 2000 125000 32 500" > /etc/sysctl.conf && \tar -xvf /home/demo/gcc_rpm.tar -C /tmp/gcc && \rpm -ivh /tmp/gcc/*.rpm --nodeps --force && \yum clean all && \rm -rf /var/cache/yum# 设置启动命令
CMD ["/bin/bash", "/home/demo/start.sh"]
3.2 镜像构建命令
# 构建镜像
docker build -t demo:v2 -f Dockerfile .# 运行测试
docker run -d --name demo-instance \-p 18089:8080 \-v /sys:/sys:ro \--privileged \demo:v2
四、方案对比与选型建议
| 维度 | Commit方案 | Dockerfile方案 |
|---|---|---|
| 构建速度 | ⭐⭐⭐⭐(快速) | ⭐⭐(需完整编译) |
| 可追溯性 | ⭐(难维护) | ⭐⭐⭐⭐⭐(版本可控) |
| 镜像体积 | ⭐⭐(较大) | ⭐⭐⭐(可优化) |
| 适用场景 | 紧急调试/快速验证 | 生产环境/CI-CD流水线 |
生产环境强烈推荐:
- 使用Dockerfile构建,结合多阶段构建(Multi-stage)优化镜像体积
- 集成到Jenkins/GitLab CI自动化流程
五、高级调优技巧
5.1 信号量持久化方案
# 在容器启动时注入
docker run --sysctl "kernel.sem=2000 125000 32 500" ...
5.2 安全加固措施
# 创建非root用户
RUN groupadd -r appuser && \useradd -r -g appuser appuser
USER appuser
5.3 镜像瘦身方法
# 使用Alpine基础镜像
FROM alpine:3.18# 精简JDK(jlink定制)
RUN jlink --strip-debug --compress=2 ...
六、常见问题排查
Q1:动态库加载失败
现象:java.lang.UnsatisfiedLinkError
解决:
- 检查
LD_LIBRARY_PATH是否包含.so文件目录 - 执行
ldd libdemo_jni.so验证依赖
Q2:信号量未生效
排查:
docker exec -it demo-instance cat /proc/sys/kernel/sem
Q3:Supervisor启动异常
日志定位:
docker logs --tail 100 demo-instance | grep "supervisord"
如果本教程帮助您解决了问题,请点赞❤️收藏⭐支持!欢迎在评论区留言交流技术细节!欲了解密码学知识,请订阅《密码学实战》专栏 → 密码学实战
相关文章:
企业级实战:将Java服务打包为Docker镜像的两种高效方法
企业级实战:将Java服务打包为Docker镜像的两种高效方法 摘要:本文针对Java服务容器化部署场景,提供 基于容器Commit 和 Dockerfile构建 两种镜像制作方案。重点解决动态库依赖、信号量配置、环境变量注入等企业级痛点问题,并提供…...
专题十六:虚拟路由冗余协议——VRRP
一、VRRP简介 VRRP(Virtual Router Redundancy Protocol)虚拟路由冗余协议通过把几台设备联合组成一台虚拟的设备,使用一定的机制保证当主机的下一跳设备出现故障时,及时将业务切换到备份设备,从而保持通讯的连续性和…...
Java中常见的锁synchronized、ReentrantLock、ReentrantReadWriteLock、StampedLock
在Java中,锁是实现多线程同步的核心机制。不同的锁适用于不同的场景,理解其实现原理和使用方法对优化性能和避免并发问题至关重要。 一、隐式锁:synchronized 关键字 实现原理 基于对象监视器(Monitor):每…...
DDPM(diffusion)原理
DDPM(diffusion)原理 1、DDPM(原理)2、DDPM和 Conditional DDPM(原理解释)2.1. Diffusion Models 原理详解核心思想前向扩散过程(Forward Diffusion)反向去噪过程(Revers…...
《软件设计师》复习笔记(2.2)——效验码、体系结构、指令、流水线
目录 一、校验码 码距 奇偶校验码 循环冗余校验码(CRC) 海明码 真题示例: 二、体系结构 Flynn分类法 三、指令系统 指令组成 指令执行过程 指令的寻址方式 操作数的寻址方式 CISC vs RISC 真题示例: 四、流水线技…...
BT1120 BT656驱动相关代码示例
前些年做视频输出项目的时候用过bt1120 tx与rx模块,现将部分代码进行记录整理。代码功能正常,可正常应用。 1. rx部分: /****************************************************************************** Copyright (C) 2021,All rights …...
2025.04.19-阿里淘天春招算法岗笔试-第一题
📌 点击直达笔试专栏 👉《大厂笔试突围》 💻 春秋招笔试突围在线OJ 👉 笔试突围OJ 01. 字符交换智慧 问题描述 卢小姐有一个长度为 n n n 的字符串...
IsaacSim Asserts 配置
IsaacSim Asserts 配置 背景解决方案资源准备具体操作步骤验证 背景 我是习惯使用 isaacsim 的 standalone 模式,使用 python 脚本直接运行 script,然后弹窗,按照规则正确运行即可,但是,这就导致了一些问题出现&#…...
关于viewpager常见的泄漏
在一个页面中 如果有用到tab,有需要进行fragment的切换,经常就看到了private var fragments arrayListOf<Fragment>()private fun initFragment() {arguments?.let {hopeToPosition it.getInt(IntentConstant.MAIN_PAGE_GO, 0)workoutType it.…...
深入剖析 C/S 与 B/S 架构及网络通信基础
目录 C/S 架构详解 概念与示例 优点 B/S 架构详解 概念与示例 优势 缺点 C/S 与 B/S 的区别 架构组成 使用场景 开发和维护 安全性 网络通信基础 IP 地址 MAC(物理地址) 端口 路由器 网关 子网掩…...
接口自动化 ——fixture allure
一.参数化实现数据驱动 上一篇介绍了参数化,这篇 说说用参数化实现数据驱动。在有很多测试用例的时候,可以将测试用例都存储在文件里,进行读写调用。本篇主要介绍 csv 文件和 json 文件。 1.读取 csv 文件数据 首先创建 csv 文件ÿ…...
systemctl管理指令
今天我们来继续学习服务管理指令,接下来才是重头戏-systemctl,那么话不多说,直接开始吧. systemctl管理指令 1.基本语法: systemctl [start | stop | restart | status]服务 注:systemctl指令管理的服务在/usr/lib/ systemd/system查看 2.systemctl设置服务的自…...
【文件操作与IO】详细解析文件操作与IO (二)
本篇博客是上一篇文章的续写,重点介绍数据流,还包括三道练习题. 🐎文章专栏: JavaEE初阶 🚀若有问题 评论区见 ❤ 欢迎大家点赞 评论 收藏 分享 如果你不知道分享给谁,那就分享给薯条. 你们的支持是我不断创作的动力 . 王子,公主请阅🚀 要开心…...
go-map+sync.map的底层原理
map 哈希冲突解决方式 1.拉链法 2.开放地址法 底层结构 Go 的 map 在源码中由 runtime.hmap 结构体表示,buckets-指向桶数组的指针(常规桶),oldbuckets-扩容时指向旧桶数组的指针。 type hmap struct {count int // 当前元素个数(len…...
java怎么找bug?Arthas原理与实战指南
Arthas原理与实战指南 1. Arthas简介 Arthas是阿里巴巴开源的Java诊断工具,其名字取自《魔兽世界》的人物阿尔萨斯。它面向线上问题定位,被广泛应用于性能分析、定位问题、安全审计等场景。Arthas的核心价值在于它能够在不修改应用代码、不重启Java进程…...
Windows使用SonarQube时启动脚本自动关闭
一、解决的问题 Windows使用SonarQube时启动脚本自动关闭,并发生报错: ERROR: Elasticsearch did not exit normally - check the logs at E:\Inori_Code\Year3\SE\sonarqube-25.2.0.102705\sonarqube-25.2.0.102705\logs\sonarqube.log ERROR: Elastic…...
Day53 二叉树的层序遍历
给你二叉树的根节点 root ,返回其节点值的 层序遍历 。 (即逐层地,从左到右访问所有节点)。 /*** Definition for a binary tree node.* public class TreeNode {* int val;* TreeNode left;* TreeNode right;* T…...
物联网智慧教室项目(完整版)
物联网智慧教室项目(一):智慧教室项目解决方案 一、智慧教室项目设计 (一)环境信息采集控制功能 1、硬件设计 使用STM32开发板模拟灯光控制,报警控制,光照信息采集: 灯光控制通过GPIO控制板载LED报警控…...
替代升级VMware | 云轴科技ZStack构建山西证券一云多芯云平台
通过云轴科技ZStack Cloud云平台,山西证券打造了敏捷部署、简单运维的云平台,不仅兼容x86、海光、鲲鹏三种异构服务器实现一云多芯,还通过云平台虚拟化纳管模块纳管原有VMware虚拟化资源,并对接第三方集中式存储,在保护…...
计算机网络期中复习笔记(自用)
复习大纲 –第一章 概述 计算机网络的组成 网络边缘:主机和网络应用程序(又称为“端系统”) 端系统中运行的程序之间的通信方式可划分为两大类: 客户/服务器方式(C/S方式) 对等方式(P2P方式…...
14.Chromium指纹浏览器开发教程之WebGL指纹定制
WebGL指纹概述 当在浏览器打开的网页上浏览内容时,看到的大多是平面的、静态的图像和文字。但是有时想要在网页上看到更加生动、立体的图像,如3D游戏、虚拟现实应用等。这时,就需要用到WebGL。 简单来说,WebGL(Web G…...
GitHub SSH连接终极解决方案
GitHub SSH连接终极解决方案:443端口修改多场景故障排查指南 一、问题现象速查 当开发者执行以下命令时出现连接异常: ssh -T gitgithub.com常见报错类型: 经典端口阻塞ssh: connect to host github.com port 22: Connection refused密钥验…...
Git 中修改某个特定的commit提交内容
在 Git 中修改某个特定的提交(commit)通常需要使用 交互式变基(Interactive Rebase) 或 修改提交(Commit Amend)。以下是不同场景下的具体操作步骤: 一、修改最近的提交(最新提交&am…...
每日算法【双指针算法】(Day 1-移动零)
双指针算法 1.算法题目(移动零)2.讲解算法原理3.编写代码 1.算法题目(移动零) 2.讲解算法原理 数组划分,数组分块(快排里面最核心的一步)只需把0改为tmp 双指针算法:利用数组下标来…...
B端管理系统:企业运营的智慧大脑,精准指挥
B端管理系统的定义与核心功能 B端管理系统(Business Management System)是专门设计用于支持企业内部运作和外部业务交互的一套软件工具。它集成了多种功能模块,包括但不限于客户关系管理(CRM)、供应链管理(SCM)、人力资源管理(HRM)以及财务管…...
使用Java基于Geotools的SLD文件编程式创建与磁盘生成实战
前言 在地理信息系统(GIS)领域,地图的可视化呈现至关重要,而样式定义语言(SLD)文件为地图元素的样式配置提供了强大的支持。SLD 能够精确地定义地图图层中各类要素(如点、线、面、文本等&#x…...
Git 命令速查手册
听说用美图可以钓读者? 一、基础操作核心命令 1. 仓库初始化与克隆 命令作用示例git init创建新仓库git init my-projectgit clone克隆远程仓库git clone [https://github.com/user/repo.git](https://github.com/user/repo.git)git remote add关联远程仓库git re…...
PKI 公钥基础设施
PKI 的全称是公钥基础设施(Public Key Infrastructure),是一个基于公钥加密技术,为网络环境中的各种应用提供安全服务的基础设施,由多个部分组成,各部分协同工作以实现数字证书的管理、密钥的生成与管理以及…...
android测试硬件工具 安卓硬件测试命令
Android开发常用ADB命令大全 在Android开发过程中,ADB(Android Debug Bridge)是一个非常重要的调试工具。掌握这些命令可以大大提高开发效率。如果你正在使用克魔开发助手(Keymob)这样的开发工具,你会发现它已经集成了很多ADB功能,让调试变得…...
网络编程 - 4 ( TCP )
目录 TCP 流套接字编程 API 介绍 SeverSocket Socket 用 TCP 实现一个回显服务器 服务端 客户端 运行调试 第一个问题:PrintWriter 内置的缓冲区 - flush 刷新解决 第二个问题:上述代码中,需要进行 close 操作吗? 第三…...
