在 createNativeQuery 中使用 generate_series 时出错
Posted
技术标签:
【中文标题】在 createNativeQuery 中使用 generate_series 时出错【英文标题】:Error while using generate_series in createNativeQuery 【发布时间】:2019-04-09 13:42:32 【问题描述】:在 PostgreSQL 数据库中执行的查询正在正确获取记录。但是,当使用 createnativeQuery 实现时,我收到错误
ERROR SqlExceptionHelper:131 - ERROR: syntax error at or near ":"
在 Postgres 数据库中成功执行的查询如下所述
select d.dt::date as date_current,
coalesce(underwriter_daily_cap, daily_file_cap) as file_cap,
user_id
from generate_series(date '2019-04-05', date '2019-04-09', interval '1'
day) as d(dt)
left join (
select upd.*, ud.daily_file_cap
from mod_underwriter_daily_file_count upd
join lsa_assignment_details ud on ud.user_id = upd.user_id
where ud.user_id = 350
) upd on upd.date_current = d.dt::date
ORDER BY d.dt DESC;
DAO层代码如下
@SuppressWarnings("unchecked")
public List<Object[]> getUnderWriterCap(String dynamicUserId)
log.info("Entered UserSkillMappingHibDAO :
getUnderWriterCap");
Session session = sessionFactory.getCurrentSession();
Date currentDateMinusFive =
DateUtil.getLastFiveDayOldPSTDate();
Date currentDate = DateUtil.getCurrentPSTDate();
String queryString = " select d.dt::date as date_current,
coalesce(underwriter_daily_cap, daily_file_cap) as file_cap, user_id "
+" from generate_series ( date '"+DateUtil.getDateFormat(currentDateMinusFive)+"', date '"+DateUtil.getDateFormat(currentDate)+"', INTERVAL '1' DAY ) as d(dt) "
+" left join ( select upd.*, ud.daily_file_cap from mod_underwriter_daily_file_count upd "
+" join lsa_assignment_details ud on ud.user_id = upd.user_id where ud.user_id = "+ dynamicUserId
+") upd on upd.date_current = d.dt::date ORDER BY d.dt DESC " ;
List<Object[]> records = session.createNativeQuery(queryString)
.getResultList();
return records;
执行上述代码时控制台出现的异常如下
Hibernate: select d.dt:date as date_current, coalesce(underwriter_daily_cap, daily_file_cap) as file_cap, user_id from generate_series ( date '2019-04-04', date '2019-04-09', INTERVAL '1' DAY ) as d(dt) left join ( select upd.*, ud.daily_file_cap from mod_underwriter_daily_file_count upd join lsa_assignment_details ud on ud.user_id = upd.user_id where ud.user_id = 403) upd on upd.date_current = d.dt:date ORDER通过 d.dt DESC 2019-04-09 18:58 错误 SqlExceptionHelper:131 - 错误:“:”处或附近的语法错误 位置:14
【问题讨论】:
中间(两个大代码块之间)的文本块是什么?如果它是一个错误跟踪,它应该被格式化。 【参考方案1】:这里有几个错误。比语法错误更重要的是如何将参数附加为字符串文字。
代替:
session.createNativeQuery("select * from tbl where param = '" + value + "'");
您必须使用准备好的语句(即要查找的术语)。这对安全非常重要。
Query q = session.createNativeQuery("select * from tbl where param = :par1");
q.setParameter("par1", value);
现在关于语法错误,它是由您使用 PostgreSQL 的自定义类型转换语法引起的。您必须将其替换为从d.dt::date
到cast(d.dt as date)
的标准SQL 类型转换,因为Hibernate 期望:
是命名参数的开始。
String queryString = "select cast(d.dt as date) as date_current, "
+" coalesce(underwriter_daily_cap, daily_file_cap) as file_cap, user_id "
+" from generate_series ( current_date - 5, current_date, INTERVAL '1' DAY ) as d(dt) "
+" left join ( select upd.*, ud.daily_file_cap from mod_underwriter_daily_file_count upd "
+" join lsa_assignment_details ud on ud.user_id = upd.user_id where ud.user_id = :dynuid ) upd on upd.date_current = cast(d.dt as date) ORDER BY d.dt DESC " ;
List<Object[]> records = session.createNativeQuery(queryString)
.setParameter("dynuid", dynamicUserId)
.getResultList();
我还用数据库函数替换了你当前的日期 java 计算。而不是这样做:
Date currentDateMinusFive = DateUtil.getLastFiveDayOldPSTDate();
Date currentDate = DateUtil.getCurrentPSTDate();
...
.setParameter("ds", DateUtil.getDateFormat(currentDateMinusFive))
.setParameter("de", DateUtil.getDateFormat(currentDate))
您让数据库使用current_date - 5
和current_date
为您执行此操作。
您应该注意的另一件事,因为您的 DateUtil 使用旧的 java.util.Date
类,并且相关的东西是,如果您使用 SimpleDateFormatter
来格式化它,那么它必须是每次调用的新实例.不要将其优化为静态 final,因为它不是线程安全的。我受此折磨。
编辑:我也相信您的查询可以进一步优化,它会得到相同的结果:
select upd.date_current, underwriter_daily_cap, daily_file_cap from mod_underwriter_daily_file_count upd
join lsa_assignment_details ud on ud.user_id = upd.user_id
where ud.user_id = :dynuid and
upd.date_current between current_date - 5 and current_date
ORDER BY d.dt DESC
请注意,我从选择中删除了一些字段,例如 user_id,因为您已经告诉数据库您想要哪个字段。一个主要区别是此查询将跳过没有条目的几天的结果。
【讨论】:
您好,您在第一个解决方案中提到的查询部分有效。即,有两件事待处理。 1. 没有从 lsa_assignment_details 表中获取 non-matched 记录的值。 2. user_id 在所有记录中为空。 第一个解决方案中的查询是您的查询,但具有参数安全性。lsa_assignment_details
和 mod_underwriter_daily_file_count
一起左连接到来自 generate_series
的结果。如果那里什么都没有,则它们为空。我对您的数据结构知之甚少,无法进一步提供帮助。您应该尝试将其拆分为不太复杂的查询来验证您的结果。以上是关于在 createNativeQuery 中使用 generate_series 时出错的主要内容,如果未能解决你的问题,请参考以下文章
Hibernate CreateNativeQuery返回空列表
使用 entityManager.createNativeQuery(query,foo.class)
JPA 中的最大打开游标超出异常(使用 createNativeQuery 删除记录)