生成 SelectListItem 列表的 IQueryable 扩展方法

IQueryable extension method to generate a SelectListItem list

我正在使用存储库 returning IQueryable 对象。

因此,我为 return 创建了一个扩展方法 SelectListItem 用于我的视图的列表:

public static List<SelectListItem> ToSelectList<T>(this IQueryable<T> query, Func<T, string> value, Func<T, string> text)
{
    return query.Select(x => new SelectListItem() { Text = text(x), Value = value(x) }).ToList();
}

我这样称呼:

var query = Repository<MyClass>.GetAll(); //Return MyClass IQueryable 
var test = query.ToSelectList(x => x.Property1, x => x.Property2);

效果很好(return 正确的列表),但是 SQL 从 MyClass 生成了 select 所有 属性。我希望在 db 上只查询 Property1 和 Property2。

我错过了什么?

您可能必须 select 一个只有您需要的属性的新匿名对象,然后执行 select 创建您的 SelectListItem

类似

return query.Select(p => new { value = value(p), text = text(p) ).Select(p => new SelectListItem(...) ;

在结果 x 从数据库中检索到结果后调用传入的委托,这就是返回所有字段的原因。您可以改为传入表达式,然后将它们组合成一个表达式:

public static List<SelectListItem> ToSelectList<T>(this IQueryable<T> query, Expression<Func<T, object>> value, Expression<Func<T, string>> text)
{
    var valueSrc = getProperty(value);
    var textSrc  = getProperty(text);    
    var x = Expression.Parameter(typeof(T),"x");
    var type = typeof(SelectListItem);
    var textProp = type.GetProperty("Text");
    var valueProp = type.GetProperty("Value");

    // x.valueSrc
    Expression valueExpr = Expression.Property(x,valueSrc);

    // (x.valueSrc).ToString()
    if (valueExpr.Type == typeof(int))
    {
         var toStringMethod = typeof(int).GetMethod("ToString",Type.EmptyTypes);
         valueExpr = Expression.Call(valueExpr,toStringMethod);
    }

    //x => new SelectListItem { Text = x.textSrc, Value = valueExpr }
    var lambda = Expression.Lambda<Func<T,SelectListItem>>(
                    Expression.MemberInit(
                        Expression.New(type),
                        Expression.Bind(textProp,Expression.Property(x,textSrc)),
                        Expression.Bind(valueProp,valueExpr)),
                    x);

    return query.Select(lambda).ToList();
}

static PropertyInfo getProperty(LambdaExpression exp)
{
    var body = exp.Body;

    //x => (object) x.Property
    var ue = body as UnaryExpression;
    if (ue != null)
        body = ue.Operand;

    return (PropertyInfo) ((MemberExpression) body).Member;        
}