解决由Tomcat部署前端改成nginx部署,导致大写.JPG结尾文件无法访问问题

前言:因信创替代要求,在麒麟服务器部署新的应用。

原先的架构:前端tomcat部署,源码部署java应用(ps:前后端,文件都在同一台服务器上),前端访问后端,再通过后端应用访问图片/视频/报纸等资源;

新的架构:采用了nginx部署前端(A服务器),docker部署后端(B服务器,文件也在该服务器上),访问nginx,然后通过nginx转发到后端,再通过后端应用访问图片/视频/报纸等资源。

现象:大写的.JPG结尾的图片无法访问到

按F12看到的该图片返回的格式是text/plain,打印日志发现,java代码返回的图片格式是null。

验证:我部署了一个nginx,通过nginx访问图片是可以访问的,不管是大写结尾还是小写结尾(证明图片本身没有问题);上传了一张照片以.jpg是访问的,手动改成了.JPG无法访问😌(说明是文件格式问题)。

核心问题:Java MIME映射默认不识别大写扩展名

当通过 Java 代码读取图片流(而非直接操作文件路径)时,大写.JPG扩展名无法识别contentType,核心原因是:流操作场景下,Files.probeContentType(Path)等依赖扩展名的方法失效,且纯流解析默认不关联扩展名大小写

解决方案

1. 已知扩展名(从路径 / 文件名获取)→ 手动映射

若能从图片资源路径中提取扩展名(如/path/image.JPG中的JPG),可先将扩展名转为小写,再通过 “预定义映射表” 或 “自定义逻辑” 匹配contentType,避免依赖系统配置。

import java.io.InputStream;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.HashMap;
import java.util.Map;public class StreamContentTypeResolver {// 预定义常见图片扩展名→ContentType映射(不区分大小写,统一小写匹配)private static final Map<String, String> EXTENSION_CONTENT_TYPE_MAP = new HashMap<>();static {// 初始化映射表(可扩展其他格式)EXTENSION_CONTENT_TYPE_MAP.put("jpg", "image/jpeg");EXTENSION_CONTENT_TYPE_MAP.put("jpeg", "image/jpeg");EXTENSION_CONTENT_TYPE_MAP.put("png", "image/png");EXTENSION_CONTENT_TYPE_MAP.put("gif", "image/gif");EXTENSION_CONTENT_TYPE_MAP.put("bmp", "image/bmp");}/*** 结合“资源路径”和“流”解析ContentType(优先用扩展名,兜底用流签名)* @param resourcePath 图片资源路径(如"/path/image.JPG")* @param inputStream 图片流(需支持mark/reset,避免解析后流不可用)* @return ContentType(如"image/jpeg")*/public static String resolveContentType(String resourcePath, InputStream inputStream) {try {// 1. 从路径提取扩展名,转为小写Path path = Paths.get(resourcePath);String fileName = path.getFileName().toString().toLowerCase();String extension = null;int lastDotIndex = fileName.lastIndexOf('.');if (lastDotIndex > 0 && lastDotIndex < fileName.length() - 1) {extension = fileName.substring(lastDotIndex + 1);}// 2. 用预定义的映射表匹配ContentType(处理大写扩展名)if (extension != null && EXTENSION_CONTENT_TYPE_MAP.containsKey(extension)) {return EXTENSION_CONTENT_TYPE_MAP.get(extension);}// 3. 兜底:若扩展名匹配失败,用流签名解析(见方案2)return resolveContentTypeByStreamSignature(inputStream);} catch (Exception e) {e.printStackTrace();return null;}}public static void main(String[] args) throws Exception {// 示例:模拟大写JPG路径和流String resourcePath = "/data/images/test.JPG"; // 大写扩展名InputStream inputStream = StreamContentTypeResolver.class.getResourceAsStream("/test.JPG"); // 从类路径读取流String contentType = resolveContentType(resourcePath, inputStream);System.out.println("解析的ContentType:" + contentType); // 输出 "image/jpeg"}
}

2. 未知扩展名(仅能操作流)→ 读取文件签名

