延迟加载的primefaces数据表在SQL语句完成执行之前得到更新

时间:2012-02-08 17:16:01

标签: sql jsf jsf-2 datatable primefaces

当在更新\从此表中删除实体的SQL语句完成执行之前更新primefaces数据表时,我遇到了问题。

01:08:13,255 INFO [reg.data.model.LazyEventDataModelMod]  - Loading the lazy event data between 0 and 10 
01:08:13,256 DEBUG [org.hibernate.SQL]  - 
    select
        count(*) as col_0_0_ 
    from
        events event0_ limit ? 
01:08:13,394 DEBUG [org.hibernate.SQL]  - 
    delete 
    from
        events 
    where
        id=? 

问题是实体管理器没有返回任何内容来表明它已成功执行了delete \ update查询。

我使用JSF 2,Weld 1.1.5.Final,Primefaces 3.0.1,Seam持久性/事务/焊接3.0.1.Final。
服务器是Tomcat 7。

以下是调用控制器bean上的操作的按钮的代码:

<p:commandButton id="deleteButton" value="#{msg.delete}"
    icon="ui-icon-trash" update=":events_form :messages"
    actionListener="#{eventController.delete()}" />

<p:commandButton id="refreshButton" value="#{msg.refresh}"
    icon="ui-icon-refresh" update=":events_form :messages" />

按下第一个按钮后不会发生更新,但是第二个按钮会更新表,两个都有相同的ajax更新部分。

delete()方法的代码:

public void delete() {
            if (getSelEvent() != null) {
                log.debug("deleting event " + getSelEvent().getId());

                dao.delete(getSelEvent());

    // Tried putting the code for reloading te data model here, but it is also executed before the sql statements...

            } else {
                log.warn("delete action cannot proceed, no event selected!");
            }

        }
dao中的

deleteById()方法:

public void delete(T entity) {
        if (em == null)
            init();
        log.debugf("removing %s instance", simpleName);

        entity = em.merge(entity);

// The problem lies here - em.remove() does not return any value to indicate success/failure, so the control is released before the SQL statements finish execution 

em.remove(entity);
}

数据表:

<p:dataTable id="event_list" value="#{eventController.lazyModel}"
                var="event" rowKey="#{event.id}" paginator="true" rows="20"
                selection="#{eventController.selEvent}" selectionMode="single"
                onRowSelectUpdate="content,messages"
                onRowEditUpdate="content,messages"
                paginatorTemplate="{CurrentPageReport} {FirstPageLink} {PreviousPageLink} {PageLinks} {NextPageLink} {LastPageLink} {RowsPerPageDropdown}"
                rowsPerPageTemplate="10,25,50,100">

LazyDataModel类:

public class LazyEventDataModel extends LazyDataModel<Event> {

private static final long serialVersionUID = 2386224274507659474L;

private EventDAO dao;

private Logger log = Logger.getLogger(LazyEventDataModel.class);

private List<Event> data;

public LazyEventDataModel() {
    dao = new EventDAO();
}

@Override
public Event getRowData(String rowKey) {
    return dao.find(Long.parseLong(rowKey));
}

@Override
public Object getRowKey(Event event) {
    return event.getId();
}

@Override
public List<Event> load(int first, int pageSize, String sortField,
        SortOrder sortOrder, Map<String, String> filters) {

    if (sortField == null || "".equals(sortField)) {
        sortField = "event_date";
        sortOrder = SortOrder.DESCENDING;
    }

    if (sortField.equals("date"))
        sortField = "event_date";

    // Get the paginated and sorted data
    data = dao.findPageObjects(Event.class, first, pageSize, sortField,
            sortOrder, filters);

    log.infof("Loading the lazy event data between " + first + " and "
            + (first + pageSize));

    // Set the row count
    this.setRowCount(Global.safeLongToInt(dao.count()));

    return data;
}
}

我现在能想到的唯一解决方案是在调用dao.delete(getSelEvent())并在该计时器之后刷新数据模型后添加计时器。

1 个答案:

答案 0 :(得分:0)

commandButton的update属性也必须包含数据表。无论ajax调用完成,都会发生更新中设置的内容。考虑到您没有创建单独的线程来执行数据库操作(这将使ajax请求几乎立即完成),请执行以下操作。

但有一点需要注意:出于某种原因,引用update属性中数据表的id不起作用。我通常只使用panelGroup包装它并告诉按钮更新panelGroup。像这样:

<p:commandButton value="Delete" actionListener="#{someBean.delete}" update="panel">
</p:commandButton>
<h:panelGroup id="panel">
<p:dataTable id="event_list" value="#{eventController.lazyModel}"
                var="event" rowKey="#{event.id}" paginator="true" rows="20"
                selection="#{eventController.selEvent}" selectionMode="single"
                onRowSelectUpdate="content,messages"
                onRowEditUpdate="content,messages"
                paginatorTemplate="{CurrentPageReport} {FirstPageLink} {PreviousPageLink} {PageLinks} {NextPageLink} {LastPageLink} {RowsPerPageDropdown}"
                rowsPerPageTemplate="10,25,50,100">

...

</h:panelGroup>

请注意,primeFaces按钮组件默认使用ajax,因此您根本不需要jsf2 ajax标记。