我想初始化一个集合,并在我的应用程序启动时用数据填充它。然后我想在我的应用程序中随处访问它。我希望例如我的REST API可以访问已填充数据的共享集合。
我首先尝试使用@Startup和@Singleton注释的启动类来完成它。当我在那里注入我的userService时,我遇到了一些问题,并且由于这篇文章中的建议:@Inject and @PostConstruct not working in singleton pattern我尝试使用@ApplicationScoped注释:
@Named
@ApplicationScoped
public class KwetterApp {
@Inject
private UserService service;
@PostConstruct
public void init() {
try {
User harry = new User("Harry", "[email protected]", "New York", "http://harry.com", "Hi, I'm Harry!", UserType.REGULAR);
User nick = new User("Nick", "[email protected]", "California", "http://nick.com", "Hi, I'm Nick!", UserType.REGULAR);
User jane = new User("Jane", "[email protected]", "Texas", "http://jane.com", "Hi, I'm Jane!", UserType.REGULAR);
Tweet tweet = new Tweet("eating...", harry);
Tweet tweet1 = new Tweet("swimming...", harry);
Tweet tweet2 = new Tweet("jogging...", jane);
harry.addTweet(tweet);
harry.addTweet(tweet1);
jane.addTweet(tweet2);
service.create(harry);
service.create(nick);
service.create(jane);
}
catch (Exception e){
e.printStackTrace();
}
}
public UserService getService() {
return service;
}
}
我在休息服务中注入这个课:
@RequestScoped
@Path("/user")
public class UserRest {
// @Inject
// private UserService userService;
@Inject
private KwetterApp kwetterApp;
// private KwetterApp kwetterApp = KwetterApp.getInstance();
private UserService userService = kwetterApp.getService();
@GET
@Produces({"application/json"})
public List<User> get() throws UserException {
return userService.getAll();
}
}
注入此KwetterApp时,会导致以下异常:
StandardWrapperValve[rest.RestApplication]: Servlet.service() for servlet rest.RestApplication threw exception
java.lang.NullPointerException
at rest.UserRest.<init>(UserRest.java:27)
at rest.UserRest$Proxy$_$$_WeldClientProxy.<init>(Unknown Source)
我有一个空的beans.xml文件,其中bean-discovery-mode设置为'all'。 CDI框架应该识别我的KwetterApp类注射,对吧?为什么它为空?
提前致谢,
麦克风
这里
@Inject
private KwetterApp kwetterApp;
private UserService userService = kwetterApp.getService();
我不认为kwetterApp
场将在userService
之前设置。
CDI
将在构造对象后设置该字段。
无论如何应该使用的替代方案是构造函数注入
@RequestScoped
@Path("/user")
public class UserRest {
private KwetterApp kwetterApp;
private UserService userService;
protected UserRest() {}
@Inject
public UserRest(final KwetterApp kwetterApp) {
this.kwetterApp = kwetterApp;
this.userService = kwetterApp.getService();
}
@GET
@Produces({"application/json"})
@Override
public List<User> get() throws UserException {
return userService.getAll();
}
}
需要使用protected
构造函数,因为@RequestScoped
是一个正常范围的bean,它必须是可代理的,如规范中所述。
唯一不需要空构造函数的注释是@Singleton
(来自javax.inject
)。
如果要使用注入的bean初始化对象,则必须使用@PostConstruct注释方法,因为注入的CDI bean仅在@PostContruct注释方法中的CDI中可用,之后而不是在字段初始化或构造函数调用期间。
因此,UserService已经是CDI bean,您可以将它注入到其余的服务bean中,因为它将与当前活动范围内使用的bean相同。 KwetterApp在当前活动范围内可用,因此UserService也是如此。只有@Dependend作用域bean的行为不同,因此每个bean都会获得自己的实例。