Spring Boot 跨线程调用对象报空指针异常(NullPointerException)的解决办法

参考链接: http://blog.sina.com.cn/s/blog_5a15b7d10102v6ki.html

问题发生的情景:

建立一个监听线程来监听数据库内容,但是线程中调用了主线程注入的对象,导致线程在执行的时候该对象没有被注入,bean对象为空值。

因此采取上述链接中的办法,创建了一个工具类获取主线程的对象。以下为原文内容

问题说明
今天在web应用中用到了Java多线程的技术来并发处理一些业务,但在执行时一直会报NullPointerException的错误,问题定位了一下发现是线程中的Spring
bean没有被注入,bean对象的值为null。

原因分析 web容器在启动应用时,并没有提前将线程中的bean注入(在线程启动前,web容器也是无法感知的)

解决方案 方法有多种,网上也看到了不少。

  1. 使用static声明变量 http://blog.csdn.net/bjamosgavin/article/details/6125497
    这个方法自己试了一下但是没有成功。。。
  2. 把线程设置为主程序的内部类 这也是一种简单的方法,主程序在web容器加载时肯定是可以注入Spring bean的,那么将线程的实现类放在主程序的类中便可以“共享”Spring的bean,(当然,这需要提前把线程中的需要用到的bean定义在外层的类中)。
    具体操作方法,就是将生成线程的线程池定义在主程序的类中,每个线程的实现类作为内部类也定义在主程序中。这个方法自己试过,是可以的。
  3. 使用静态方法直接取的容器中的spring对象 这个方法稍微专业点,可以线程的分发与线程的实现分离出来。在每个线程中使用静态方法直接取的容器中的spring对象。
    使用静态方法获取容器中的spring对象可以参见 http://littie1987.iteye.com/blog/937877,
    或者http://my.oschina.net/skyline520/blog/181158?p={{page}}
    但一定要记住,你定义这个工具类也要配置成spring中的bean!

下面贴一下我在使用时的代码 (1)定义工具类


public class SpringApplicationContextHolder  implements
        ApplicationContextAware {
 
     private static ApplicationContext context;
 
     @Override
     public void setApplicationContext(ApplicationContext context) throws BeansException {
        SpringApplicationContextHolder.context = context;
     }
 
    
     public static Object getSpringBean(String beanName) {
        notEmpty(beanName, "bean name is required");
        return context==null?null:context.getBean(beanName);
     }
 
     public static String[] getBeanDefinitionNames() {
        return context.getBeanDefinitionNames();
     } 
}
      

(2)在Spring中注册工具类的bean,在SpringBoot中即为工具类添加@Component注解
(3)线程中获取bean UserRepo user = (UserRepo)SpringApplicationContextHolder.getSpringBean(“userRepo”);

### 回答1: 在 Spring Boot 中使用 Principal 注入的方法如下: 1. 在 Controller 类的方法参数中加入 Principal 类型的参数,Spring Boot 会自动将当前用户的 Principal 注入进来。 例如: ``` @GetMapping("/user") public String getUser(Principal principal) { return "Hello, " + principal.getName(); } ``` 2. 在 Controller 类中使用 @AuthenticationPrincipal 注解标注一个方法参数,Spring Boot 会将当前用户的 Principal 注入进来。 例如: ``` @GetMapping("/user") public String getUser(@AuthenticationPrincipal User user) { return "Hello, " + user.getUsername(); } ``` 需要注意的是,使用 @AuthenticationPrincipal 注解时,方法参数的类型不能是 Principal,而必须是你在身份认证过程中使用的身份对象的类型(例如:User)。 ### 回答2: 在Spring Boot中,我们可以通过注解的方式将check/token Principal注入到我们的应用程序中。 首先,我们需要在我们的应用程序中使用`@EnableWebSecurity`注解来启用Web安全配置。然后,我们需要创建一个实现了`UserDetailsService`接口的类,来加载用户的信息。 具体步骤如下所示: 1. 首先,我们需要创建一个实体类来表示用户信息,该实体类需要实现`UserDetails`接口,并重写相应的方法,如`getUsername()`、`getPassword()`等。 2. 接下来,我们需要创建一个实现了`UserDetailsService`接口的类,该类用于加载用户的信息。在该类中,我们需要重写`loadUserByUsername()`方法,根据用户名从数据库或其他数据源中加载对应的用户信息,并返回一个`UserDetails`对象。 3. 然后,我们可以在其他需要使用用户信息的地方通过依赖注入的方式来获取该信息。可以使用`@Autowired`注解将`UserDetailsService`实例注入到其他类中,然后调用`loadUserByUsername()`方法获取用户信息,并进行相应的操作。 另外,如果我们需要在控制器中获取当前用户的信息,可以使用`@AuthenticationPrincipal`注解来注入用户信息。在方法参数中使用该注解,Spring Boot会自动将当前用户的信息注入到该参数中。 总结来说,在Spring Boot中注入check/token Principal的方式主要有两种: 1. 通过注入`UserDetailsService`实例来获取用户信息,然后在需要使用的地方调用`loadUserByUsername()`方法获取用户信息。 2. 在控制器中使用`@AuthenticationPrincipal`注解来注入当前用户的信息。 ### 回答3: 在Spring Boot中,我们可以使用Spring Security框架来进行身份验证和授权。在进行身份验证时,Spring Security将会为每个认证过的用户生成一个Principal对象,该对象包含了用户的信息和角色。 在Spring Boot中注入Principal对象有以下几种方法: 1. 在Controller中注入Principal对象: 在Controller方法的参数列表中添加Principal对象Spring Boot会自动将认证用户的Principal对象注入进来,代码示例如下: ```java @RestController public class MyController { @GetMapping("/user") public String getUser(Principal principal) { // 使用Principal对象获取用户信息 String username = principal.getName(); // ... } } ``` 2. 在Service或Component中使用SecurityContextHolder获取Principal对象: 通过SecurityContextHolder获取SecurityContext对象,在SecurityContext对象中可以获取Principal对象。代码示例如下: ```java @Service public class MyService { public void doSomething() { SecurityContext context = SecurityContextHolder.getContext(); Principal principal = context.getAuthentication().getPrincipal(); // 使用Principal对象获取用户信息 String username = principal.getName(); // ... } } ``` 需要注意的是,在使用第二种方法时,必须确保在调用doSomething方法之前,用户已经进行了身份验证,否则可能会出现NullPointerException。 总结来说,Spring Boot中注入Principal对象可以通过在Controller中将Principal对象作为方法参数来实现,也可以在Service或Component中通过SecurityContextHolder获取SecurityContext对象,并从中获取Principal对象
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值