若无法获取扩展名(如流来自网络下载、数据库 BLOB),需直接读取流的前几个字节文件签名 判断类型 ——JPEG 的签名是FF D8 FF与扩展名大小写无关,是最可靠的方式。

关键注意点:
  • 流必须支持mark/reset(如BufferedInputStream),避免解析后流被 “读空”,影响后续业务使用
  • 读取签名后需reset()流,恢复到初始位置。
import java.io.BufferedInputStream;
import java.io.IOException;
import java.io.InputStream;public class StreamContentTypeResolver {// 续上方案1的类,添加流签名解析方法/*** 通过流的文件签名解析ContentType(不依赖扩展名)* @param inputStream 图片流(需包装为BufferedInputStream以支持mark/reset)* @return ContentType(如"image/jpeg")*/public static String resolveContentTypeByStreamSignature(InputStream inputStream) throws IOException {// 包装为BufferedInputStream,确保支持mark/reset(若原流已支持可跳过)BufferedInputStream bis = new BufferedInputStream(inputStream);bis.mark(10); // 标记前10字节(足够读取所有图片格式的签名)try {// 读取前3字节(多数图片格式的签名长度≤3)byte[] signature = new byte[3];int readLen = bis.read(signature);if (readLen < 3) {return null; // 流过短,无法识别}// 匹配文件签名(按优先级排序)// JPEG签名:FF D8 FFif (signature[0] == (byte) 0xFF && signature[1] == (byte) 0xD8 && signature[2] == (byte) 0xFF) {return "image/jpeg";}// PNG签名:89 50 4Eif (signature[0] == (byte) 0x89 && signature[1] == (byte) 0x50 && signature[2] == (byte) 0x4E) {return "image/png";}// GIF签名:47 49 46if (signature[0] == (byte) 0x47 && signature[1] == (byte) 0x49 && signature[2] == (byte) 0x46) {return "image/gif";}// 其他格式可继续扩展(如BMP:42 4D 36)return null;} finally {// 重置流到初始位置,供后续业务使用(关键!)bis.reset();}}public static void main(String[] args) throws Exception {// 示例:仅通过流解析大写JPG(未知扩展名)InputStream rawStream = StreamContentTypeResolver.class.getResourceAsStream("/test.JPG");InputStream bufferedStream = new BufferedInputStream(rawStream); // 包装为支持mark的流String contentType = resolveContentTypeByStreamSignature(bufferedStream);System.out.println("流签名解析的ContentType:" + contentType); // 输出 "image/jpeg"// 后续可继续使用bufferedStream(已reset,位置正确)// ... 业务逻辑(如保存流、处理图片)...}
}

3.统一扩展名大小写(全新系统推荐)

在Java代码中,先将文件路径/文件名的扩展名转为小写,再匹配MIME类型,从源头避免大小写问题。(但是对已上线的问题不友好)

4.手动添加大写扩展名的MIME映射

若无法修改文件名,可在代码中手动注册.JPG对应的MIME类型,强制让java识别

