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

记一次上环境获取资源失败的案例

代码结构以及资源位置

在这里插入图片描述

测试代码

@RestController
@RequestMapping("/json")
public class JsonController {@GetMapping("/user/1")public String queryUserInfo() throws Exception {// 如果使用全路径, 必须使用/开头String path = JsonController.class.getPackage().getName().replace(".", File.separator);// 注意第一个字符不能是File.separator, 必须是/开头,String fileName = "/" + path + File.separator + "UserInfo.json";URL resource = JsonController.class.getResource(fileName);// 使用URL转换成uri可以兼容前面有/E:/idea-workspace/learning的windows盘符路径byte[] bytes = Files.readAllBytes(Paths.get(resource.toURI()));return new String(bytes, StandardCharsets.UTF_8);}@GetMapping("/user/2")public String queryUserInfo2() throws Exception {String path = JsonController.class.getResource("UserInfo.json").getPath();byte[] bytes = Files.readAllBytes(Paths.get(path.substring(1)));return new String(bytes, StandardCharsets.UTF_8);}@GetMapping("/user/3")public String queryUserInfo3() throws Exception {String path = JsonController.class.getPackage().getName().replace(".", File.separator);// classloader不能使用/开始获取文件, 因为文件不能包含/字符URL resource = JsonController.class.getClassLoader().getResource(path + File.separator + "UserInfo.json");// 很显然toUri的兼容性好些byte[] bytes = Files.readAllBytes(Paths.get(resource.toURI()));return new String(bytes, StandardCharsets.UTF_8);}@GetMapping("/user/4")public String queryUserInfo4() throws Exception {// 使用classloader获取资源和使用class的使用/开始获取资源本质逻辑一模一样URL resource = JsonController.class.getClassLoader().getResource("static/UserInfo.json");// 很显然toUri的兼容性好些byte[] bytes = Files.readAllBytes(Paths.get(resource.toURI()));return new String(bytes, StandardCharsets.UTF_8);}
}

本地调试一切正常, 但是上环境就报500了

{"timestamp": "2023-02-11T14:14:07.500+00:00","status": 500,"error": "Internal Server Error","path": "/json/user/4"
}

也可以通过 java -jar springboot-demo-0.0.1-SNAPSHOT.jar启动也可以复现问题

说明: 查看jar包确认资源文件已经被正确的打到jar中

说明代码存在兼容性

使用org.springframework.core.io.ClassPathResource获取

// 在jar包运行之后仍然不能找到文件@GetMapping("/user/5")public String queryUserInfo5() {try {// 如果使用全路径, 必须使用/开头String path = JsonController.class.getPackage().getName().replace(".", File.separator);// 注意第一个字符不能是File.separator, 必须是/开头,String fileName = "/" + path + File.separator + "UserInfo.json";ClassPathResource pathResource = new ClassPathResource(fileName);byte[] bytes = Files.readAllBytes(Paths.get(pathResource.getURI()));return new String(bytes, StandardCharsets.UTF_8);} catch (IOException e) {e.printStackTrace();}return "";}// 必须使用ClassPathResource并且使用其getInputStream()方法才能获取的到数据@GetMapping("/user/6")public String queryUserInfo6() {try {// 如果使用全路径, 必须使用/开头String path = JsonController.class.getPackage().getName().replace(".", File.separator);// 注意第一个字符不能是File.separator, 必须是/开头,String fileName = "/" + path + File.separator + "UserInfo.json";ClassPathResource pathResource = new ClassPathResource(fileName);byte[] buf = new byte[1024];int len = -1;StringBuilder sb = new StringBuilder();try (InputStream is = pathResource.getInputStream()) {while ((len = is.read(buf)) != -1) {sb.append(new String(buf, 0, len, StandardCharsets.UTF_8));}}return sb.toString();} catch (IOException e) {e.printStackTrace();}return "";}

queryUserInfo5方法只是用了ClassPathResource来获取URI, jar运行依然不能获取到资源

ClassPathResource pathResource = new ClassPathResource(fileName);
byte[] bytes = Files.readAllBytes(Paths.get(pathResource.getURI()));

queryUserInfo6使用了ClassPathResource的getInputStream()方法, jar运行依然可以好获取到

ClassPathResource pathResource = new ClassPathResource(fileName);
...
try (InputStream is = pathResource.getInputStream()) {
...}
}

