Spring Boot从服务中调用Rest Controller方法

时间:2018-08-10 07:35:08

标签: java spring spring-boot spring-rabbitmq

我正在使用Spring Boot从服务中调用rest控制器方法。调用该方法时,出现错误java.lang.NullPointerException广义的情况是,我的服务从RabbitMQ队列接收一个有效负载,并提取该有效负载的内容,然后将其传递给控制器​​以保存到数据库中。队列部分起作用(我可以从队列中接收消息并提取内容)。数据库部分也起作用。问题是从服务中调用控制器方法。

这是我的休息管理员

@RestController
@RequestMapping("/auth")
public class AuthController implements AuthService {
    @Autowired
    RabbitTemplate rabbitTemplate;

    @Autowired
    AuthRepository authRepository;


    public AuthModel addAuthenticatable(AuthModel auth){
        auth.setCreatedAt(DateTimeUtility.getDateTime());
        return authRepository.save(auth);
    }
}

我的服务代码:

public class QueueListener extends AuthController implements MessageListener{
    private String identifier;
    private JSONArray globalObject;
    private int userId;
    private String pin;

    @Autowired
    AuthController authController;

    @Override
    public void onMessage(Message message) {
        String msg = new String(message.getBody());
        String output = msg.replaceAll("\\\\", "");
        String jsonified = output.substring(1, output.length()-1);

        JSONArray obj = new JSONArray(jsonified);
        this.globalObject = obj;
        this.identifier = obj.getJSONObject(0).getString("identifier");
        resolveMessage();
    }
    public void resolveMessage() {
        if(identifier.equalsIgnoreCase("ADD_TO_AUTH")) {
            for(int i = 0; i < globalObject.length(); i++){ 
                JSONObject o = globalObject.getJSONObject(i);
                this.userId = Integer.parseInt(o.getString("userId"));
                this.pin = o.getString("pin");
            }

            AuthModel authModel = new AuthModel();
            authModel.setUserId(userId);
            authModel.setPin(pin);

            authController.addAuthenticatable(authModel);
        }
    }
}

当我调用AuthController中的方法addAuthenticatable()时发生错误。任何帮助将不胜感激。

3 个答案:

答案 0 :(得分:2)

问题是您的班级QueueListener不是spring bean。因此,您拥有的所有@Autowired变量将不会被注入。使用注解@Service@Component .....

中的一个将其变为spring bean。

例如。

@Component
public class QueueListener extends AuthController implements MessageListener{
   .....
}

不能说这将完全解决您的问题(由于您的超级类),因为我没有足够的信息。但是我相信它将消除NPE。

答案 1 :(得分:1)

我希望这不会超出主题,但是通常我们想要实现的是一种洋葱架构。依赖关系应该有一个方向。

您的控制器是您的应用程序的集成点。您希望每个REST 触发某些逻辑的执行。您的控制器不应扩展与业务逻辑有关的类或实现接口。这部分属于另一层。

关于逻辑的一切都属于服务:

@Service
public class AuthService {
    @Autowired
    private AuthRepository authRepository;

    private String attribute;

    public boolean isAuthenticated(String username) {
        authRepository.doSomething();
        //implementation of the logic to check if a user is authenticated.
    }

   public boolean authenticate(String username, char[] password) {
       // implementation of logic to authenticate.
       authRepository.authenticate();
   }

   public AuthModel save(AuthModel model) {
       //implementation of saving the model
   }

}

在服务层中提取逻辑,使事情可重用。 现在您可以将服务注入controller

@RestController
@RequestMapping("/auth")
public class AuthController {

   @Autowired
   private AuthService authService;

   public AuthModel addAuthenticatable(AuthModel auth){
       //process input etc..
       return authService.save(auth);
   }
}

或在amqListener

@Component
public class QueueListener implements MessageListener {
   @Autowired
   private AuthService authService;

   @Autowired
   private SomeOtherService otherService;

   @Override
   public void onMessage(Message message) {
      JSONArray array = processInput();

      JSONArray obj = new JSONArray(jsonified);
      String identifier = obj.getJSONObject(0).getString("identifier");
      // extract the business logic to the service layer. Don't mix layer responsibilities
      otherService.doYourThing(obj, identifier);

      resolveMessage();
  }

  private JSONArray processInput(Message message) {
     String msg = new String(message.getBody());
     String output = msg.replaceAll("\\\\", "");
     String jsonified = output.substring(1, output.length()-1);


}

和您的配置,以便您可以让spring知道在哪里寻找带注释的类。

@Configuration
@ComponentScan({"your.service.packages"})
@EntityScan(basePackages = "your.model.package")
@EnableJpaRepositories("your.repository.packages")
@EnableRabbit // probaby
@EnableWebMvc // probably
public class Config {
   //you could also define other beans here
   @Bean
   public SomeBean someBean() {
       return new SomeBean();
   }
}

@pvpkiran回答了您的实际问题。希望对您有帮助

答案 2 :(得分:0)

不需要扩展自动连接该控制器。如果要扩展,则不需要在扩展类中放置依赖项,这只会使设计复杂化。因此,要么通过自动装配来使控制器成为依赖项,要么对其进行扩展(在这种情况下,不需要它具有依赖项)。 使用@Autowired时-确保按照建议正确注释所有类,并确保它们已被主应用程序类扫描。 docs中推荐了一些好的做法。

编辑 控制器使用@Controller进行注释,服务类使用@Service进行注释,该类基本上是经过修饰的@Component注释。 docs很好地解释了所有这些,这是一个开始探索的好地方。

返回问题:

如果您仍在使用NPE,请检查您的项目结构,可能是因为Spring的组件扫描未拾取您的类,或者未注释其中一个类。您的主类(用@SpringBootApplication注释的主类)应位于根目录/默认包中,以便扫描所有其他@Component