Spring Boot 自动装配bean为空

作者: 不返y | 来源:发表于2024-04-13 17:11 被阅读0次

Spring Boot 无法自动装配静态成员变量以及new出的对象,导致运行时出现空指针异常的解决方法。

问题:

在项目中使用 WebSocket便于双向通信,在类 WebSocketServer中进行了基础的配置后,在其内部封装了一些发送信息的方法,因此我希望它需要被外界调用。


WebSocketServer中的部分代码:

/**
 * websocket信息处理
 */
@Slf4j
@Component
@ServerEndpoint(value = "/ws/{sid}") //将类定义成一个WebSocket服务器端
public class WebSocketServer {

        //存放会话对象
    private static Map<String, Session> sessionMap = new HashMap();

   ......

    /**
     * 群发消息
     *
     * @param message 发送的消息
     */
    public void sendToAllClient(String message) {
        Collection<Session> sessions = sessionMap.values();
        for (Session session : sessions) {
            //服务器向客户端发送消息
            try {
                session.getBasicRemote().sendText(message);
            } catch (IOException e) {
                e.printStackTrace();
            }
        }s
    }
    
    ......
    
}

因为此类已经添加了@Component注解交付给Spring容器管理,所以我在我的工具类MyUtil中通过@Autowired注解自动注入,以便于使用。


MyUtil工具中的部分代码:

@Component
public class MyUtil {

    @Autowired //自动注入
    private static WebSocketServer webSocketServer;

        //使用服务
    public static void useServer() {
        //群发消息
        webSocketServer.sendToAllClient("hello");
    }
    
}

紧接着我们模拟使用场景,在 SpringText 下进行使用测试。看看会怎么样

@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
class XXXServerApplicationTests {
    @Test
    public void autowiredText() {
        MyUtil.useServer();
    }
}

很好,很快完美的报错,null指针异常。

NullPointerException

原因:

这个原因是 Spring Boot 不会自动装配静态的成员(这个看上面的结果应该也能看出来,因为静态的成员一般是不允许实体对象所调用的,所以Spring所管理的bean对象无法通过对象.set的方法去自动注入bean)。

其实在我后面寻找解决方案的时候发现,Spring Boot 也不会自动装配新new出来的对象(因为新new出来的对象都不是spring容器所管理的,所以肯定也不能完成自动装配,但是一般人也不会像我这样new出来使用吧哈)。

解决方法:

既然 Spring Boot 不会自动装配,那就只能通过最原始的方法,即通过获取ApplicationContext对象获取spring 的 bean 对象。

具体实现方法是实现一个类,该类实现ApplicationContextAware接口,并且重写其setApplicationContext()方法,以存储spring容器对象(目的就是为了获取ApplicationContext并存储进此类中)。然后通过容器对象获取spring的bean;

具体代码如下:

package com.anyi.util;

import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;

import javax.annotation.Nonnull;

/**
 * springContext ,实现接口,进行 springBean 的手动装配
 */
@Component
public class ManagementSpringBeans implements ApplicationContextAware {

    /**
     * 创建 ApplicationContext 保存spring容器对象
     */
    private static ApplicationContext context;

    /**
     * 通过class文件获取对应的bean你
     *
     * @param requiredType 请求类的class文件
     * @param <T>          请求bean的类
     * @return 对应的bean
     */
    public static <T> T getBean(Class<T> requiredType) {
        //通过 类class 获取对象
        return context.getBean(requiredType);
    }

    /**
     * 重写方法获取spring的上下文对象
     *
     * @param applicationContext spring容器
     * @throws BeansException beans异常
     */
    @Override
    public void setApplicationContext(@Nonnull ApplicationContext applicationContext) throws BeansException {
        //存储spring容器对象
        ManagementSpringBeans.context = applicationContext;
    }
}

实现完成这个 ManagementSpringBeans 后,以后要想注入bean的对象,就可以调用其内部的 getBean() 静态方法,传参对应的类的class就可获取对应的bean,手动装配。具体实现如下所示:

public class MyUtil {
  
    private static WebSocketServer webSocketServer = ManagementSpringBeans.getBean(WebSocketServer.class);

    public static void useServer() {
        //群发消息
        webSocketServer.sendToAllClient("hello");
    }
  
}
@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
class XXXServerApplicationTests {
    @Test
    public void autowiredText() {
        MyUtil.useServer();
    }
}

成功 !

成功

注意:使用这个方法当然很好,因为几乎可以在任何地方都能装配bean了。但是这个方案太自由了,可能会造成代码上的一些破坏。(就和原先的 goto语句一样)

所以虽然这是一个解决方法但是也不是那么的好。最后我还是使用原生的 springboot自动装配来解决问题。当你在开发时使用这个方法进行手动装配的时候,可能是因为原始的设计布局有些问题。

相关文章

网友评论

    本文标题:Spring Boot 自动装配bean为空

    本文链接:https://www.haomeiwen.com/subject/xwhlxjtx.html