2

我正在尝试为 primefaces 数据列表实现类似于数据表的延迟加载模型,如下所示

我具有正常 AJAX 分页功能的初始代码工作得非常好。但是,当我尝试使用延迟加载模型时,页面加载时出现以下异常:

com.sun.faces.application.view.FaceletViewHandlingStrategy handleRenderException
SEVERE: Error Rendering View[/pages/index.xhtml]
java.io.NotSerializableException: java.util.ArrayList$SubList
    at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1180)
    at java.io.ObjectOutputStream.defaultWriteFields(ObjectOutputStream.java:1528)
    at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1493)
    at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1416)
    at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1174)
    at java.io.ObjectOutputStream.defaultWriteFields(ObjectOutputStream.java:1528)
    at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1493)
    at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1416)
    at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1174)
    at java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:346)
    at java.util.HashMap.writeObject(HashMap.java:1100)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:601)
    at java.io.ObjectStreamClass.invokeWriteObject(ObjectStreamClass.java:975)
    at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1480)
    at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1416)
    at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1174)
    at java.io.ObjectOutputStream.defaultWriteFields(ObjectOutputStream.java:1528)
    at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1493)
    at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1416)
    at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1174)
    at java.io.ObjectOutputStream.writeArray(ObjectOutputStream.java:1362)
    at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1170)
    at java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:346)
    at java.util.HashMap.writeObject(HashMap.java:1100)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:601)
    at java.io.ObjectStreamClass.invokeWriteObject(ObjectStreamClass.java:975)
    at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1480)
    at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1416)
    at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1174)
    at java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:346)
    at com.sun.faces.renderkit.ClientSideStateHelper.doWriteState(ClientSideStateHelper.java:325)
    at com.sun.faces.renderkit.ClientSideStateHelper.writeState(ClientSideStateHelper.java:173)
    at com.sun.faces.renderkit.ResponseStateManagerImpl.writeState(ResponseStateManagerImpl.java:122)
    at com.sun.faces.application.StateManagerImpl.writeState(StateManagerImpl.java:166)
    at com.sun.faces.application.view.WriteBehindStateWriter.flushToWriter(WriteBehindStateWriter.java:225)
    at com.sun.faces.application.view.FaceletViewHandlingStrategy.renderView(FaceletViewHandlingStrategy.java:418)
    at com.sun.faces.application.view.MultiViewHandler.renderView(MultiViewHandler.java:131)
    at com.sun.faces.lifecycle.RenderResponsePhase.execute(RenderResponsePhase.java:121)
    at com.sun.faces.lifecycle.Phase.doPhase(Phase.java:101)
    at com.sun.faces.lifecycle.LifecycleImpl.render(LifecycleImpl.java:139)
    at javax.faces.webapp.FacesServlet.service(FacesServlet.java:594)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:305)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:210)
    at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:222)
    at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:123)
    at org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:472)
    at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:168)
    at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:99)
    at org.apache.catalina.valves.AccessLogValve.invoke(AccessLogValve.java:929)
    at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:118)
    at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:407)
    at org.apache.coyote.http11.AbstractHttp11Processor.process(AbstractHttp11Processor.java:1002)
    at org.apache.coyote.AbstractProtocol$AbstractConnectionHandler.process(AbstractProtocol.java:585)
    at org.apache.tomcat.util.net.JIoEndpoint$SocketProcessor.run(JIoEndpoint.java:310)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1110)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:603)
    at java.lang.Thread.run(Thread.java:722)

这是index.html的代码

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
    "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">

<html xmlns="http://www.w3.org/1999/xhtml"
   xmlns:ui="http://java.sun.com/jsf/facelets"
   xmlns:h="http://java.sun.com/jsf/html"
   xmlns:f="http://java.sun.com/jsf/core"
   xmlns:p="http://primefaces.org/ui">

<ui:composition template="/pages/templates/template.xhtml">
   <ui:define name="content">

      <h:form prependId="false" id="form">
         <p:dataList value="#{movies.lazyMovieModel}" var="movie" id="movies" paginator="true" rows="10"
            paginatorTemplate="{FirstPageLink} {PreviousPageLink} {PageLinks} {NextPageLink} {LastPageLink}"
            type="none" paginatorAlwaysVisible="false" lazy="true">

            <h:outputText value="#{movie.movieName}, #{movie.releaseYear}" style="margin-left:10px">
            </h:outputText>
            <br/>
         </p:dataList>
      </h:form>

   </ui:define>
</ui:composition>
</html>

MovieListBean.java

import org.primefaces.model.LazyDataModel;

import com.clixflix.enitities.Movie;
import com.clixflix.jsf.extensions.LazyMovieDataModel;

