微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

来自Spring Boot的@Autowired服务在WAR中为空

如何解决来自Spring Boot的@Autowired服务在WAR中为空

首先,对于给您带来的不便,我深表歉意。我之所以进行咨询,是因为我花了数天时间解决此问题,但找不到合适的解决方案。我有一个使用swagger代码生成器设计的名为fact-electronica-api的项目,通过该项目,我具有执行REST调用的路径。一旦用户进行了这种类型的调用,api项目就会与另一个具有使用Spring Boot的核心并使用数据库执行查询的项目进行通信。我遇到的问题是,当我从api自动连接到核心中定义的服务时,此参数为null。

打扰一下,我来自乌拉圭,希望您能理解我所面临的问题。为进一步理解,请附上图片

  • 事实电子产品Api

enter image description here

  • Factelec核心

enter image description here

服务,通过该服务我可以接收剩余呼叫并与核心进行通信。

enter image description here

核心服务

enter image description here

对api的依赖

enter image description here

Spring Boot初始化程序,当您在Wildfly中启动耳朵时,该项目会自动启动并进行相关设置。

enter image description here

如果有人能指导我解决这个问题,我将不胜感激,

解决方法

似乎Get-Command -Name Copy-Item | Select-Object -ExpandProperty DLL 不是Spring bean,因此Spring不会考虑用 bottomNavigationView= findViewById(R.id.bottom_navigation); bottomNavigationView.setOnNavigationItemSelectedListener(new BottomNavigationView.OnNavigationItemSelectedListener() { @SuppressLint("NonConstantResourceId") @Override public boolean onNavigationItemSelected(@NonNull MenuItem item) { Fragment temp =null; switch (item.getItemId()) { case R.id.library: temp = new LibraryFragment(); break; case R.id.search: temp = new SearchFragment(); break; case R.id.profile: temp = new UserprofileFragment(); break; default: } getSupportFragmentManager().beginTransaction().replace(R.id.fragment,temp).commit(); return true; } }); getSupportFragmentManager().beginTransaction().replace(R.id.fragment,new SearchFragment()).commit(); 注释的所有类字段进行依赖项注入。尝试将此类注释为SolicitudApiServicelmpl,以将该类的实例添加到Spring上下文中,或为 @Auowired会生成Spring代理而不是简单的对象。

,

您没有创建SolicitudApiServicelmpl春天实例。

Spring上下文(女巫使@Autowired工作)需要您创建以Spring而不是Java原生new SolicitudApiServicelmpl()的实例。

您可以使用多种技术来创建该实例,这里有一些技术:

带有“手动接线”的静态上下文(不推荐)

public class Main {

   private static ConfigurableApplicationContext context;

   public static void main(String[] args) {
      context = SpringApplication.run(MyApplication.class,args);
   }

   public static ConfigurableApplicationContext getContext() {
      return context;
   }

}
public class SolicitudApiServicelmpl {

   private ConfigurableApplicationContext context = Main.getContext();

   private SolicitudInformacionService solInfServ = context.getBean(SolicitudInformacionService.class);

   // Now `solInfServ` is not null anymore.
   // ...
}

在Spring上下文中创建SolicitudApiServicelmpl实例

@Component
public class SomeClassWithSpringContext {

   @Autowired private SolicitudApiServicelmpl solicitudService;
   // with this @Autowired should work in the `service` instance.

}

或上下文实例:

@Component
public class SomeClassWithSpringContext {

   @Autowired private WebApplicationContext context;

   public void someMethodOrSomewhere() {
      SolicitudApiServicelmpl service = context.getBean(SolicitudApiServicelmpl.class);
      // with this @Autowired should work in the `service` instance.
   }

}
,

在努里奥·费尔南德斯(NurioFernández)和安德烈·科夫罗夫(Andrei Kovrov)给我的帮助之后,我得以找到正确的方法来完成我想做的事情。

首先,我将Spring不仅集成到api中,还将Spring Boot从核心移到api中,从而在api中实现了该类。

Api-ApplicationInitializer.java

@Order(Ordered.HIGHEST_PRECEDENCE)
@SpringBootApplication(scanBasePackages = {"com.factelcore.*","uy.com.antel.fact.*"})
@EnableJpaRepositories("com.factelcore.repository")
public class ApplicationInitializer extends SpringBootServletInitializer {
    
    public static void main(String[] args){
        SpringApplication.run(ApplicationInitializer.class,args);
    }
    
    @Override
    protected SpringApplicationBuilder configure(SpringApplicationBuilder application) {
        return application.sources(applicationClass);
    }
 
    private static Class<ApplicationInitializer> applicationClass = ApplicationInitializer.class;
    
}

通过这样做,我确保可以在api中使用spring上下文。当我从这里启动Spring时,我扫描了所有api和core文件,并对bean进行了适当的配置。

完成此操作后,我需要捕获上下文以恢复内核中定义的bean,因此我可以使用内核中定义的@Service类。为此,我创建了以下类:

Api-AppContextUtil.java

@Component
public class AppContextUtil implements ApplicationContextAware {
    
    @Autowired 
    private WebApplicationContext context;

    private static ApplicationContext appContext;
    
    @PostConstruct
    public void init() {
        if (context != null) {
            setApplicationContext(context);         
        }
    }

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        AppContextUtil.appContext = applicationContext;
    }

    public static ApplicationContext getContext() {
        return appContext;
    }
}

完成此操作后,使用SolicitudApiServiceImpl,我可以在特定时间执行以下操作来调用所需的bean:

Api-SolicitudApiServiceImpl.java

ApplicationContext cont = AppContextUtil.getContext();
SolicitudInformacionService solInfServ = cont.getBean(SolicitudInformacionService.class);

我不知道这是否是理想的方法,但就我而言,它对我有用,如果有人可以更好地做到这一点,请随时告诉我,以便我改进代码。非常感谢!

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。