【问题标题】:Removing second item from a list crashes REST API从列表中删除第二项会使 REST API 崩溃
【发布时间】:2020-04-21 21:11:29
【问题描述】:

我正在尝试向我的 REST API 添加一项功能,它按年份过滤数据,因此只有对象会显示与我请求的年份相同的对象。

我用来测试这个特性的路径是

  http://localhost:8080/Assignment2C/beers?year=2019

出现 500 错误

在 Beers_Service.java 中,它第二次遍历列表的地方是失败的地方

public List<Beers> getAllBeersByYear(int year) {
EntityManager em = DBUtil.getEMF().createEntityManager();

List<Beers> list = null;

try {
    list = em.createNamedQuery("Beers.findAll", Beers.class)
            .getResultList();
    if (list == null || list.isEmpty()) {
        list = null;
    }

} finally {
    em.close();
}
Calendar cal = Calendar.getInstance();
for(Beers beer: list){
    cal.setTime(beer.getLastMod());
    if(cal.get(Calendar.YEAR)!=year){

        list.remove(beer);
    }
}
return list;

数据库中的所有日期都是 2019 年,但第一个对象是 2018 年

当使用调试器时,它会遍历 for 循环,因为数据库中的第一个条目是 2018,所以它会很好地删除它,删除对象后,它会回到 for 循环的顶部并在这一行崩溃.

 for(Beers beer: list){

当它在调试模式下中断for循环后,它进入到这个源代码中

                   beforeExecute(wt, task);
                    try {
                        task.run();
                        afterExecute(task, null);
                    } catch (Throwable ex) {
                        afterExecute(task, ex);
                        throw ex;
                    }

排队

afterExecute(task, null);

我得到的堆栈跟踪是

<h1>HTTP Status 500 – Internal Server Error</h1>
    <hr class="line" />
    <p><b>Type</b> Exception Report</p>
    <p><b>Message</b> Request processing failed; nested exception is java.util.ConcurrentModificationException</p>
    <p><b>Description</b> The server encountered an unexpected condition that prevented it from fulfilling the request.
    </p>
    <p><b>Exception</b></p>
    <pre>org.springframework.web.util.NestedServletException: Request processing failed; nested exception is java.util.ConcurrentModificationException
    org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:1014)
    org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:898)
    javax.servlet.http.HttpServlet.service(HttpServlet.java:634)
    org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:883)
    javax.servlet.http.HttpServlet.service(HttpServlet.java:741)
    org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:53)
</pre>
    <p><b>Root Cause</b></p>
    <pre>java.util.ConcurrentModificationException
    java.base&#47;java.util.Vector$Itr.checkForComodification(Vector.java:1320)
    java.base&#47;java.util.Vector$Itr.next(Vector.java:1276)
    main.Beers_Service.getAllBeersByYear(Beers_Service.java:53)
    main.Beers_Controller.GetAllBreweries(Beers_Controller.java:72)
    java.base&#47;jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    java.base&#47;jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    java.base&#47;jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    java.base&#47;java.lang.reflect.Method.invoke(Method.java:567)
    org.springframework.web.method.support.InvocableHandlerMethod.doInvoke(InvocableHandlerMethod.java:190)
    org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:138)
    org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:105)
    org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:879)
    org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:793)
    org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:87)
    org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:1040)
    org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:943)
    org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:1006)
    org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:898)
    javax.servlet.http.HttpServlet.service(HttpServlet.java:634)
    org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:883)
    javax.servlet.http.HttpServlet.service(HttpServlet.java:741)
    org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:53)

【问题讨论】:

    标签: java list rest for-loop http-status-code-500


    【解决方案1】:

    您不应该在迭代集合时修改它: https://docs.oracle.com/javase/7/docs/api/java/util/ConcurrentModificationException.html

    " 例如,如果一个线程在使用 fail-fast 迭代器对集合进行迭代时直接修改了一个集合,则迭代器将抛出此异常。"

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2015-04-12
      • 1970-01-01
      • 2022-06-17
      • 1970-01-01
      • 2020-12-30
      • 1970-01-01
      • 2017-09-16
      • 1970-01-01
      相关资源
      最近更新 更多