@ManagedBean(name = "movies")
@ViewScoped
public class MovieListBean extends BaseBean implements Serializable
{
   private static final long serialVersionUID = -5719443344065177588L;

   private LazyDataModel<Movie> lazyMovieModel;

   @PostConstruct
   public void initialize() {
      lazyMovieModel = new LazyMovieDataModel();
   }

   public LazyDataModel<Movie> getLazyMovieModel() 
   {
      List<Movie> movieList = getServiceLocator().getMovieService().getMovieList();
      ((LazyMovieDataModel) lazyMovieModel).setMovieList(movieList);
      return lazyMovieModel;
   }
}

LazyMovieDataModel.java(LazyModel 实现)

public class LazyMovieDataModel extends LazyDataModel<Movie>
{
   private static final long serialVersionUID = 8745562148994455749L;

   private List<Movie> movieList;

   public LazyMovieDataModel() {
      this.movieList = Collections.emptyList();
   }

   @Override
   public List<Movie> load(int first, int pageSize, String sortField, SortOrder sortOrder, Map<String, String> filters) {
      // Sorting
      if (null != sortField) {
         LazySorter sorter = new LazySorter(sortField, sortOrder);
         Collections.sort(movieList, sorter);
         sorter = null;
      }

      // RowCount
      int rowCount = movieList.size();
      this.setRowCount(rowCount);

      // Pagination
      if (rowCount > pageSize) {
         return movieList.subList(first, (first + pageSize));
      }
      else {
         return movieList;
      }
   }


   private class LazySorter implements Comparator<Movie>
   {
      private String sortField;

      private SortOrder sortOrder;

      LazySorter(String sortField, SortOrder sortOrder) {
         this.sortField = sortField;
         this.sortOrder = sortOrder;
      }

      @SuppressWarnings("unchecked")
      @Override
      public int compare(Movie movie1, Movie movie2) {
         Object value1 = null, value2 = null;

         try {
            value1 = Movie.class.getField(this.sortField).get(movie1);
            value2 = Movie.class.getField(this.sortField).get(movie2);

            int value = ((Comparable<Object>) value1).compareTo(value2);
            return SortOrder.ASCENDING.equals(sortOrder) ? value : -1 * value;
         }
         catch (IllegalArgumentException | IllegalAccessException | NoSuchFieldException | SecurityException e) {
            e.printStackTrace();
            return 0;
         }
      }
   }

   public void setMovieList(List<Movie> movieList) {
      this.movieList = movieList;
   }
}

我假设例外是在这一行:

return movieList.subList(first, (first + pageSize));

谁能指导我我错过了什么?

另外,我在日志中观察到,当我使用惰性模型时,数据库被查询了 3 次,但是当我使用正常的 AJAX 分页时,数据库只被查询一次:|


更新:我弄清楚了数据库被查询 3 次的原因。这是因为我在 LazyModel 的 getter 中调用我的服务,而不仅仅是在 load 方法中。

我在类中进行了以下更改:

LazyMovieDataModel.java

public class LazyMovieDataModel extends LazyDataModel<Movie>
{
   private static final long serialVersionUID = 8745562148994455749L;

   public LazyMovieDataModel() {}

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

      List<Movie> movieList = getServiceLocator().getMovieService().getMovieList(first, (first + pageSize));

      // RowCount
      int rowCount = ((Number)getServiceLocator().getMovieService().getMovieCount()).intValue();
      this.setRowCount(rowCount);
   }
}

MovieListBean.java 中的 LazyModel getter

/* Removed PostConstruct init method */

public LazyDataModel<Movie> getLazyMovieModel() 
{
   return lazyMovieModel;
}

上述更改在初始页面加载时工作正常。但是,当我点击下一页按钮(或任何分页按钮)时,我getServiceLocator()在加载方法中得到了一个 NPE。

serviceLocator 是protected从 BaseBean 继承并使用 Spring 注入的访问修改托管属性。

getter 在后续调用中返回 null 的任何原因???

4

2 回答 2

0

ArrayList$SubList 是个问题。返回的 subList 没有实现可序列化。
尝试使用:

return new ArrayList(movieList.subList(first, (first + pageSize)));
于 2013-09-30T11:01:41.413 回答
0

Same problem in this link.There are 2 load methods in LazyDataModel:

public List<T> load(int first, int pageSize, String sortField, SortOrder sortOrder, Map<String,String> filters) {
    throw new UnsupportedOperationException("Lazy loading is not implemented.");
}

public List<T> load(int first, int pageSize, List<SortMeta> multiSortMeta,Map<String,String> filters) {
    throw new UnsupportedOperationException("Lazy loading is not implemented.");
}

This is where the error is thrown. You are using multisort, so you should override the second.Default method is the first one.

于 2020-02-16T11:00:44.700 回答