0

我是一个有志于学习的业余编程,但我遇到了一种我什至不知道从哪里开始寻找的新型问题——java中的内存泄漏。我四处搜寻,并没有真正找到任何对我有帮助的东西。我使用了 Tomcat v9.0 和 Java 1.8。我什至不知道您需要查看哪些代码才能提供帮助。

当我尝试向我的 REST api 发送请求时收到此警告

VARNING: The web application [School] appears to have started a thread named [pool-2-thread-1] but has failed to stop it. This is very likely to create a memory leak. Stack trace of thread:
 sun.misc.Unsafe.park(Native Method)
 java.util.concurrent.locks.LockSupport.parkNanos(Unknown Source)
 java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.awaitNanos(Unknown Source)
 java.util.concurrent.ScheduledThreadPoolExecutor$DelayedWorkQueue.take(Unknown Source)
 java.util.concurrent.ScheduledThreadPoolExecutor$DelayedWorkQueue.take(Unknown Source)
 java.util.concurrent.ThreadPoolExecutor.getTask(Unknown Source)
 java.util.concurrent.ThreadPoolExecutor.runWorker(Unknown Source)
 java.util.concurrent.ThreadPoolExecutor$Worker.run(Unknown Source)
 java.lang.Thread.run(Unknown Source)

服务器可以处理一两个请求,然后它就停止了。由于我是这类问题的新手,我不知道是什么原因造成的,而且四处搜索并没有真正帮助我以我业余的方式。然而,我猜我正在以某种方式创建线程,但它们并没有被关闭。

我尝试使用 get 方法访问的控制器

package se.consys.controllers;

import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import javax.ws.rs.core.Response.Status;
import javax.persistence.NoResultException;
import javax.ws.rs.Consumes;
import javax.ws.rs.DELETE;
import javax.ws.rs.DefaultValue;
import javax.ws.rs.GET;
import javax.ws.rs.PATCH;
import javax.ws.rs.POST;
import javax.ws.rs.PUT;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
import javax.ws.rs.Produces;
import javax.ws.rs.QueryParam;
import se.consys.Entities.Course;
import se.consys.Entities.Lecture;
import se.consys.Entities.Student;
import se.consys.Entities.Teacher;
import se.consys.Utilities.HibernateUtility;
import se.consys.dataaccess.DaoGenericHibernateImpl;
import se.consys.params.LocalDateParam;
import se.consys.params.LocalDateTimeParam;
import se.consys.params.MapHelper;
import se.consys.services.GenericService;

import java.time.LocalDate;
import java.time.LocalDateTime;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import java.util.Map;


@SuppressWarnings("rawtypes, unchecked")
@Path("courses")
@Produces(MediaType.APPLICATION_JSON)
@Consumes(MediaType.APPLICATION_JSON)
public class CourseController {

    private GenericService courseService = GenericService.getGenericService(new DaoGenericHibernateImpl<>(Course.class));
    private GenericService teacherService = GenericService.getGenericService(new DaoGenericHibernateImpl<>(Teacher.class));
    private GenericService studentService = GenericService.getGenericService(new DaoGenericHibernateImpl<>(Student.class));
    private String noCourseFoundMsg = "No course found with the specified id.";


    @GET
    public Response getAll() {
        List<Course> courses = courseService.findAll();
        return Response.status(200).build();
    }

    @GET
    @Path("/{id}")
    public Response getById(@PathParam("id") int id) {
        try {
            Course course = (Course) courseService.findById(id);
            return Response.ok().entity(course).build();
        } catch (NoResultException e) {
            System.out.println(noCourseFoundMsg);
            return Response.status(204).build();
        }
    }

    @SuppressWarnings("unchecked")
    @POST
    public Response create(Course entity) {
        courseService.create(entity);
        return Response.status(201).entity(entity).build();
    }

