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

SEPARATOR关键字在休眠公式中无法正常工作

钱浩荡
2023-03-14
问题内容

我有以下Hibernate forumla查询,我可以在mysql workbanch中执行该查询。

select group_concat(distinct t.column_1_name SEPARATOR ', ') from table_name t and t.fk_record_id = record_id

使用Hibernate执行此查询时,hibernate会将父表追加到SEPRATOR关键字,如以下查询所示。

select group_concat(distinct t.column_1_name parent_table.SEPARATOR ', ') from table_name t and t.fk_record_id = record_id

在这里,hibernate不将SEPRATOR当作关键字。有人对此有任何想法吗?


问题答案:

您可以添加SEPARATOR为关键字。实现自己DialectResolver的关键字,并将关键字 以小写形式 添加到所得的方言中:

public class MyDialectResolver implements DialectResolver {

    public Dialect resolveDialect(DialectResolutionInfo info) {
        for (Database database : Database.values()) {
            Dialect dialect = database.resolveDialect(info);
            if (dialect != null) {
                dialect.getKeywords().add("separator");
                return dialect;
            }
        }

        return null;
    }
}

对于5.2.13 / 5.3.0 之前的 Hibernate版本相同:

public class MyDialectResolver extends StandardDialectResolver {

    protected Dialect resolveDialectInternal(DatabaseMetaData metaData) throws SQLException {
        Dialect dialect = super.resolveDialectInternal(metaData);
        dialect.getKeywords().add("separator");
        return dialect;
    }

}

然后,您将不得不告诉Hibernate使用您的方言解析器。例如,在JPA中,您可以在persistence.xml中执行此操作:

<persistence>
  <persistence-unit>
    ...
    <property name="hibernate.dialect_resolvers" value="mypackage.MyDialectResolver"/>
  </persistence-unit>
</persistence>

其他方言中的汇总功能也是如此。例如,在Oracle中,WITHIN缺少关键字。

还有另一种选择,它更独立于数据库(我更喜欢)。创建以下内容SQLFunction

public class ListAggFunction implements SQLFunction {

    /**
     * The pattern that describes how the function is build in SQL.
     *
     * Replacements:
     * {path} - is replaced with the path of the list attribute
     * {separator} - is replaced with the separator (defaults to '')
     * {orderByPath} - is replaced by the path that is used for ordering the elements of the list
     */
    private String pattern;

    /**
     * Creates a new ListAggFunction definition which uses the ANSI SQL:2016 syntax.
     */
    public ListAggFunction() {
        this("LISTAGG(DISTINCT {path}, {separator}) WITHIN GROUP(ORDER BY {orderByPath})");
    }

    /**
     * Creates a new ListAggFunction definition which uses a database specific syntax.
     *
     * @param pattern  The pattern that describes how the function is build in SQL.
     */
    public ListAggFunction(String pattern) {
        this.pattern = pattern;
    }

    public Type getReturnType(Type firstArgumentType, Mapping mapping) throws QueryException {
        return StringType.INSTANCE;
    }

    public boolean hasArguments() {
        return true;
    }

    public boolean hasParenthesesIfNoArguments() {
        return true;
    }

    public String render(Type firstArgumentType, List arguments,
            SessionFactoryImplementor factory) throws QueryException {
        if (arguments.isEmpty() || arguments.size() > 3) {
            throw new IllegalArgumentException(
                    "Expected arguments for 'listagg': path [, separator [, order by path]]");
        }

        String path = (String) arguments.get(0);
        String separator = arguments.size() < 2 ? "''" : (String) arguments.get(1);
        String orderByPath = arguments.size() <= 2 ? path : (String) arguments.get(2);

        return StringUtils.replaceEach(this.pattern, new String[] { "{path}", "{separator}", "{orderByPath}" },
                new String[] { path, separator, orderByPath });
    }

}

您可以使用与上面的关键字相同的方式在DialectResolver中注册此功能:

 if ("MySQL".equals(info.getDatabaseName()) || "H2".equals(info.getDatabaseName())) {
   dialect.getFunctions().put("listagg", new ListAggFunction("GROUP_CONCAT(DISTINCT {path} ORDER BY {orderByPath} SEPARATOR {separator})"));
 } else {
   dialect.getFunctions().put("listagg", new ListAggFunction());
 }

现在,您可以在JPQL / HQL / Criteria查询中使用此功能,而无需考虑方言的语法:

 SELECT e.group, listagg(e.stringProperty, ', ') FROM Entity e GROUP BY e.group


 类似资料:
  • 问题内容: 我正在使用hibernate4.1.9。我的代码是 包是 仍然在hibernate查询中,它不起作用并将属性放入查询中。 查询片段的一部分(assetasset0_.ldapIdTemp作为ldapIdTemp16_0_,) 我不确定自己在做什么错。 问题答案: 您可以尝试为该字段创建setter和getter并使用来注释get方法,如下所示:

  • 问题内容: 我有一个复合主键的表,其中指的是实体 HolidayPackage 有 许多 到 一个 关系 HolidayPackageVariant 和 HolidayPackage 。 当我尝试在HolidayPackageVariant中进行复杂的PK映射时,出现以下错误: 最初的SessionFactory创建失败。 有人可以告诉我我在做什么错吗? 我的POJO如下所示: HolidayPa

  • 问题内容: 我想使用设置实体。 我尝试了以下操作,但结果始终是(我想是因为该列始终是): 似乎忽略了。 如何使用某实体引用实体? 问题答案: 被忽略,因为它仅可替代。而且那个不用于关系映射。 但是您可以改为使用它,它存在于集合中: 要访问原始对象,可以使用特定的吸气剂:

  • 问题内容: 我对 Hibernate 有问题。我尝试解析为List,但抛出异常:。当我调试时,它在线路上出错… 我的示例代码在这里 我的豆子 例外: 我的数据库: 问题答案: 该批注指定列的名称被用作对目标实体的外键。 在上面的类中,连接列的名称设置为。 但是,表上的外键称为 您需要更改表上的列名或您在中使用的名称,以使它们匹配。参见http://docs.jboss.org/hibernate/

  • 问题内容: 当我运行以下脚本时,两个lambda都在同一文件junk.txt上运行os.startfile()。我希望每个Lambda都使用创建Lambda时设置的值“ f”。有没有办法让它按我的预期运行? 问题答案: 一种方法是执行此操作: 否则,在调用该函数时将进行查找,因此您将获得当前(循环后)的值。 我更喜欢的方式: 甚至(在python 2.5+中):

  • 问题内容: 尽管我很确定这是昨天或前一天工作的,例如,在IE10中不再起作用。我已经测试了我的浏览器,但是它不再起作用了。还有谁有相同的问题吗?或者,它永远都行不通吗? 问题答案: IE不支持输入type =“ number”,但您可以使用jQueryUISpinner小部件。它非常易于使用,并且具有许多对开发人员友好的API。