使用spring的ClassPathResource确实解决了问题, 但是为什么会出现这样的问题?

tomcat项目和springboot项目差异

传统的tomcat项目都是提供war包, 然后war就会解压到webapps或者配置的应用目录下面, 所以我们的资源是在普通的目录里面
但是随着springboot 的出现, 问题开始有点不同了, springboot通过内嵌tomcat容器, 是的我们可以直接运行jar包, 所以此时我们要获取的资源文件是在jar包里面, 显然我们要获取资源就必须解析jar

所以我们的问题就此出现了, 如果仅仅使用传统的resouce api, 是不能获取到jar包中的文件, 就是说不会解析jar。

查看ClassPathResource源码找不同

进入
在这里插入图片描述
使用的类加载器是: TomcatEmbeddedWebappClassLoader
调用getResourceAsStream方法在org.apache.catalina.loader.WebappClassLoaderBase中
在这里插入图片描述
WebappClassLoaderBase位于tocmat-ebed-core-xxx包中

getResourceAsStream的实现

public InputStream getResourceAsStream(String name) {
...// (1) Delegate to parent if requestedif (delegateFirst) {stream = parent.getResourceAsStream(name);if (stream != null) {return stream;}}// (2) Search local repositoriesString path = nameToPath(name);WebResource resource = resources.getClassLoaderResource(path);if (resource.exists()) {stream = resource.getInputStream();trackLastModified(path, resource);}try {if (hasExternalRepositories && stream == null) {URL url = super.findResource(name);if (url != null) {stream = url.openStream();}}} catch (IOException e) {// Ignore}if (stream != null) {return stream;}
...}

主要就是两个一个是委派/一个是搜索本地存储(还有个是无条件委派本质和委派一样)

所以不太点就是WebResourceRoot回去搜索本地存储的文件, 然后返回resource

然后这个代码就很明显了, 回去classes下搜索

    @Overridepublic WebResource getClassLoaderResource(String path) {return getResource("/WEB-INF/classes" + path, true, true);}

直接使用getResourceAsStream方法也是可以获取

  private static String getString(InputStream inputStream) throws IOException {byte[] buf = new byte[1024];int len = -1;StringBuilder sb = new StringBuilder();try (InputStream is = inputStream) {while ((len = is.read(buf)) != -1) {sb.append(new String(buf, 0, len, StandardCharsets.UTF_8));}}return sb.toString();}
@GetMapping("/user/0")
public String queryUserInfo0() throws Exception {// 使用classloader获取资源和使用class的使用/开始获取资源本质逻辑一模一样InputStream inputStream = JsonController.class.getClassLoader().getResourceAsStream("static/UserInfo.json");return getString(inputStream);
}

测试发现只要使用了InputStream就都可以获取到了

@GetMapping("/user/000")
public String queryUserInfo000() throws Exception {URL resource = JsonController.class.getResource("UserInfo.json");return getString(resource.openStream());
}@GetMapping("/user/00")
public String queryUserInfo00() throws Exception {URL resource = JsonController.class.getResource("UserInfo.json");BufferedReader reader =new BufferedReader(new InputStreamReader(resource.openStream(), StandardCharsets.UTF_8));String line;StringBuilder sb = new StringBuilder();while ((line = reader.readLine()) != null) {sb.append(line);}return sb.toString();
}@GetMapping("/user/0")
public String queryUserInfo0() throws Exception {// 使用classloader获取资源和使用class的使用/开始获取资源本质逻辑一模一样InputStream inputStream = JsonController.class.getClassLoader().getResourceAsStream("static/UserInfo.json");return getString(inputStream);
}

resource.openStream()

搞半天问题其实出在这行代码上

byte[] bytes = Files.readAllBytes(Paths.get(path.substring(1)));

不具有jar包兼容性, 坑!!!

如果我们在编写获取文件的代码时候, 最好使用jar包本地测试一下, 防止出现不兼容问题

相关文章:

记一次上环境获取资源失败的案例

