oracle - 我可以在不使用 where 子句的情况下创建自定义表达式吗?

标签 oracle nhibernate connect-by nhibernate-criteria

虽然我已经在上一个问题中使用 native 查询解决了这个问题。我现在想知道是否可以在不使用 where 子句的情况下创建可在 Criteria 中使用的自定义表达式?我不想要 where 子句的原因是因为 Oracle 的 connect by ... start with ... ( here ) 语句。我关注了this页面开始。但是,这将生成类似 select * from foo where connect by start with...

的代码

这是我正在使用的。查看生成的内容,我可以说它生成了正确的语句,但不包含 where 子句。

public class StartWithConnectByCriteria : AbstractCriterion
{
    public StartWithConnectByCriteria(string parentName, string parentValue, string childName)
    {
        ParentName = parentName;
        ParentValue = parentValue;
        ChildName = childName;
    }

    public string ParentName { get; set; }
    public string ParentValue { get; set; }
    public string ChildName { get; set; }
    public IProjection P { get; set; }

    public override IProjection[] GetProjections()
    {
        if(P != null)
        {
            return new IProjection[] {P};
        }
        return null;
    }

    public override TypedValue[] GetTypedValues(ICriteria criteria, ICriteriaQuery criteriaQuery)
    {
        return
            CriterionUtil.GetTypedValues(criteriaQuery, criteria, P, ParentName, ParentValue.ToString());
    }

    public override SqlString ToSqlString(ICriteria criteria, ICriteriaQuery criteriaQuery,
                                          IDictionary<string, IFilter> enabledFilters)
    {
        var sqlBuilder = new SqlStringBuilder();
        SqlString[] parentColumnNames = CriterionUtil.GetColumnNames(ParentName,
                                                               P, criteriaQuery,
                                                               criteria, enabledFilters);
        SqlString parentColumnName = parentColumnNames[0];

        SqlString[] childColumnNames = CriterionUtil.GetColumnNames(ChildName,
                                                   P, criteriaQuery,
                                                   criteria, enabledFilters);
        SqlString childColumnName = childColumnNames[0];

        criteriaQuery.AddUsedTypedValues(GetTypedValues(criteria, criteriaQuery));
        sqlBuilder
            .Add("start with " + parentColumnName + " = '" + ParentValue + "'")
            .Add(" connect by prior " + childColumnName + " = " + parentColumnName);

        return sqlBuilder.ToSqlString();
    }

    public override string ToString()
    {
        return "";
    }
}

我是这样用的

StartWithConnectByCriteria criterion = 
    new StartWithConnectByCriteria(
        "parent", 
        "parent_value", 
        "child");

DetachedCriteria dc = DetachedCriteria.For<NormalUpstream>("nu")
    .Add(criterion);

我感觉它与 DetachedCriteria 中的 .Add() 有关,但不是 100% 确定。不幸的是,我似乎找不到很多关于创建自定义表达式的文档。

编辑:现在我想起来好像我找错了树。虽然这并不重要(我已经有了一个不错的实现)。我仍然对了解如何进一步自定义 NHibernate 很感兴趣。

编辑 2:由于开箱即用的 NHibernate 不支持 Oracle 的专有功能,开始于 ... connect by。我试图通过添加对 NHibernate 的 native 支持来了解更多关于扩展 NHibernate 的信息。我知道我可以使用自定义方言注册这些功能。但我想知道是否有可能将它作为一个条件来实现,以便我可以将它与我的其他条件查询一起使用。我发布的代码工作正常并正确地创建了有效的 SQL,但是当我将 StartWithConnectByCriteria 添加到我的条件时,NHibernate 将发出一个查询,例如 select this_.id from table where start with ... connect by。这是无效查询,因为该子句不属于 where。

这是我希望 NHibernate 生成的查询。

select
    random_column
from
    table
start with parent_id = 'parent_node_id'
connect by prior child_up_id = parent_id

请注意此查询中没有 where 子句。但是,start with ... connect by 仍可与 where 子句 一起使用。您可以阅读有关这些关键字如何工作的更多信息 here .

最佳答案

我不知道现有的 NHibernate 语法是否允许这样做,但是有一个用于分层查询的 ANSI 标准语法可能会很有用。但是,我相信它只适用于 11R2 及更高版本,所以我不确定它是否对您有用。参见 Recursive Subquery Refactoring获取更多信息。

关于oracle - 我可以在不使用 where 子句的情况下创建自定义表达式吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3524101/

相关文章:

java - 远程服务器中的 DatabaseChangeRegistration

c# - NHibernate 演示文稿的一个很好的示例域是什么?

hibernate - 在一个 Hibernate 查询中更新多个列?

postgresql - Postgres - 选择没有表格但来自函数的数据的行号

sql - oracle数据库中的解码函数

database - 错误 : invalid identifier 00904. 00000 - “%s:外键约束的无效标识符

sql - 如何获取所有列,但必须在sql查询中仅按2列进行分组

c# - NHibernate 缓存问题

sql - Oracle "connect by prior"连同 "max() over partition by"删除重复的子树