NHibernate QueryByExample仅包含某些属性

时间:2011-06-07 10:21:19

标签: c# nhibernate query-by-example

我创建了一个自定义属性选择器来接受构造函数中的数组,以说明搜索中应包含哪些属性。只要没有组件类型,该方法就可以正常工作,但我该如何处理?这是一个例子:

public class Customer
{
    public virtual int Id { get; private set; }
    public virtual Name Name { get; set; }
    public virtual bool isPreferred { get; set; }


    //...etc
}

public class Name
{
        public string Title { get; set; }
        public string Firstname { get; set; }
        public string Lastname { get; set; }
        public string Fullname { get; }
}


public class CustomerPropertySelector : Example.IPropertySelector
    {
        private string[] _propertiesToInclude = { };

        public CustomerPropertySelector(string[] propertiesToInclude)
        {
            this._propertiesToInclude = propertiesToInclude;
        }

        public bool Include(object propertyValue, String propertyName, NHibernate.Type.IType type)
        {
            //...Checking for null and zeros etc, excluded for brevity

            if (!_propertiesToInclude.Contains(propertyName))
                return false;

            return true;
        }
   }

我希望能够按名字搜索,但不一定是最后一次。但是,属性名称是Name,因此名字和姓氏似乎都是同一属性的一部分,而Name.Firstname这个通常用作标准的东西似乎在这里不起作用。最好的方法是什么?

示例:

Customer exampleCust = new Customer(FirstName: "Owen");
IList<Customer> matchedCustomers = _custRepo.GetByExample(exampleCust, new string[] { "Name.FirstName" });

鉴于db中有2个客户,只有一个名为“Owen”,但都有isPreferred = false,我希望我的查询只返回第一个。标准QBE将根据isPreferred属性返回两者。

SOLUTION:

感谢您的回答,解决方案主要是基于atalmitchconnors的回答,但是如果没有Mark Perry的答案,我也无法做到。

诀窍是要意识到,我实际上想要排除Name.FirstName,而不是包含Name.LastName属性,因为QBE只允许我们排除属性。我使用了一个改编自atalmitchconnors答案的方法来帮助我确定属性的完全限定名称。这是工作代码:

public IList<T> GetByExample(T exampleInstance, params string[] propertiesToInclude)
{
    ICriteria criteria = _session.CreateCriteria(typeof(T));
    Example example = Example.Create(exampleInstance);

    var props = typeof(T).GetProperties();
    foreach (var prop in props)
    {
        var childProperties = GetChildProperties(prop);
        foreach (var c in childProperties)
        {
            if (!propertiesToInclude.Contains(c))
                example.ExcludeProperty(c);
        }
    }
    criteria.Add(example);

    return criteria.List<T>();
}

private IEnumerable<string> GetChildProperties(System.Reflection.PropertyInfo property)
{
    var builtInTypes = new List<Type> { typeof(bool), typeof(byte), typeof(sbyte), typeof(char), 
        typeof(decimal), typeof(double), typeof(float), typeof(int), typeof(uint), typeof(long), 
        typeof(ulong), typeof(object), typeof(short), typeof(ushort), typeof(string), typeof(DateTime) };

    List<string> propertyNames = new List<string>();
    if (!builtInTypes.Contains(property.PropertyType) && !property.PropertyType.IsGenericType)
    {
        foreach (var subprop in property.PropertyType.GetProperties())
        {
            var childNames = GetChildProperties(subprop);
            propertyNames = propertyNames.Union(childNames.Select(r => property.Name + "." + r)).ToList();
        }
    }
    else
        propertyNames.Add(property.Name);

    return propertyNames;
}

我不确定确定属性是否为组件类的最佳方法,非常欢迎任何有关如何改进代码的建议。

2 个答案:

答案 0 :(得分:1)

以下代码将替换您用于填充propertiesToInclude的逻辑。我将它从一个数组更改为一个列表,所以我可以使用Add方法,因为我很懒,但我认为你得到了图片。这仅适用于一个子级别的属性。对于n级,你需要递归。

        List<string> _propertiesToInclude = new List<string>();

        Type t;
        var props = t.GetProperties();
        foreach (var prop in props)
        {
            if (prop.PropertyType.IsClass)
                foreach (var subprop in prop.PropertyType.GetProperties())
                    _propertiesToInclude.Add(string.Format("{0}.{1}", prop.Name, subprop.Name));
            else
                _propertiesToInclude.Add(prop.Name);
        }

答案 1 :(得分:1)

我认为我有一些东西,但再次阅读你的问题,你想知道为什么QBE NHibernate代码不适用于组件属性。

我认为你需要为Name部分创建一个子标准查询。

也许是这样的:

public IList<Customer> GetByExample(Customer customer, string[] propertiesToExclude){
    Example customerQuery = Example.Create(customer);
    Criteria nameCriteria = customerQuery.CreateCriteria<Name>();
    nameCriteria.Add(Example.create(customer.Name));
    propertiesToExclude.ForEach(x=> customerQuery.ExcludeProperty(x));
    propertiesToExclude.ForEach(x=> nameCriteria.ExcludeProperty(x));
    return customerQuery.list();
}

这是NHibernate测试项目的一个示例,它显示了如何排除组件属性。

[Test]
public void TestExcludingQBE()
{
        using (ISession s = OpenSession())
        using (ITransaction t = s.BeginTransaction())
        {
            Componentizable master = GetMaster("hibernate", null, "ope%");
            ICriteria crit = s.CreateCriteria(typeof(Componentizable));
            Example ex = Example.Create(master).EnableLike()
                .ExcludeProperty("Component.SubComponent");
            crit.Add(ex);
            IList result = crit.List();
            Assert.IsNotNull(result);
            Assert.AreEqual(3, result.Count);

            master = GetMaster("hibernate", "ORM tool", "fake stuff");
            crit = s.CreateCriteria(typeof(Componentizable));
            ex = Example.Create(master).EnableLike()
                .ExcludeProperty("Component.SubComponent.SubName1");
            crit.Add(ex);
            result = crit.List();
            Assert.IsNotNull(result);
            Assert.AreEqual(1, result.Count);
            t.Commit();
        }
    }

Source code link