Java.SpringBoot InputStream流复制.

本文档产生场景.

在项目中,由于需要一个服务做为文件中转,先定义这个中转服务为C。C服务需要将内部文件服务器(这里定义为A)的文件转移到新的文件服务器(定义为B中)。由于编程经验不足,在C中直接将A的文件读成inputStream流,然后将此inputStream流交给B。在测试环境,由于网络正常,没有出现过任何问题,或者说问题不明显,但在生产环境中,由于网络速度的不对称,造成上传文件时好时坏。以下,将些问题进行记录。

这个问题出现,会报以下错误:

Premature end of Content-Length delimited message body (expected: 11,791,936; received: 3,011,273)
org.springframework.web.util.NestedServletException: Request processing failed; nested exception is org.springframework.web.multipart.MultipartException: Failed to parse multipart servlet request; nested exception is java.io.IOException: org.apache.tomcat.util.http.fileupload.FileUploadBase$IOFileUploadException: Processing of multipart/form-data request failed. Unexpected EOF read on the socket
Caused by: org.apache.tomcat.util.http.fileupload.FileUploadBase$IOFileUploadException: Processing of multipart/form-data request failed. Unexpected EOF read on the socket
        at org.apache.tomcat.util.http.fileupload.FileUploadBase.parseRequest(FileUploadBase.java:297)
        at org.apache.catalina.connector.Request.parseParts(Request.java:2884)
        ... 33 more
1. 问题代码

由于不能将所有代码都复制到这里,因此,这里的代码可以认为是半伪代码.

CloseableHttpClient httpClient = HttpClients.createDefault();
// 设置CloseableHttpClient的超时时间.
RequestConfig requestConfig = RequestConfig.custom().setConnectionRequestTimeout(100000)
                    .setSocketTimeout(100000)
                    .setStaleConnectionCheckEnabled(true)
                    .setConnectTimeout(100000).build();
httpPost.setConfig(requestConfig);
// uploadUrl  这个是上传服务器,也是在前面问题说到的B服务地址.
HttpPost httpPost = new HttpPost(uploadUrl);
try {
   // 此部分只是一个下载地址.
   // 可以认为是执行下载.
    InputStream inputStreamTemp = 从A服务器获取的InputStream流
    // 将获取到的inputStreamTemp流放入BufferedInputStream.
    // 这一步如果只针对下载文件,可以很好的提高效率,但是如果做为一个中转,是没有办法实现的.
    BufferedInputStream inputStreamBuffer = new BufferedInputStream(inputStreamTemp,1024*1024*100);
    // 将inputStreamBuffer放入HttpEntity, 这里的代码未写全,如果需要,可根据实际情况进行写.
    HttpEntity reqEntity = MultipartEntityBuilder.create().addBinaryBody(
                    "file", inputStreamBuffer,
                    ContentType.MULTIPART_FORM_DATA
                  ).build();
    httpPost.setEntity(reqEntity);
    //执行上传
    CloseableHttpResponse response = httpClient.execute(httpPost);
}

上面的代码,如果在测试环境,不会出任何问题,因在测试环境的网络非常好,下载和上载都有很高的速度,但是放在生产环境,A和B不在一个网段,因此,这样操作就会时不时的出现问题。

以上就是这段代码的问题所在。接下来,我们需要对这段代码进行更改,让其可以在生产环境中使用。

当然,有人会问到,为什么不先把文件下载到本地,然后再把文件上传到服务器呢?
答: 不想在中转服务器生成任何文件或临时文件,不喜欢。

2. 正常代码

CloseableHttpClient httpClient = HttpClients.createDefault();
RequestConfig requestConfig = RequestConfig.custom().setConnectionRequestTimeout(100000)
                    .setSocketTimeout(100000)
                    .setStaleConnectionCheckEnabled(true)
                    .setConnectTimeout(100000).build();
httpPost.setConfig(requestConfig);
HttpPost httpPost = new HttpPost(uploadUrl);
try {
    InputStream inputStreamTemp = 从A服务器获取的InputStream流
    
    **以上都没有作改变,所以把注释就删除了**

   // 创建一个ByteArrayOutputStream,用于存放复制流.
    ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
    // 将从A服务器获取的InputStream流inputStreamTemp 复制到 byteArrayOutputStream流中.
    byte[] buffer = new byte[1024];
            int len;
            while ((len = inputStreamTemp.read(buffer)) > -1) {
                byteArrayOutputStream.write(buffer,0,len);
            }
            byteArrayOutputStream.flush();

    // 接下来,将复制的流byteArrayOutputStream转换成上传流uploadInputStream
    InputStream uploadInputStream = new ByteArrayInputStream(byteArrayOutputStream.toByteArray());
    // 将inputStreamBuffer放入HttpEntity, 这里的代码未写全,如果需要,可根据实际情况进行写.
    HttpEntity reqEntity = MultipartEntityBuilder.create().addBinaryBody(
                    "file", inputStreamBuffer,
                    ContentType.MULTIPART_FORM_DATA
                  ).build();
    httpPost.setEntity(reqEntity);
    //执行上传
    CloseableHttpResponse response = httpClient.execute(httpPost);
}

以上就是流复制的简单方法,以及代码.

欢迎关注,及提问.


