素数 数据表/排序/错误项目



我有一个简单的数据豆,如下所示:

@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 如何结束对话。

相关内容

  • 没有找到相关文章