    @PATCH
    @Path("/{id}")
    public Response partialUpdate(@DefaultValue("0") @PathParam("id") int id, 

            @DefaultValue("null") @QueryParam("name") String courseName,
            @DefaultValue("-1") @QueryParam("duration") int durationInMonths,
            @DefaultValue("") @QueryParam("end") LocalDateParam endDate,
            @DefaultValue("") @QueryParam("start") LocalDateParam startDate,
            @DefaultValue("") @QueryParam("timestamp") LocalDateTimeParam timeStamp,
            @DefaultValue("-1") @QueryParam("supervisor") int supervisor)
            {
        Course courseToBeUpdated = (Course) courseService.findById(id); 
        System.out.println(courseName);
        if (courseName != null) courseToBeUpdated.setCourseName(courseName);
        if (durationInMonths != -1) courseToBeUpdated.setDurationInMonths(durationInMonths);
        if (endDate != null && !endDate.getLocalDate().equals(LocalDate.MIN)) courseToBeUpdated.setEndDate(endDate.getLocalDate());
        if (startDate != null && !startDate.getLocalDate().equals(LocalDate.MIN)) courseToBeUpdated.setStartDate(startDate.getLocalDate());
        if (timeStamp != null && !timeStamp.getLocalDateTime().equals(LocalDateTime.MIN)) courseToBeUpdated.setTimeStamp(timeStamp.getLocalDateTime());
        if (supervisor != -1) courseToBeUpdated.setSupervisor((Teacher) teacherService.findById(supervisor));

        courseService.update(courseToBeUpdated);
        return Response.status(200).build();
    }

    @PATCH
    @Path("/{id}/students")
    public Response partialUpdateOnStudents(
            @DefaultValue("0") @PathParam("id") int id,
            @DefaultValue("null") @QueryParam("update") String studentString) {
        String[] seperatedIds = studentString.split("-");
        List<Integer> studentIds = new ArrayList<Integer>();
        for (int i = 0; i < seperatedIds.length; i++) {
            studentIds.add((int) Integer.parseInt(seperatedIds[i]));
        }

        List<Student> allStudents = studentService.findAll();
        List<Student> StudentsToAddIntoCourse = new ArrayList<Student>();
        for (int i = 0; i < allStudents.size(); i++) {
            for(int j = 0; j < studentIds.size(); j++) {
                if (allStudents.get(i).getId() == studentIds.get(j)) {
                    StudentsToAddIntoCourse.add(allStudents.get(i));
                }
            }
        }

        Course courseToBeUpdated = (Course) courseService.findById(id);
        if (studentString != null) courseToBeUpdated.setStudents(StudentsToAddIntoCourse);
        courseService.update(courseToBeUpdated);

        return Response.status(200).build();
    }

    @PUT
    @Path("/{id}")
    public Response update(@DefaultValue("0") @PathParam("id") int id, Course entity) {
        try {
            Course courseToBeUpdated = (Course) courseService.findById(id);
            courseToBeUpdated.setCourseName(entity.getCourseName());
            courseToBeUpdated.setDurationInMonths(entity.getDurationInMonths());
            courseToBeUpdated.setEndDate(entity.getEndDate());
            courseToBeUpdated.setScheduledLectures(entity.getScheduledLectures());
            courseToBeUpdated.setStartDate(entity.getStartDate());
            courseToBeUpdated.setStudents(entity.getStudents());
            courseToBeUpdated.setSupervisor(entity.getSupervisor());
            courseToBeUpdated.setTimeStamp(entity.getTimeStamp());
            courseService.update(courseToBeUpdated);
            return Response.status(200).entity(entity).build();
        } catch (NoResultException e) {
            System.out.println(noCourseFoundMsg);
            return Response.ok().status(204).build();
        }
    }

    @DELETE
    @Path("/{id}")
    public Response delete(@DefaultValue("0") @PathParam("id") int id) {
        try {
            Course courseToBeDeleted = (Course) courseService.findById(id);
            courseService.delete(courseToBeDeleted);
            return Response.status(200).build();
        } catch (NoResultException e) {
            System.out.println(noCourseFoundMsg);
            return Response.status(204).build();
        } 
    }
}

我怀疑问题实际上是我的通用 dao 和 dao 服务可能完全错误但在纸上有效。我是泛型的新手,并设法将一些东西放在一起。

DaoGenericHibernateImpl

package se.consys.dataaccess;

import java.io.Serializable;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.List;

import org.hibernate.Session;

import se.consys.Utilities.Helper;
import se.consys.Utilities.HibernateUtility;
import se.consys.services.GenericService;

public class DaoGenericHibernateImpl<T extends Serializable> implements IGenericDao<T> {

    Session session = HibernateUtility.getSessionFactory().openSession();
    private String activeClassName;
    private String wrongClassError = "ERROR: Wrong class used on the established service.";