如果有什么问题,可以加入QQ群进行讨论。QQ群:839421316

谢谢支持。
在这里插入图片描述

<think>嗯,用户遇到了Java中ApplicationContext导致的NullPointerException问题,想要知道原因和解决方案。首先,我需要回忆一下Spring框架中ApplicationContext的加载机制。记得ApplicationContext负责管理Bean的生命周期,而如果Bean在初始化时没有被正确注入依赖,就会导致NullPointerException。 用户提供的引用里有几个关键点。引用[1]提到静态代码块中使用@Autowired会导致NullPointerException,因为静态代码块执行时Spring还没初始化Bean。这时候应该避免在静态块里使用依赖注入的Bean,或者用@PostConstruct方法。引用[2]指出资源文件未正确加载到target目录,导致初始化失败,需要检查文件位置并重新编译。引用[3]提到Spring整合Junit时的NullPointerException,可能和测试配置有关,比如没有使用正确的Runner或ContextConfiguration。 接下来,我要整理常见的原因。首先,Bean未被正确注入,可能因为缺少注解如@Service或@Autowired,或者组件扫描配置错误。然后是生命周期问题,比如在构造函数中使用未初始化的Bean,应该改用@PostConstruct。静态方法或代码块中的依赖注入也会有问题,因为静态成员不属于Spring管理的Bean。资源文件未加载的情况,比如文件不在classpath下,或者构建工具没有正确复制。测试环境配置错误,比如Junit没有配置SpringRunner,或者上下文配置不正确。 解决方案部分,针对每个原因给出对应的解决办法。比如检查注解是否正确,使用@PostConstruct,确保资源文件位置正确,重新编译项目,配置测试上下文等。用户可能遇到的问题还可能包括循环依赖,或者作用域不正确,比如prototype作用域的Bean被错误使用。不过根据提供的引用,重点应该放在用户提到的几个案例上。 需要确保回答结构清晰,分点说明原因和解决方案,同时结合用户提供的引用,自然添加引用标识。可能用户是Spring新手,在配置Bean或测试时遇到了问题,需要具体的步骤指导,比如如何检查组件扫描路径,如何配置测试类。此外,用户可能没有意识到静态代码块和构造函数中的初始化顺序问题,需要强调这些点。 最后,生成相关问题,帮助用户进一步理解可能的相关问题,比如如何正确初始化Bean,测试配置注意事项,资源文件处理等。这些问题应该围绕用户遇到的NullPointerException展开,覆盖常见的原因和解决方案。</think>### Java ApplicationContext导致的NullPointerException原因及解决方案 #### 一、常见原因分析 1. **Bean未正确注入** - 使用`@Autowired`或`@Resource`注解的字段/方法未被Spring管理(如未添加`@Component`、`@Service`等注解)[^1] - 组件扫描范围未包含目标Bean所在包,例如`@ComponentScan(basePackages = "com.example")`配置错误 - 构造函数中直接调用未初始化的Bean方法(Spring依赖注入发生在对象构造之后) 2. **生命周期冲突** - 静态代码块或静态方法中尝试访问Spring管理的Bean(静态代码块执行时,Spring容器尚未完成初始化)[^1] - 在`@PostConstruct`方法之外过早调用Bean方法 3. **资源文件加载失败** - 配置文件未正确放置在`resources`目录,或构建工具未将其复制到`target/classes`目录[^2] - 文件路径使用硬编码(如`new File("data.txt")`)而非类路径引用(如`getClass().getResourceAsStream("/data.txt")`) 4. **测试环境配置错误** - Junit测试类未添加`@RunWith(SpringRunner.class)`注解[^3] - 缺少`@ContextConfiguration(classes = {AppConfig.class})`上下文配置 #### 二、解决方案步骤 1. **基础检查** - 确认目标类已添加`@Component`/`@Service`等注解 - 检查组件扫描配置是否包含当前包: ```java @SpringBootApplication(scanBasePackages = "com.example") ``` - 在Bean初始化方法使用`@PostConstruct`代替构造函数: ```java @Service public class MyService { @Autowired private Dependency dep; @PostConstruct public void init() { dep.initialize(); // 正确时机调用 } } ``` 2. **静态代码块问题处理** - 避免在静态上下文中直接使用Bean,改用延迟加载: ```java private static MyBean myBean; @Autowired public void setMyBean(MyBean myBean) { MyClass.myBean = myBean; } ``` 3. **资源文件加载方案** - 将文件放在`src/main/resources`目录下 - 使用Maven/Gradle重新构建: ```bash mvn clean compile # 强制重新生成target目录 ``` - 通过类加载器读取资源: ```java InputStream input = getClass().getResourceAsStream("/config.txt"); ``` 4. **测试环境修复** - 完整测试类配置示例: ```java @RunWith(SpringRunner.class) @SpringBootTest public class MyTest { @Autowired private MyService service; // 正确注入 } ``` #### 三、进阶调试技巧 1. **堆栈分析** 定位报错行号,重点关注`NullPointerException`发生位置的前序调用链 2. **依赖检查** 使用`applicationContext.getBean("beanName")`手动验证Bean是否存在 3. **作用域验证** 检查是否误用`@Scope("prototype")`导致实例化异常
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值