当前位置: 首页 > 面试题库 >

通过服务API公开休眠条件

龙隐水
2023-03-14
问题内容

这更多的是设计而不是实现问题,而且会很长,请耐心等待。最好用一个例子来解释:

假设我有一个名为 Product 的业务实体,该实体具有许多属性( 名称价格供应商 等)。

它由一个接口( 产品 )和实现( ProductImpl ,在Hibernate中映射)以及基本的CRUD服务接口(
ProductService )和实现( ProductServiceImpl )表示。
ProductProductService 作为API公开,而其实现则没有。

我想向 ProductService* 添加一个 List findProducts(QueryCriteria条件)
方法,该方法将返回满足给定条件的产品列表。要求是:
*

  1. 通过直接 产品 属性查询(例如product.price gt 50.0
  2. 按关联查询(例如product.vendor.name = "Oracle"
  3. 排序结果(例如order by product.vendor.name desc, product.price asc"
  4. 应用其他过滤器。与API客户端指定的以上3个项目不同,服务可以基于客户端的身份应用其他过滤器(例如,调用此方法的客户端可能仅限于查看给定供应商生产的产品)。此类过滤器优先于客户端指定的任何条件(例如,如果过滤器设置为product.vendor.name = "Microsoft",则上述(2)中的查询应生成空结果集。

因此,问题是,这种方法使用的 QueryCriteria 接口应该是什么样?我可以想到3个解决方案,但我不喜欢其中任何一个:

  • 允许客户端直接指定HQL(以“ where”子句开头)。这是最直接的解决方案,也是最有问题的安全性。即使假设过滤器(上面的#4)足够简单,可以通过Hibernate的会话过滤器实现,HQL仍然需要解析为-至少-确保将查询参数指定为参数而不是内联。
  • 使用薄包装的Hibernate的 DetachedCriteria 代替 QueryCriteria 。“薄包装”是因为不允许客户端直接创建 DetachedCriteria ,因为无法控制为其创建的映射实体。而且,对于某些查询而言,这不像HQL那样灵活,无法通过Criteria API轻松(或根本无法)表达。与HQL方法一样,过滤器(上面的#4)将限于Hibernate会话过滤器。
  • 编写我自己的 QueryCriteria 接口/实现,该接口/实现将在后台形成DetachedCriteria或HQL。虽然这可能是最灵活的解决方案,但这将不得不复制Criteria API的许多代码,这似乎不太理想。

任何对上述方法的有效性的评论,或者-手指交叉-我没有想到的简单优雅的解决方案,将不胜感激。

PS在我的特定情况下,所有API客户端都是内部的并且是“半信任的”-也就是说,我与试图故意破坏某些内容的人并没有因为编程不良而导致生成5张表的笛卡尔积无关:-)但是,提出一个可以承受API对公众开放的解决方案将是非常不错的。


问题答案:

我实现的实际解决方案使用混合方法。

使用定义明确的查询的方法(例如,其他服务内部使用的方法,预定义的报告等)的签名类似于HibernateTemplate的findBy方法:

public List<Entity> findEntities(String queryName, QueryParameters parameters);

其中QueryParameters是一个方便类,用于显式指定命名参数或从Bean中获取它们。用法示例为:

List<Product> products = findProducts("latestUpdates",
 new QueryParameters()
  .add("vendor", "Oracle")
  .add("price", "50.0")
);

要么

List<Product> products = findProducts("latestUpdates",
 new QueryParameters(product, "vendor", "price"));

对此类方法的访问仅限于“可信”代码;显然,必须在Hibernate映射中定义使用的查询。过滤器内置于查询中或定义为会话过滤器。这样做的好处是代码更简洁(没有类似Criteria的内容散布在半页上)和明确定义的HQL(更容易在必要时优化和处理缓存)。

暴露给UI的方法或需要更加动态的方法,请使用Hibernate-Generic-
DAO
项目的Search界面。它与Hibernate的DetachedCriteria有点相似,但具有几个优点:

  1. 可以创建它而不必绑定到特定实体。对我来说这很重要,因为实体接口(用户可见的API的一部分)和实现(在Hibernate中映射为POJO)是两个不同的类,并且实现在编译时对用户不可用。

  2. 这是一个经过深思熟虑的开放界面;与DetachedCriteria完全不同,后者几乎不可能从中提取任何内容(是的,我知道DC并非为此目的而设计的;但仍然如此)

  3. 内置分页/带有总数的结果/一堆其他小细节。

  4. 与Hibernate没有明确的联系(尽管我个人并不十分在意;我不会突然放弃Hibernate并明天使用EclipseLink);有可用的Hibernate和通用JPA实现。

可以将过滤器添加到服务端的“搜索”中;那也是指定实体类的时候。唯一缺少的是,如果指定了无效的属性名称,则客户端会快速失败,这可以通过编写我自己的ISearch
/ IMutableSearch实现来解决,但我还没有解决。



 类似资料:
  • 问题内容: 是否可以使用hibernatecritiria api创建“选择输入”查询? 示例:我有两个1:n关系中的表,公司和部门 问题答案: 您可以使用此DetachedCriteria

  • 问题内容: 我在Heroku服务器上托管了一个网站( 我是Heroku btw的新用户 ),并且由于它是免费软件包,因此在 闲置30m 后便 进入 休眠 状态 ,并在用户点击它后再次将其投入使用,大约需要7秒才能成功。 我正在考虑运行nodejs作业或每隔29m就会打开网站的一次作业,以使服务器永不休眠,最初,我得到的是这样的内容: 注意:那只是在浏览器中打开它,而不是关闭它。 首先,这样做是否合

  • 问题内容: JodaTime有一个提供Hibernate持久性的库。最近,我开始研究Joda-Money,开始了解如何使用hibernate来保持这种状态,而我看不到任何库。 有什么建议么? 问题答案: 好吧,我把你的意见,并炮制了钱的自定义类型为乔达库中的定义,作为参考的人可以看看它在这里,在这里使用和测试自定义类型在这里

  • 问题内容: 我需要使用detachedCriteria进行此SQL查询: 这个想法是从不属于用户的游戏中获取ID。我尝试了与detachedCriteria类似的10种不同方法,但得到了“未知实体:null” MappingException代码应类似于: 还设置投影以仅返回游戏的ID。 有任何想法吗?我认为Hibernate在加入没有别名的查询时会遇到一些麻烦。添加别名是可行的,但结果是完全错误

  • 此任务用于在指定时间内进入睡眠过程。时间可以是毫秒,秒或小时。 在部署应用程序时需要在任务之间间隔时非常有用。 它使用下面给出的各种属性。 1. Apache Ant睡眠任务属性 属性 描述 必需 设定睡眠小时数 否 设定睡眠的分钟数 否 设置睡眠秒数 否 设置睡眠时间(以毫秒为单位) 否 标志控制是否打断错误的构建 否 2. Apache Ant睡眠任务示例 指定睡眠的毫秒数: 指定睡眠的秒数:

  • 问题内容: 我有这样的JPA实体类: 我需要这样的搜索方法 用法示例: 我对findAll方法的想法是 但这太丑陋了,看起来有些开销。谁能帮助我更礼貌地邀请该方法?我可以使用任何Java技术或框架。 问题答案: 所以您要寻找 原型 吗?Hibernate为此提供了一个方便的标准,因此,如果您不介意将自己与Hibernate API捆绑在一起,请尝试以下来自docs的示例: 它具体说: 版本属性,标