如何在@FacesConverter中注入@EJB,@PersistenceContext,@Inject,@Autowired等? [英] How to inject @EJB, @PersistenceContext, @Inject, @Autowired, etc in @FacesConverter?

查看:127
本文介绍了如何在@FacesConverter中注入@EJB,@PersistenceContext,@Inject,@Autowired等?的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

如何注入一个依赖关系,如 @EJB @PersistenceContext @Inject @AutoWired 等在 @FacesConverter 中?在我的具体情况下,我需要通过 @EJB 注入EJB:

How can I inject a dependency like @EJB, @PersistenceContext, @Inject, @AutoWired, etc in a @FacesConverter? In my specific case I need to inject an EJB via @EJB:

@FacesConverter
public class MyConverter implements Converter {

  @EJB
  protected MyService myService;    

  @Override
  public Object getAsObject(FacesContext context, UIComponent component, String value) {
    // myService.doSomething
  }

}

但是,它没有被注入,它仍然是 null ,导致NPE。似乎 @PersistenceContext @Inject 也无效。

However, it didn't get injected and it remains null, resulting in NPEs. It seems that @PersistenceContext and @Inject also doesn't work.

如何在转换器中注入服务依赖关系,以便我可以访问数据库?

How do I inject a service dependency in my converter so that I can access the DB?

推荐答案


我可以使用 @EJB 将我的服务注入到一个 @FacesConverter

Can I use @EJB to inject my service into a @FacesConverter?

不,直到JSF 2.3发布。 JSF / CDI的人正在为JSF 2.3工作。另请参阅 JSF规范问题1349 以及相关的JSF 2.3的新功能。只有这样,依赖注入就像 @EJB @PersistenceContext @Inject 等将在 @FacesConverter 中工作,当您将 managed = true 属性添加到注释时。 p>

No, not until JSF 2.3 is released. The JSF/CDI guys are working on that for JSF 2.3. See also JSF spec issue 1349 and this related "What's new in JSF 2.3?" article of my fellow Arjan Tijms. Only then dependency injection like @EJB, @PersistenceContext, @Inject, etc will work in a @FacesConverter when you explicitly add managed=true attribute to the annotation.

@FacesConverter(value="yourConverter", managed=true)
public class YourConverter implements Converter {

    @Inject
    private YourService service;
    // ...
}








如果没有,那么正确的方法是什么?

在JSF 2.3之前,您有几个选项:

Before JSF 2.3, you have several options:


  1. 将其设为托管bean。您可以通过 @ManagedBean @Named @Component 。以下示例使其成为JSF托管的bean。

  1. Make it a managed bean instead. You can make it a JSF, CDI or Spring managed bean via @ManagedBean, @Named or @Component. The below example makes it a JSF managed bean.

@ManagedBean
@RequestScoped
public class YourConverter implements Converter {

    @EJB
    private YourService service;
    // ...
}

下面的例子使它成为CDI管理的bean。

And the below example makes it a CDI managed bean.

@Named
@RequestScoped
public class YourConverter implements Converter {

    @Inject
    private YourService service;
    // ...
}

将其引用为< h:inputXxx converter =#{yourConverter}> 而不是< h:inputXxx converter =yourConverter> ,或作为< f:转换器绑定=#{yourConverter}> 而不是< f:converter converterId =yourConverter > 。不要忘记删除 @FacesConverter 注释!

Reference it as <h:inputXxx converter="#{yourConverter}"> instead of <h:inputXxx converter="yourConverter">, or as <f:converter binding="#{yourConverter}"> instead of <f:converter converterId="yourConverter">. Don't forget to remove the @FacesConverter annotation!

缺点是您不能指定 forClass ,因此需要在必要时在视图中手动定义转换器。

The disadvantage is that you cannot specify forClass and thus need to manually define the converter everywhere in the view where necessary.

将其注入常规托管bean。

Inject it in a regular managed bean instead.

@ManagedBean
@RequestScoped
public class YourBean {

    @EJB
    private YourService service;
    // ...
}

在转换器中,通过EL调用它。

And in your converter, grab or call it via EL.

YourBean yourBean = context.getApplication().evaluateExpressionGet(context, "#{yourBean}", YourBean.class);

// Then e.g. either
YourEntity yourEntity = yourBean.getService().findByStringId(value);
// Or
YourEntity yourEntity = yourBean.findEntityByStringId(value);

这样你可以继续使用 @FacesConverter

This way you can keep using @FacesConverter.

从JNDI手动抓取EJB。

Manually grab the EJB from JNDI.

YourService yourService = (YourService) new InitialContext().lookup("java:global/appName/YourService");

缺点是有一定的风险,这不是完全可移植的。另请参阅以编程方式从JSF托管的bean中注入EJB bean

The disadvantage is that there is a certain risk that this is not entirely portable. See also Inject EJB bean from JSF managed bean programmatically.

安装 OmniFaces 。自1.6版以来,它在中透明地添加了对 @EJB (和 @Inject )的支持。 @FacesConverter ,无需进一步修改。另见展示。如果您恰好需要< f:selectItem(s)> 的转换器,则替代方案是使用其 SelectItemsConverter 它将根据选择的项目自动执行转换作业,而不需要任何数据库交互。

Install OmniFaces. Since version 1.6, it transparently adds support for @EJB (and @Inject) in a @FacesConverter without any further modification. See also the showcase. If you happen to need the converter for <f:selectItem(s)>, then the alternative is to use its SelectItemsConverter which will automatically perform the conversion job based on select items without the need for any database interaction.

<h:selectOneMenu ... converter="omnifaces.SelectItemsConverter">

另请参见转换器的转换错误设置值



另见:



查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