    public DaoGenericHibernateImpl(Class<T> type) {
        activeClassName = type.getSimpleName();
    }

    @Override
    public void create(T entity) {
        if (entity.getClass().getSimpleName().equals(activeClassName)) {
            session.beginTransaction();
            session.persist(entity);
            session.getTransaction().commit();
        } else {
            System.out.println(wrongClassError + " Entity has not been saved to the database. "
                + "Class used: " + entity.getClass().getSimpleName() + ". "
                + "Class expected: " + activeClassName + ".");
        }
    }

    @Override
    public T update(T entity) {
        if (entity.getClass().getSimpleName().equals(activeClassName)) {
            session.beginTransaction();
            session.merge(entity);  
            session.getTransaction().commit();
            return entity;
        } else {
            System.out.println(wrongClassError + " Entity has not been updated. "
                + "Class used: " + entity.getClass().getSimpleName() + ". "
                + "Class expected: " + activeClassName + ".");
        }
        return entity;
    }

    @Override
    public void delete(T entity) {
        if (entity.getClass().getSimpleName().equals(activeClassName)) {
            session.beginTransaction();
            //session.update(entity);
            session.delete(entity);

            session.getTransaction().commit();
        } else {
            System.out.println(wrongClassError + " Entity has not been deleted. "
                + "Class used: " + entity.getClass().getSimpleName() + ". "
                + "Class expected: " + activeClassName + ".");
        }       
    }

    @Override
    public T findById(int id) {
        final String HQL_BY_ID = "FROM " + activeClassName + " WHERE id=:id";

        @SuppressWarnings("unchecked")
        T result = (T) session.createQuery(HQL_BY_ID)
            .setParameter("id", id)
            .setMaxResults(1)
            .getSingleResult();     
        return  result;
    }

    @Override
    public List<T> findAll() {
        String HQL_FIND_ALL = "FROM " + activeClassName;

        @SuppressWarnings("unchecked")
        List<T> result = (List<T>) session.createQuery(HQL_FIND_ALL)
            .getResultList();
        return  result;
    }

    @Override
    public void removeReference(T entity, Class<?> reference) {
        Method setter = Helper.findSetter(entity, reference);
        try {
            setter.invoke(entity, null);
        } catch (IllegalAccessException | IllegalArgumentException | InvocationTargetException e) {
            throw new RuntimeException(e.getMessage());
        }
    }
}

我觉得我可以继续添加代码块,所以我会在我的github上添加一个链接,一切都在那里,希望没关系。

谢谢您的帮助。

4

1 回答 1

0

一般来说,在这种情况下,您首先需要考虑的是,如果您无法推断代码的内存安全和/或线程安全,那么您实际上有两个问题需要解决。

就您实际看到的情况而言,您需要的是用于识别哪些线程正在运行以及哪些内存正在使用的标准工具。您可以使用 jconsole (GUI) 或 jstack (CLI) 等工具找出正在运行的线程。它们都作为 JDK 的标准部分包含在内,将为您提供系统中运行的所有线程的堆栈跟踪。它们不一定会告诉您这些线程存在的原因,但堆栈跟踪本身可能会帮助您确定它们的来源。

您描述的问题听起来像是一个失控的线程,尽管它可能与内存有关,并且错误消息表明这可能是根本原因,因此查看内存使用情况也可能很有用。为此,您还可以使用 jconsole(配置文件部分)或 jhat & jmap(如果您是 CLI 爱好者)。其中任何一个都会告诉您 VM 堆中存在哪些对象以及它们是什么类型。这些信息非常有用,但也非常分散注意力——大多数堆都由字符串、映射和列表主导,因为大多数程序都是由这些组成的。尽管如此,您通常可以通过比较两个配置文件之间的差异来获得有用的见解,一个是在“一切似乎都很好”时拍摄的,另一个是在“它停止工作”时拍摄的。

这些工具可以帮助您识别正在运行的系统中的问题。完成此操作后,您可能想看看为什么从程序文本中不清楚会有线程池或内存使用问题。使用 Map 作为缓存之类的事情是一件大事。或者根据可变字段实现equals/hashcode。对于线程,任何可能无法终止的东西都会很快堵塞线程池——例如阻塞网络服务器中的 IO。

于 2018-04-29T14:17:37.880 回答