Hibernate query.list()需要很长时间才能响应

时间:2016-11-02 19:06:19

标签: java mysql json spring hibernate

我正在开发一个接受并返回JSON数据的REST服务器。对于特定的输入值(来自输入JSON),Hibernate(与Spring集成)查询数据库并将JSON发送回Rest Controller。客户端应用程序设计为超时5秒,用于读取响应,不能更改。

现在问题有时(并非总是如此)Hibernate无法在给定的时间限制内处理数据,因此在客户端收到超时错误。

我在网上检查了其他帖子,以便在bean类中使用完整的构造函数并进行延迟加载。在我的情况下,两者都是真的。下面是导致此问题的DAOImpl方法之一。在这里,我必须查询2个表(它们没有很多数据,每个表中大约有20个条目),将所有数据添加到json数组并发回。

@Override
   public String getOfferOnRuleNameBalance(String inputJsonString) {
    JSONObject saveJsonObject = new JSONObject(inputJsonString);
    String ruleName    =   saveJsonObject.getString("rulename");
    int currentAccountBalance    =   saveJsonObject.getInt("currentAccountBalance");
    Session session1 = getSession();
    Criteria criteria =session1.createCriteria(PaymentPlanOffers.class);
    criteria.add(Restrictions.eq("rulename", ruleName));
    @SuppressWarnings("unchecked")
    List<PaymentPlanOffers> offerList=criteria.list();
    JSONArray jsonArray = new JSONArray();
    for(PaymentPlanOffers object:offerList)
    {
        JSONObject jsonObject1 =new JSONObject();
        jsonObject1.put("instAmount",object.getAmountPercent());
        jsonObject1.put("instNumber", object.getNumInstallment());
        jsonObject1.put("frequency", object.getFrequency());
        jsonObject1.put("offerId", object.getId());
        jsonObject1.put("offerName", object.getOfferName());
        jsonObject1.put("active", object.isActive());
        jsonObject1.put("accepted", object.isAccepted());
        jsonObject1.put("totalAmount", currentAccountBalance);
        jsonObject1.put("startDate", object.getStartDate());
        jsonObject1.put("endDate", object.getEndDate());
        jsonArray.put(jsonObject1);
    }

    Criteria criteria2 =session1.createCriteria(CustomPlanOffer.class);
    criteria2.add(Restrictions.eq("rulename", ruleName));
    @SuppressWarnings("unchecked")
    List<CustomPlanOffer> customOfferList=criteria2.list();
    for(CustomPlanOffer object:customOfferList)
    {
        JSONObject jsonObject1 =new JSONObject();
        jsonObject1.put("instAmount", object.getAvgInstallment());
        jsonObject1.put("instNumber", object.getNumOfInstallment());
        jsonObject1.put("frequency", object.getFrequency());
        jsonObject1.put("offerId", object.getId());
        jsonObject1.put("offerName", object.getName());
        jsonObject1.put("active", object.isActive());
        jsonObject1.put("accepted", object.isAccepted());
        jsonObject1.put("totalAmount", object.getTotalPaymentAmount());
        jsonObject1.put("startDate", object.getStartDate());
        jsonObject1.put("endDate", object.getEndDate());
        jsonArray.put(jsonObject1);
    }

    JSONObject mainObj = new JSONObject();
    mainObj.put("allOffers", jsonArray);
    session1.close();
    return mainObj.toString();
}

如果我以正确的方式实施,请告诉我。

编辑:发布另一种导致所有修改类似问题的方法

@Override
public String getAllOffers(String inputJsonString) {
    Session session = getSession();
    Transaction t = session.beginTransaction();
    String hql = "FROM PaymentPlanOffers";
    Query query = session.createQuery(hql);
    @SuppressWarnings("unchecked")

    List<PaymentPlanOffers> results = query.list(); //this is where it goes on hang


    JSONArray jsonArray = new JSONArray();
    for(PaymentPlanOffers object:results)
    {
        JSONObject jsonObject1 =new JSONObject();
        jsonObject1.put("offername", object.getOfferName());
        jsonObject1.put("rulename", object.getRulename());
        jsonObject1.put("id", object.getId());
        jsonObject1.put("offerMessage", object.getOfferMessage());
        jsonArray.put(jsonObject1);
    }
    JSONObject mainObj = new JSONObject();
    mainObj.put("allOffers", jsonArray);
    t.commit();
    session.close();
    return mainObj.toString();
}

1 个答案:

答案 0 :(得分:0)

我要做的第一件事就是记录正在生成的SQL查询(see here)。一旦你拥有它们,你可以做更多的挖掘:

  • 您确定数据库本身不是问题吗?错误的表和/或连接索引是常见原因。
    • 记录getOfferOnRuleNameBalance
    • 的开始和结束时间
    • 记录每次查询的开始和结束时间
    • 检查生成的SQL查询在手动运行数据库时是否表现不佳
  • 您的hibernate映射和代码是否导致N + 1选择?您将知道是否为延迟加载的集合中的每个项目生成了多个SELECT语句。这也可能是由某个循环引起的,或者是由equals()toString()hashCode()方法访问延迟加载的集合并检查其中的每个元素引起的。

记录每个查询还会告诉您它是否只是导致问题的一个问题,或者它是否都是查询。

为了进行更多的故障排除,我必须看到hibernate对象本身的映射,但这至少可以为你提供一个起点。