代码结构以及资源位置 测试代码 RestController RequestMapping("/json") public class JsonController {GetMapping("/user/1")public String queryUserInfo() throws Exception {// 如果使用全路径, 必须使用/开头String path JsonController.class.ge…...

实战超详细MySQL8离线安装

在RedHat中,RPM Bundle 方式安装MySQL8。建议一定要用 RPM Bndle 版本安装,包全。官网下载:https://dev.mysql.com/downloads/mysql/1.卸载mariadb,会与MySQL安装冲突。rpm -qa | grep mariadb 查看有无mariadb如果有&#xff0…...

依赖倒置原则|SOLID as a rock

文章目录 意图动机:违反依赖倒置原则解决方案:C++中依赖倒置原则的例子依赖倒置原则的优点1、可复用性2、可维护性在C++中用好DIP的标准总结本文是关于 SOLID as Rock 设计原则系列的五部分中的 最后一部分。 SOLID 设计原则侧重于开发 易于维护、可重用和可扩展的软件。 在…...

Webpack的知识要点

在前端开发中,一般情况下都使用 npm 和 webpack。   npm是一个非常流行的包管理工具,帮助开发者管理项目中使用的依赖库和工具。它可以方便地为项目安装第三方库,并在项目开发过程中进行版本控制。   webpack是一个模块打包工具&#xff…...

handler解析(2) -Handler源码解析

目录 基础了解: 相关概念解释 整体流程图: 源码解析 Looper 总结: sendMessage 总结: ThreadLocal 基础了解: Handler是一套 Android 消息传递机制,主要用于线程间通信。实际上handler其实就是主线程在起了一…...

【算法】kmp

KMP算法 名称由来 是由发明这个算法的三个科学家的名称首字母组成 作用 用于字符串的匹配问题 举例说明 字符串 aabaabaaf 模式串 aabaaf 传统匹配方法 第一步 aabaabaaf aabaaf 此时,b和f不一致,则把模式串从头和文本串的第二个字符开始比 第…...

git 常用命令之 git checkout

大家好,我是 17。 git checkout 是 git 中最重要最常用的命令之一,本文为大家详细解说一下。 恢复工作区 checkout 的用途之一是恢复工作区。 git checkout . checkout . 表示恢复工作区的所有更改,未跟踪的文件不会有变化。 恢复工作区的所有文件风…...

一些常见错误

500状态码: 代表服务器业务代码出错, 也就是执行controller里面的某个方法的过程中报错, 此时在IDEA的控制台中会显示具体的错误信息, 所以需要去看IDEA控制台的报错404状态码: 找不到资源找不到静态资源 检查请求地址是否拼写错误 检查静态资源的位置是否正确 如果以上都没有问…...

[单片机框架][调试功能] 回溯案发现场

程序莫名死机跑飞,不知道问题,那么下面教你回溯错误源 回溯案发现场一、修改HardFault_Handler1. xx.s 在启动文件,找到HardFault_Handler。并修改。2. 定义HardFault_Handler_C函数。(主要是打印信息并存储Flash)3. 根…...

MySQL主从同步-(二)搭建从机服务器

在docker中创建并启动MySQL从服务器:**端口3307docker run -d \-p 3307:3306 \-v /atguigu/mysql/slave1/conf:/etc/mysql/conf.d \-v /atguigu/mysql/slave1/data:/var/lib/mysql \-e MYSQL_ROOT_PASSWORD123456 \--name atguigu-mysql-slave1 \mysql:8.0.3创建MyS…...

Linux系列 备份与分享文档

作者简介:一名在校云计算网络运维学生、每天分享网络运维的学习经验、和学习笔记。 座右铭:低头赶路,敬事如仪 个人主页:网络豆的主页​​​​​​ 目录 前言 一.备份与分享文档 1.使用压缩和解压缩工具 (1&…...

SNI生效条件 - 补充nginx-host绕过实例复现中SNI绕过的先决条件

文章目录1.前置环境搭建2.测试SNI生效条件(时间)3. 证书对SNI的影响3.1 双方使用同一个证书:3.2 双方使用不同的证书与私钥4. 端口号区分测试4.1 端口号区分,证书区分:4.2 端口号区分,证书不区分:5.总结SNI运行机制6. SNI机制绕过…...

傻白探索Chiplet,Modular Routing Design for Chiplet-based Systems(十一)

阅读了Modular Routing Design for Chiplet-based Systems这篇论文,是关于多chiplet通信的,个人感觉核心贡献在于实现了 deadlock-freedom in multi-chiplet system,而不仅仅是考虑单个intra-chiplet的局部NoC可以通信,具体的一些…...

C语言静态库、动态库的封装和注意事项

1、动态库、静态库介绍 参考博客:《静态库和动态库介绍以及Makefile》; 2、代码目录结构和编译脚本 参考博客:《实际工作开发中C语言工程的目录结构分析》; 3、编写库的流程 (1)明确需求:需求是否合理、需求的使用场景、需求可能遇…...

MyBatis-Plus分页插件和MyBatisX插件

MyBatis-Plus分页插件和MyBatisX插件六、插件1、分页插件a>添加配置类b>测试八、代码生成器1、引入依赖2、快速生成十、MyBatisX插件1、新建spring boot工程a>引入依赖b>配置application.ymlc>连接MySQL数据库d>MybatisX逆向生成2、MyBatisX快速生成CRUD申明…...

年前无情被裁,面试大厂的这几个月…

2月份了,金三银四也即将来临,在这个招聘季,大厂也开始招人,但还是有很多人吐槽说投了很多简历,却迟迟没有回复… 另一面企业招人真的变得容易了吗?有企业HR吐槽,简历确实比以前多了好几倍&…...

基于Java的分片上传功能

起因:最近在工作中接到了一个大文件上传下载的需求,要求将文件上传到share盘中,下载的时候根据前端传的不同条件对单个或多个文件进行打包并设置目录下载。 一开始我想着就还是用老办法直接file.transferTo(newFile)就算是大文件&#xff0c…...

KDS安装步骤

KDS kinetis design studio 软件 第一步官网(https://www.nxp.com/ 注册账号下载set成功下载软件。 随着AI,大数据这些技术的快速发展,与此有关的知识也普及开来。如何在众多网站中寻找最有价值的信息,如何在最短的时间内获得最新的技…...

JavaSE-线程池(1)- 线程池概念

JavaSE-线程池(1)- 线程池概念 前提 使用多线程可以并发处理任务,提高程序执行效率。但同时创建和销毁线程会消耗操作系统资源,虽然java 使用线程的方式有多种,但是在实际使用过程中并不建议使用 new Thread 的方式手…...

开源代码的寿命为何只有1年?

说实话,如果古希腊的西西弗斯是一个在2016年编写开源代码的开发者,那他会有宾至如归的感觉。著名的西西弗斯处罚,是神话流传下来的,他被迫推一块巨大的石头上山,当登顶之后,只能眼睁睁看着它滚下去&#xf…...

完善登录功能--过滤器的使用

系列文章目录 Spring Boot读取配置文件内容的三种方式 Spring Boot自动配置–如何切换内置Web服务器 SpringBoot项目部署 上述为该系列部分文章,想了解更多可看我博客主页哦! 文章目录系列文章目录前言一、创建自定义过滤器LoginCheckFilter二、在启动类…...

CSS基础:属性和关系选择器

字体属性 color 文本颜色 div{ color:red;} div{ color:#ff0000;} div{ color:rgb(255,0,0);} div{ color:rgba(255,0,0,.5);}font-size 文本大小 h1 {font-size:40px;} h2 {font-size:30px;} p {font-size:14px;}注意:chrome浏览器接受最小字体是12px font-we…...

设计模式:原型模式解决对象创建成本大问题

一、问题场景 现在有一只猫tom,姓名为: tom, 年龄为:1,颜色为:白色,请编写程序创建和tom猫属性完全相同的10只猫。 二、传统解决方案 public class Cat {private String name;private int age;private String color;…...

驱动开发(二)

一、驱动流程 驱动需要以下几个步骤才能完成对硬件的访问和操作&#xff1a; 模块加载函数 module_init注册主次设备号 <应用程序通过设备号找到设备>驱动设备文件 <应用程序访问驱动的方式> 1、手动创建 &#xff08;mknod&#xff09;2、程序自动创建file_oper…...

《狂飙》大结局,这22句经典台词值得细品

最近爆火的热播剧《狂飙》大家都看了吗&#xff1f; 剧情紧凑、演技炸裂、豆瓣评分9.0&#xff0c;可以说是开年评分最高的一部国产剧。 ​ 虽然大结局了。 里面有很多经典台词&#xff0c;值得每个人细细品味。 01 这世界不缺梦想 有本事你就去实现它 02 你这么善良 怎么跟坏…...

【计算机网络期末复习】第二章 物理层

✍个人博客&#xff1a;https://blog.csdn.net/Newin2020?spm1011.2415.3001.5343 &#x1f4e3;专栏定位&#xff1a;为想复习学校计算机网络课程的同学提供重点大纲&#xff0c;帮助大家渡过期末考~ &#x1f4da;专栏地址&#xff1a; ❤️如果有收获的话&#xff0c;欢迎点…...

多核异构核间通信-mailbox/RPMsg 介绍及实验

1. 多核异构核间通信 由于MP157是一款多核异构的芯片&#xff0c;其中既包含的高性能的A7核及实时性强的M4内核&#xff0c;那么这两种处理器在工作时&#xff0c;怎么互相协调配合呢&#xff1f; 这就涉及到了核间通信的概念了。 IPCC (inter-processor communication contr…...

【Rust日报】2023-02-11 从头开始构建云数据库 RisingWave - 为什么我们从 C++ 转向 Rust...

GTK4发布v0.60gtk4-rs代码库包含GTK4的Rust crates。还有个庞大的GObject库生态系统&#xff0c;其中许多库基于gtk-rs中包含的Rust绑定工具。 特别是&#xff1a;gtk-rs-core&#xff0c;一些核心库的绑定&#xff0c;例如 glib、gio、pango、graphenegstreamer-rs&#xff0c…...

Linux驱动开发(一)

linux驱动学习记录 一、背景 在开始学习我的linux驱动之旅之前&#xff0c;先提一下题外话&#xff0c;我是一个c语言应用层开发工作人员&#xff0c;在工作当中往往会和硬件直接进行数据的交互&#xff0c;往往遇到数据不通的情况&#xff0c;常常难以定位&#xff0c;而恰巧…...

Spring MVC 之返回数据(静态页面、非静态页面、JSON对象、请求转发与请求重定向)

文章目录1. 默认情况下返回静态页面2. 返回一个非静态页面的数据2.1 ResponseBody 返回页面内容2.2 RestController ResponseBody Controller3. 实现登录功能&#xff0c;返回 JSON 对象3.1 前端使⽤ ajax&#xff0c;后端返回 json 给前端3.2 前端发送 JSON 的标准格式4. 请…...

赣州有没有做网站的/网站设计制作

点击蓝字关注我获取 高效/实用/好玩 的工具软件和教程前言大家电脑都会有一些比较重要和私密的文件&#xff0c;不方便给他人看到。比如账号密码、个人信息、工作资料等&#xff0c;这些数据泄漏出去会很麻烦……虽然现在大多都有开机密码&#xff0c;但也有借用电脑给别人的情…...

网络运营者开展经营和服务活动必须遵守法律行政法规/什么是搜索引擎优化?

题目1-平衡二叉树 给定一个二叉树&#xff0c;判断它是否是高度平衡的二叉树。 本题中&#xff0c;一棵高度平衡二叉树定义为&#xff1a; 一个二叉树每个节点 的左右两个子树的高度差的绝对值不超过1。 示例 1: 给定二叉树 [3,9,20,null,null,15,7] 3 / \ 9 2…...

怎样自己做免费的网站/seo外贸网站制作

描述&#xff1a;SpringBoot使用jdbc链接mysql时出错&#xff1a; yml数据源配置信息&#xff1a; 修改 在连接url后面加上useUnicodetrue&characterEncodingUTF-8&serverTimezoneUTC 其中UTC是统一标准世界时间,而useUnicode和characterEncoding解决乱码问题...

微页制作网站模板下载/企业培训心得体会

import java.awt.*;import javax.swing.*;import java.awt.event.*;import java.awt.image.*;import javax.imageio.*;import java.io.*;public class Gobang{// 下面三个位图分别代表棋盘、黑子、白子BufferedImage table;BufferedImage black;BufferedImage white;// 当鼠标移…...

彩票网站建设哪里/seo 推广怎么做

昨天米醋跟大家分享一款聊天界面的设计今天我们继续来制作个人中心页和聊天页两个页面&#xff0c;快来一起学习吧&#xff01;设计个人中心页效果图(1)启动PhotoshopCC 2017&#xff0c;按【CtrlN】组合键新建一个文件&#xff0c;将【文档类型】设置为【画板】&#xff0c;将…...

wordpress 统计2次/设计师培训班多少钱

【编者按】在编程时&#xff0c;开发人员常常会遭遇各式各样莫名错误。近日。Sushil Das在 Geek On Java上列举了 Java 开发中常见的 5 个错误&#xff0c;与君共「免」。 以下为译文&#xff1a; 1. Null 的过度使用 避免过度使用 null 值是一个最佳实践。比如。更好的做法是让…...