Springboot 项目测试环境OK,产线环境找不到文件。

本文介绍了一个在产线环境中遇到的文件读取问题,即通过特定路径读取文件时,在测试环境正常但在产线环境报错。文章详细解释了原因在于测试环境与产线环境对文件路径解析的不同,并提供了有效的解决方案,包括使用Spring注解方法注入Resource对象,以及以流的方式读取文件。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

项目在测试环境正常发布,但是到了产线环境报/xxx/xxx.jar!/BOOT-INF/classes!/xx.xx;

原因:在测试环境照文件路径跟产线jar包寻找文件路径的方法不一样。

测试环境可以通过
 

java.io.File.File secretFile= org.springframework.util.ResourceUtils.getFile(hylOpenApiProperties.getFilePath());

直接获取指定路径下的文件。

但是产线jar包中该方法会报File not exists。

解决方法

通过Spring注解方法注入Resource

	@Value(value="classpath:secret/local/gnete_wg.pfx")
	private Resource resource;

然后以流的方式获取

BufferedReader br = new BufferedReader(new InputStreamReader(resource.getInputStream()));
StringBuffer message=new StringBuffer();
String line = null;
while((line = br.readLine()) != null) {
	message.append(line);
}
String defaultString=message.toString();
String result=defaultString.replace("\r\n", "").replaceAll(" +", "");
System.out.println(result);

// 转换成File
org.apache.commons.io.FileUtils.copyInputStreamToFile(resource.getInputStream(), secretFile);

试了别的读取方式,但是没成功,先做记录以后再做尝试

// 读取文件为空
//		InputStream secretStream = getClass().getClassLoader().getResourceAsStream("path");
//		FileUtils.copyInputStreamToFile(secretStream, secretFile);
		
		// resource 读取路径
//		ClassPathResource resource = new ClassPathResource("path");
//		File sourceFile = resource.getFile();


 

### SpringBootTest 中 @Autowired 注解报错的原因分析 在 `@SpringBootTest` 测试类中,如果发现 `@Autowired` 注解无法正常工作并引发错误,则可能由以下几个原因引起: #### 1. **未正确加载上下文** 如果测试类上缺少必要的注解来指示 Spring 加载应用程序上下文,可能会导致依赖注入失败。通常情况下,`@SpringBootTest` 应该能够自动配置整个应用环境[^1]。 解决方案:确保测试类被标记为 `@SpringBootTest` 并且包含 `@RunWith(SpringRunner.class)` 或其替代品(如 JUnit 5 的 `@ExtendWith(SpringExtension.class)`)。例如: ```java import org.junit.jupiter.api.Test; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.test.context.junit4.SpringRunner; @SpringBootTest public class MyTest { // Test cases here... } ``` #### 2. **Bean 定义缺失或不匹配** 当目标 Bean 没有定义或者存在多个候选者时,`@Autowired` 可能会抛出异常。这通常是由于组件扫描路径设置不当、Bean 名称冲突或其他配置问题引起的[^2]。 解决方法:确认相关类已通过适当的方式注册到 Spring 容器中,比如使用 `@Component`, `@Service`, `@Repository` 等注解标注,并验证这些类位于正确的包结构下以便于组件扫描可以到它们。如果有多个相同类型的 Beans 存在,考虑利用限定符(`@Qualifier`)指定具体哪一个实例应该被注入。 #### 3. **Mockito 使用中的潜在干扰** 在某些场景下,当 Mockito 被引入项目用于模拟对象创建时,它可能导致真实的依赖关系未能建立起来从而影响正常的 DI 过程。特别是当你尝试同时运用 `@MockBean` 和标准的 `@Autowired` 结合在一起的时候需要注意这一点[^3]。 处理方式:仔细审查你的单元/集成测试代码逻辑,避免不必要的混淆;对于那些确实需要替换成 mock 版本的服务层接口可以直接采用 `@MockBean` 替代原始形式的声明即可解决问题。 --- 以下是修正后的完整示例代码片段展示如何正确实现上述建议: ```java import static org.mockito.Mockito.*; import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest; import org.springframework.boot.test.mock.mockito.MockBean; import org.springframework.http.MediaType; import org.springframework.test.web.servlet.MockMvc; import com.example.demo.controller.MyController; import com.example.demo.service.MyService; @WebMvcTest(MyController.class) public class ControllerTests { private MockMvc mockMvc; @MockBean private MyService myService; @BeforeEach void setup() throws Exception {} @Test public void testExampleEndpoint() throws Exception{ when(myService.performAction()).thenReturn("mocked response"); this.mockMvc.perform(get("/example-endpoint")) .andExpect(status().isOk()) .andExpect(content().string(containsString("mocked"))); verify(myService, times(1)).performAction(); } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值