编程语言
首页 > 编程语言> > Java-JAX-RS依赖项注入

Java-JAX-RS依赖项注入

作者:互联网

我已经使用Spring Rest完成了项目.现在,我们有一个小型休息项目,并计划与Jersey JAX-RS一起使用.我是新手,因此推荐SO和其他博客来成功实现具有依赖项注入的Rest api.

有以下代码.

AppConfig.java

import javax.ws.rs.ApplicationPath;
import javax.ws.rs.core.Application;

@ApplicationPath("/")
public class AppConfig extends Application {    
    @Override
    public Set<Class<?>> getClasses() {
        System.out.println("AppConfig");
        final Set<Class<?>> s = new HashSet<Class<?>>();
        s.add(Controller.class);
        s.add(AppFeature.class);
        return s;
    }
}

AppBinder.java

import org.glassfish.hk2.utilities.binding.AbstractBinder;

public class AppBinder extends AbstractBinder {
    @Override
    protected void configure() {
        System.out.println("AppBinder");
        bind(ReflectionService.class).to(ReflectionService.class);
    }
}

AppFeature.java

import javax.ws.rs.core.Feature;
import javax.ws.rs.core.FeatureContext;

public class AppFeature implements Feature {
    @Override
    public boolean configure(FeatureContext context) {
        System.out.println("AppFeature");
        context.register(new AppBinder());
        return true;
    }
}

Controller.java

@Path("/")
public class Controller {   
    @Inject
    Service service;    
    public Controller(){
        System.out.println("Controller created");
    }
    // other methods
}

Service.java

@Singleton
public class Service    
    public Service(){
        System.out.println("Service instance created");
    }
    // other methods
}

我假设Controller和Service的每个实例都是在Tomcat 8服务器启动时创建的,并且依赖项注入已完成.但是在启动过程中,将其放在控制台上

INFO: Registering the Jersey servlet application, named
com.sample.auto2.AppConfig, at the servlet mapping /*, with the
Application class of the same name.

AppConfig

AppConfig

Nov 15, 2016 12:22:20 PM org.glassfish.jersey.server.ApplicationHandler initialize
INFO: Initiating Jersey application, version Jersey: 2.6 2014-02-18
21:52:53…

AppFeature

AppBinder

Nov 15, 2016 12:22:21 PM
org.apache.catalina.startup.HostConfig deployDirectory

每次,我们发送一个请求,在控制台中得到关注

Service instance created

Controller created

我的问题

>服务,每当我们发送一个
http请求;它在每个请求中创建实例还是只是
调用构造函数?
>为什么AppConfig中的System.out被调用两次?
>是否有更好的方法来设置我的小型项目,该项目没有任何数据库访问权限并且只有三个发布端点?

编辑:

根据@Harikrishnan提供的链接,为Controller类添加了@Singleton.现在,构造函数仅调用一次(在第一个请求时-为什么在服务器启动期间不!).

但是,为什么Service类构造函数会在每个请求上调用(在将@Singleton添加到Controller之前),即使它是单例的呢?另外,其他问题仍然存在.

编辑2:

谢谢@peeskillet.所以这些对我来说是结果.

>这在第一次请求时仅调用一次构造函数

bind(ReflectionService.class).to(ReflectionService.class).in(Singleton.class);
bind(Controller.class).to(Controller.class).in(Singleton.class);

>这给http请求错误

bind(ReflectionService.class).to(ReflectionService.class).in(Immediate.class);

java.lang.IllegalStateException: Could not find an active context for org.glassfish.hk2.api.Immediate

没关系,因为
>这在服务器启动时称为构造函数,仅一次

bind(new ReflectionService()).to(ReflectionService.class);
bind(new Controller()).to(Controller.class);

>这样,启动时就完成了注入,但是http请求上却出现404错误. (在AppBinder中配置了控制器,然后在AppConfig中配置了)

@ApplicationPath("/")
public class AppConfig extends ResourceConfig  {    
    public AppConfig() {
        register(new AppBinder());
    }
}

>就像您所说的那样,它可以运行!

@ApplicationPath("/")
public class AppConfig extends ResourceConfig  {        
    public AppConfig() {
        register(new AppBinder());
        register(Controller.class);
    }
}

最后,这些都是我所需要的

public class AppBinder extends AbstractBinder {
    @Override
    protected void configure() {
        bind(new ReflectionService()).to(ReflectionService.class);
        bind(new Controller()).to(Controller.class);
    }
}

@ApplicationPath("/")
public class AppConfig extends ResourceConfig  {    
    public AppConfig() {
        register(new AppBinder());
        register(Controller.class);
    }
}

解决方法:

Service,Controller constructors is being called whenever we send an http request; does it create instances in each request or just calling constructor?

不幸的是,@ Singleton在使用AbstractBinder绑定时没有任何作用.我们需要明确地说应该是单身

bind(ReflectionService.class).to(ReflectionService.class).in(Singleton.class);

默认行为是“每次查询”范围,这意味着每次请求服务时都会创建一个新实例

(at the very first request – Why not during server startup!!)

这就是它的工作方式.还有一个InstantScope,它将使它在启动时创建

 bind(ReflectionService.class).to(ReflectionService.class).in(ImmediateScope.class)

或者,您可以使用实例代替类,它将自动成为单例

bind(new ReflectionService()).to(ReflectionService.class)

Service,Controller constructors is being called whenever we send an http request; does it create instances in each request or just calling constructor?

这是默认行为.每个请求的资源类的新实例.如前所述,如果只需要一次实例,则将其标记为@Singleton

Why System.out in AppConfig is called twice?

不确定,可能只是引导程序上的Jersey内部处理所需的

Is there a better way for setting up my small project, which does not have any db access and only three post endpoints?

设置很好,但是如果使用Jersey,则应使用ResourceConfig(应用程序扩展)类而不是Application

@ApplicationPath("/")
public class AppConfig extends ResourceConfig {
    public AppConfig() {
        register(new AppBinder());
        register(Controller.class);
    }
}

这样,您无需将AppBinder包装在AppFeature中.泽西岛已经知道如何处理AppBinder

标签:jax-rs,jersey,dependency-injection,java,java-ee
来源: https://codeday.me/bug/20191112/2024315.html