我有一个简单的数据豆,如下所示:
@Model
Class DataBean{
private List<Elements> elements;
@PostConstruct
private void loadElements(){
//fetch data from database.
}
}
im 使用 Primefaces 数据表来显示数据,如下所示:
<h:form>
<p:dataTable
value="#{dataBean.elements}"
var="element" >
<p:column sortBy="#{element.id}"
sortFunction="#{sortingHelper.sortNumericCallback}">
<f:facet name="header">ID</f:facet>
<p:commandLink action="#{pageController.navigateToDetailView(element)}"
value="#{element.id}">
</p:commandLink>
</p:column>
</p:datatable>
</h:form>
pageController.navigateToDetailView(element)
简单设置下一页的 dataBean 上的选定元素,因此 detailView 准备好了其数据,然后返回 detail-Navigation-Result。
现在:问题:如果我单击其中一个命令链接而不进行任何排序,一切都很好。如果我按 id 排序并单击详细信息链接,则会发生以下情况:
- 请求已启动
- 数据Bean加载(后构造)(排序消失)
现在 - 在第二个请求中 - 页面再次重建(为了触发导航到详细视图操作)并且数据表"知道",我单击了第 5 行。但是如果不再次排序,第 5 行现在是一个不同的条目,因为 bean 被重建了。
不同点的控制台输出。
首先,我单击显示数据表的页面。"."'s 是我的自定义排序函数的一个比较,只是为了指示集合已排序。
13:47:56,046 INFO [stdout] (http--0.0.0.0-8090-1) -- Started Request --
13:47:56,047 INFO [stdout] (http--0.0.0.0-8090-1) ---- Started RESTORE_VIEW 1 ----
13:47:56,048 INFO [stdout] (http--0.0.0.0-8090-1) ---- Started RENDER_RESPONSE 6 ----
13:47:56,087 INFO [stdout] (http--0.0.0.0-8090-1) PostConstruct DataBean
13:47:56,566 INFO [stdout] (http--0.0.0.0-8090-1) -- Finished Request --
没关系。现在通过单击 id 标头进行排序
13:48:15,008 INFO [stdout] (http--0.0.0.0-8090-2) -- Started Request --
13:48:15,009 INFO [stdout] (http--0.0.0.0-8090-2) ---- Started RESTORE_VIEW 1 ----
13:48:15,051 INFO [stdout] (http--0.0.0.0-8090-2) ---- Started APPLY_REQUEST_VALUES 2 ----
13:48:15,052 INFO [stdout] (http--0.0.0.0-8090-2) PostConstruct DataBean
13:48:15,124 INFO [stdout] (http--0.0.0.0-8090-2) ..............................................................
13:48:15,124 INFO [stdout] (http--0.0.0.0-8090-2) ---- Started PROCESS_VALIDATIONS 3 ----
13:48:15,126 INFO [stdout] (http--0.0.0.0-8090-2) ---- Started UPDATE_MODEL_VALUES 4 ----
13:48:15,127 INFO [stdout] (http--0.0.0.0-8090-2) ---- Started INVOKE_APPLICATION 5 ----
13:48:15,127 INFO [stdout] (http--0.0.0.0-8090-2) ---- Started RENDER_RESPONSE 6 ----
13:48:15,387 INFO [stdout] (http--0.0.0.0-8090-2) -- Finished Request --
那也没关系。现在,表已按原样排序。现在我点击在第 10 行选择ID 为 53 的项目;
13:48:28,295 INFO [stdout] (http--0.0.0.0-8090-4) -- Started Request --
13:48:28,296 INFO [stdout] (http--0.0.0.0-8090-4) ---- Started RESTORE_VIEW 1 ----
13:48:28,361 INFO [stdout] (http--0.0.0.0-8090-4) ---- Started APPLY_REQUEST_VALUES 2 ----
13:48:28,363 INFO [stdout] (http--0.0.0.0-8090-4) PostConstruct DataBean
13:48:28,487 INFO [stdout] (http--0.0.0.0-8090-4) ---- Started PROCESS_VALIDATIONS 3 ----
13:48:28,501 INFO [stdout] (http--0.0.0.0-8090-4) ---- Started UPDATE_MODEL_VALUES 4 ----
13:48:28,514 INFO [stdout] (http--0.0.0.0-8090-4) ---- Started INVOKE_APPLICATION 5 ----
13:48:28,514 INFO [stdout] (http--0.0.0.0-8090-4) navigateToDetail() called
13:48:28,516 INFO [stdout] (http--0.0.0.0-8090-4) Constructing ElementEditDataBean
13:48:28,517 INFO [stdout] (http--0.0.0.0-8090-4) Setting ActiveElement to 42
13:48:28,518 INFO [stdout] (http--0.0.0.0-8090-4) ---- Started RENDER_RESPONSE 6 ----
13:48:28,748 INFO [stdout] (http--0.0.0.0-8090-4) -- Finished Request --
请注意,AFTER PostConstruct DataBean
不进行排序。(我假设因为我使用表单在表中,数据表不知道排序可能已更改。
结果,传递 id 为 42 的元素。(元素 42 在未排序的情况下位于第 10 位)
结果,navigateToDetailView(元素)现在被另一个元素触发,而不是预期的...
问题出在 ofc。 排序的集合被后构造方法重置。我也知道,它可以通过对话范围解决。
但我想知道是否有一种无状态的方法可以做到这一点?(我不想为每个排序/page2page导航启动对话)
有什么想法吗?
编辑 1:SortingHelper 是一个自己的类,看起来像这样:
@Named
public class SortingHelper {
/**
* Sorts two integers correctly.
* @param o1 integer 1
* @param o2 integer 2
* @return negative value if o1 is less, 0 if equal, or positive value if greater
*/
public int sortNumericCallback(Object o1, Object o2) {
System.out.print(".");
int i1 = Integer.parseInt((String) o1);
int i2 = Integer.parseInt((String) o2);
return (i1 == i2) ? 0 : (i1 > i2) ? 1 : -1;
}
}
(Primefaces Datatable 在对整数进行排序时失败,或者假设它对数字进行排序字典:11 <5 等)
但即使我不介意排序和使用 NO 自定义排序功能,结果也是相同的。
截至评论,我现在修改了 Databean 以在加载后开始对话。
@ConversationScoped
Class DataBean{
private List<Elements> elements;
@Inject
private Conversation conversation;
@PostConstruct
private void loadElements(){
if (this.conversation.isTransient())
this.conversation.begin();
//fetch data from database.
}
}
在我的页面控制器的导航功能中,我再次停止该对话:
public String navigateToDetailView(Element element) {
//pass element to next databean.
conversation.end();
//...
return "detailView";
}
然而,这导致了一个问题,即使用浏览器导航回去会导致无效的对话(它们已结束)。
为了解决这个问题,我创建了一个自定义过滤器,该过滤器基本上禁用了浏览器的缓存,因此它们在history.back()
刷新页面,因此具有全新的有效对话 ID。
筛选器如下所示:
public class NoCacheFilter implements Filter {
@Override
public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {
HttpServletResponse hsr = (HttpServletResponse) res;
hsr.setHeader("Cache-Control", "no-cache, no-store, must-revalidate"); // HTTP 1.1.
hsr.setHeader("Pragma", "no-cache"); // HTTP 1.0.
hsr.setDateHeader("Expires", 0); // Proxies.
chain.doFilter(req, res);
}
@Override
public void destroy() {
// TODO Auto-generated method stub
}
@Override
public void init(FilterConfig filterConfig) throws ServletException {
// TODO Auto-generated method stub
}
}
在 Web 中使用此部分.xml:
<filter>
<filter-name>noCacheFilter</filter-name>
<filter-class>com.example.NoCacheFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>noCacheFilter</filter-name>
<url-pattern>*.xhtml</url-pattern>
</filter-mapping>
现在我只需要弄清楚,如果用户选择不从侧面导航,而不是调用 showDetail()-Action 如何结束对话。