我有一个(请求范围的)列表,用户可以从中选择“PQ”(链接列表)。单击或以其他方式输入浏览器时,应显示每个PQ的主页面。每个PQ的页面都是
的形式 http://localhost:8080/projectname/main.jsf?id=2
首先是PQ bean:
@Named
@ViewScoped
public class PqHome implements Serializable
{
@PersistenceContext(unitName="...")
private EntityManager em;
private Integer id;
private PQ instance;
@PostConstruct
public void init()
{
System.out.println("ID is " + id); // ID from URL param
instance = em.find(PQ.class, id);
}
public Integer getId()
{
return id;
}
public void setId(Integer id)
{
this.id = id;
}
public PQ getInstance()
{
return instance;
}
}
这是main.xhtml:
<ui:composition xmlns="http://www.w3.org/1999/xhtml"
...>
<ui:define name="metadata">
<f:metadata>
<f:viewParam name="id" value="#{pqHome.id}">
<f:convertNumber integerOnly="#{true}" />
</f:viewParam>
<!--f:event type="preRenderView" listener="#{pqHome.init}" /-->
</f:metadata>
</ui:define>
<ui:define name="title">
<h:outputText value="Main" />
</ui:define>
...
</ui:composition>
每当我选择或以其他方式刷新页面/网址时,我都会从NullPointerException
获得EntityManager
:
org.jboss.weld.exceptions.WeldException: WELD-000049 Unable to invoke [method] @PostConstruct public de.mycomp.myproj.beans.PqHome.init() on de.mycomp.myproj.beans.PqHome@4f0ea68f
at org.jboss.weld.bean.AbstractClassBean.defaultPostConstruct(AbstractClassBean.java:595)
...
Caused by: java.lang.IllegalArgumentException: id to load is required for loading
at org.hibernate.event.spi.LoadEvent.<init>(LoadEvent.java:87)
at org.hibernate.event.spi.LoadEvent.<init>(LoadEvent.java:59)
at org.hibernate.internal.SessionImpl.get(SessionImpl.java:961)
at org.hibernate.internal.SessionImpl.get(SessionImpl.java:957)
at org.hibernate.ejb.AbstractEntityManagerImpl.find(AbstractEntityManagerImpl.java:787)
at org.hibernate.ejb.AbstractEntityManagerImpl.find(AbstractEntityManagerImpl.java:762)
at org.jboss.as.jpa.container.AbstractEntityManager.find(AbstractEntityManager.java:221)
at de.mycomp.myproj.beans.PqHome.init(PqHome.java:47)
... 56 more
[第47行是em.find(...)]
该行
<f:event type="preRenderView" listener="#{pqHome.init}" />
不会让事情变得更好。我现在很沮丧。
如何将URL GET请求参数传入@ViewScoped
bean?
注意:我敢打赌这不是一件小事。有可能我在概念上做错了,所以欢迎任何有关如何改进的提示。我觉得我需要选择@ViewScoped
,因为在该页面上会有更复杂的基于AJAX的GUI,我真的希望通过URL GET参数来访问它。
由于
答案 0 :(得分:6)
在bean构造和所有依赖注入(例如@PostConstruct
,@PersistenceContext
,@EJB
,{{1}之后,@ManagedProperty
直接调用 等等。等等。
@Inject
在更新模型值阶段设置其值,这是在构建bean之后很久(后)。因此,<f:viewParam>
内的@PostConstruct
值尚未设置。那时它仍然是<f:viewParam>
。
您与null
关闭,但您必须删除 <f:event type="preRenderView">
注释。
所以:
@PostConstruct
与
<f:viewParam name="pq" value="#{pqHome.id}">
<f:convertNumber integerOnly="#{true}" />
</f:viewParam>
<f:event type="preRenderView" listener="#{pqHome.init}" />
无关,我建议改为使用private Integer id;
public void init() {
instance = em.find(PQ.class, id);
}
。另请参阅Communication in JSF 2.0 - Converting and validating GET request parameters。
组合Converter
也不会按预期工作。特定于JSF的@Named @ViewScoped
仅与JSF特定的@ViewScoped
结合使用。您的CDI特定@ManagedBean
的行为方式与@Named
相同。使用@RequestScoped
代替@ManagedBean
或使用CDI特定的@Named
代替@ConversationScoped
。
答案 1 :(得分:6)
有一种更好的方式从网址获取ID。只需在@PostConstruct init()方法中使用它从url获取“id”:
FacesContext.getCurrentInstance().getExternalContext().getRequestParameterMap().get("id");
您仍然可以使用ViewScoped和@PostConstruct。