5.使用Files.probeContentType()并处理大小写(我采用下面这个方式解决的

思考:为什么tomcat部署就没问题

HTTP服务器通过“文件后缀 + MIME配置表” 匹配Content-Type。Tomcat可能默认包含了大小写后缀的MIME配置。Nginx默认不区分文件后缀大小写,且未为大写.JPG配置对应的MIME类型映射

给Nginx添加大写.JPG的MIME映射,我也试了这种方式,修改nginx的MIME配置表(sed -i 's/jpeg jpg/jpeg jpg JPG JPEG/g' /etc/nginx/mime.types),但是未解决问题,可能某个地方配置问题,空了还是想试下这个方法。

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

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

相关文章

【设计模式】三大原则 单一职责原则、开放-封闭原则、依赖倒转原则

系列文章目录 文章目录系列文章目录一、单一职责原则方块游戏的设计二、开放-封闭原则原则介绍何时应对变化三、依赖倒转原则依赖倒转原则介绍里氏代换原则总结一、单一职责原则 单一职责原则&#xff0c;听字面意思&#xff0c;就是说功能要单一&#xff0c;他的准确解释是&a…

(3dnr)多帧视频图像去噪 (一)

一、多帧视频图像去噪 原理当摄像机每秒捕捉的图像达到60FPS&#xff0c;除了场景切换或者一些快速运动的场 景外&#xff0c;视频信号中相邻的两帧图像内容大部分是相同的。并且视频信号中的噪 声大部分都是均值为零的随机噪声&#xff0c;因此在时间上对视频信号做帧平均&…

从静态到智能:用函数式接口替代传统工具类

在 Java 早期开发中&#xff0c;我们习惯使用**静态实用程序类&#xff08;Utility Class&#xff09;**来集中放置一些通用方法&#xff0c;例如验证、字符串处理、数学计算等。这种模式虽然简单直接&#xff0c;但在现代 Java 开发&#xff08;尤其是 Java 8 引入 Lambda 和函…

免杀伪装 ----> R3进程伪装实战(高阶) ---->培养红队免杀思路

目录 R3进程伪装(免杀技术)高阶技术说明 深入剖析Windows进程规避免杀技术 学习R3进程伪装的必备技能 R3进程伪装的核心知识点与实现步骤 核心知识点 实现步骤 免杀实现步骤 PEB与EPROCESS的深入解析 1. PEB&#xff08;进程环境块&#xff09; 2. EPROCESS 3. PEB与…

深度学习——基于卷积神经网络实现食物图像分类(数据增强)

文章目录 引言 一、项目概述 二、环境准备 三、数据预处理 3.1 数据增强与标准化 3.2 数据集准备 四、自定义数据集类 五、构建CNN模型 六、训练与评估 6.1 训练函数 6.2 评估函数 6.3 训练流程 七、关键技术与优化 八、常见问题与解决 九、完整代码 十、总结 引言 本文将详细介…

【开题答辩全过程】以 基于微信小程序的教学辅助系统 为例,包含答辩的问题和答案

个人简介一名14年经验的资深毕设内行人&#xff0c;语言擅长Java、php、微信小程序、Python、Golang、安卓Android等开发项目包括大数据、深度学习、网站、小程序、安卓、算法。平常会做一些项目定制化开发、代码讲解、答辩教学、文档编写、也懂一些降重方面的技巧。感谢大家的…

【代码解读】Deepseek_vl2中具体代码调用

【代码解读】Deepseek_vl2中具体代码调用 文章目录【代码解读】Deepseek_vl2中具体代码调用DeepseekVLV2Processor解读DeepseekVLV2ForCausalLM - 多模态模型DeepSeek-VL2 Processor的输入格式单样本格式多样本格式DeepSeek-VL2模型的输出形式总结主要输出类型&#xff1a;Deep…

Git 9 ,.git/index.lock 文件冲突问题( .git/index.lock‘: File exists. )

目录 前言 一、问题背景 1.1 问题出现场景 1.2 典型报错信息 1.3 问题影响 二、问题原因分 2.1 Git 的 index 与锁机制 2.2 主要作用 2.3 根本原因 三、解决方案 3.1 确认进程 3.2 手动删除 3.3 再次执行 四、注意事项 4.1 确保运行 4.2 问题排查 4.3 自动化解…

Proteus8 仿真教学全指南:从入门到实战的电子开发利器

在电子设计、单片机课程设计或创客实践中&#xff0c;你是否常因实物采购贵、新手怕烧板、调试排错难而头疼&#xff1f;Proteus8 作为一款 “全能型” EDA 仿真工具&#xff0c;完美解决这些痛点 —— 它集「原理图绘制 PCB 设计 虚拟仿真」于一体&#xff0c;支持 51、STM3…

系统科学:结构、功能与层级探析

摘要本文旨在系统性地梳理和辨析系统科学中的核心概念——结构、功能与层级。文章首先追溯系统思想的理论源流&#xff0c;确立其作为一种超越还原论的整体性研究范式。在此基础上&#xff0c;深度剖析系统结构的内在构成&#xff08;组分、框架、动态性&#xff09;、系统层级…

面试官问:你如何看待薪资待遇?

在面试过程中&#xff0c;“你如何看待薪资待遇&#xff1f;”这个问题&#xff0c;是很多面试官都会提出的经典问题之一。虽然表面上看起来是一个简单的提问&#xff0c;但它实则关乎候选人的职业价值观、工作态度以及对自己能力的认知。薪资是工作的重要动力之一&#xff0c;…

HarmonyOS 应用开发新范式:深入剖析 Stage 模型与 ArkUI 最佳实践

好的&#xff0c;请看这篇基于 HarmonyOS (鸿蒙) 最新技术栈的深度技术文章。 HarmonyOS 应用开发新范式&#xff1a;深入剖析 Stage 模型与 ArkUI 最佳实践 引言 随着 HarmonyOS 4、5 的持续演进和未来 6 的规划&#xff0c;其应用开发框架经历了革命性的重构。对于技术开发者…

【Python数据可视化:Matplotlib高级技巧】

Python数据可视化&#xff1a;Matplotlib高级技巧引言在数据科学和分析领域&#xff0c;数据可视化是理解和传达信息的关键工具。Python中最流行的可视化库之一就是Matplotlib。虽然初学者可以快速上手Matplotlib的基础功能&#xff0c;但掌握其高级技巧才能真正发挥这个强大库…

LazyLLM教程 | 第7讲:检索升级实践:亲手打造“更聪明”的文档理解系统!

本节&#xff0c;我们将首先介绍如何评价 RAG 的检索组件&#xff0c;帮助您理解如何衡量 RAG 系统的检索能力。随后&#xff0c;我们会深入探讨几种提升 RAG 系统检索组件效果的策略实现以及对应的效果对比&#xff1a;1.基于 LazyLLM 实现查询重写策略。2.介绍 LazyLLM 中的节…

rust语言 (1.88) egui (0.32.1) 学习笔记(逐行注释)(二十四)窗口颜色、透明度、居中显示

一、窗口颜色和透明度 &#xff08;一&#xff09;效果预览&#xff08;二&#xff09;透明窗体主要代码 use eframe::egui; use egui::Color32;fn main() -> eframe::Result<()> {let options eframe::NativeOptions {viewport: egui::ViewportBuilder::default() …

基于无人机的风电叶片全自动智能巡检:高精度停角估计与细节优先曝光调控技术

【导读】 本文致力于解决一个非常实际的工业问题&#xff1a;如何利用无人机&#xff08;UAV&#xff09;全自动、高效、可靠地检查风力涡轮机叶片。叶片是风力发电机组中最昂贵且易损的部件之一&#xff0c;定期检查至关重要。然而&#xff0c;当前的技术在自动化过程中面临几…

腾讯云上有性能比较强的英伟达GPU

腾讯云上有性能比较强的英伟达GPU A100&#xff0c;虽然落后3~4代&#xff0c;但是估计是最强的英伟达GPU了。

AI任务相关解决方案13-AI智能体架构方案(意图识别+多任务规划+MCP+RAG)与关键技术深度解析研究报告,以及实现代码

文章目录 1. 总体技术方案 2. 生成式大模型(LLM):Data Agent的大脑 3. 意图识别:准确理解用户意图 3.1 基于BERT的微调方法 3.2 基于大语言模型(LLM)的零样本/少样本方法 4. 多任务规划:提升架构的灵活性 4.1 任务分解与规划 4.2 多智能体协作规划 4.3 基于强化学习的规划方…

每日五个pyecharts可视化图表日历图和箱线图:从入门到精通

&#x1f4ca; 本文特色&#xff1a;从零开始掌握日历图和箱线图可视化技巧&#xff0c;包含多个完整实例、核心配置项解析和实用场景指南&#xff0c;助您快速构建专业数据可视化图表。pyecharts源码 目录什么是日历图和箱线图&#xff1f;&#x1f4c5; 日历图&#xff08;Ca…

在本地获取下载chrome,然后离线搬运到 ECS

场景&#xff1a; 阿里云 ECS 无Y网&#xff0c;无法直接拉取 storage.googleapis.com。因此需先在本地里拿到直链并下载&#xff0c;再上传到 ECS。 注&#xff1a; 这个链接是显示近期的几个版本 https://googlechromelabs.github.io/chrome-for-testing/ 这个链接是所有版…