我必须创建一个方法,用于从具有指定类型的集合中选择firts属性.
我已经创建了这样的方法(为了简洁,我删除了一些部分):
public static IQueryable<TResult> SelectFirstPropertyWithType<T, TResult>(this IQueryable<T> source)
{
// Get the first property which has the TResult type
var propertyName = typeof(T).GetProperties()
.Where(x => x.PropertyType == typeof(TResult))
.Select(x => x.Name)
.FirstOrDefault();
var parameter = Expression.Parameter(typeof(T));
var body = Expression.Convert(Expression.PropertyOrField(parameter, propertyName), typeof(TResult));
var expression = Expression.Lambda<Func<T, TResult>>(body, parameter);
return source.Select(expression);
}
我可以将此方法称为:
List<Person> personList = new List<Person>();
// .. initialize personList
personList.AsQueryable()
.SelectFirstPropertyWithType<Person, int>()
.ToList();
一切正常.
但是,我不想将第一个参数类型设置为Person,因为编译器可以从集合的源中推断出这个参数类型.有没有办法像这样调用这个方法:
.SelectFirstPropertyWithType< INT>()
问题是我的方法中需要T参数,我不想在运行时使用反射创建Func.
谢谢.
解决方法:
C#Generics不允许您指定类型参数的子集.这是全有或全无.
解决这个问题的方法是编写一个流畅的界面.您将此操作分解为一系列方法.
public class FirstPropertyWithTypeSelector<T>
{
private readonly IQueryable<T> _source;
public FirstPropertyWithTypeSelector(IQueryable<T> source)
{
_source = source;
}
public IQueryable<TResult> OfType<TResult>()
{
// Get the first property which has the TResult type
var propertyName = typeof(T).GetProperties()
.Where(x => x.PropertyType == typeof(TResult))
.Select(x => x.Name)
.FirstOrDefault();
var parameter = Expression.Parameter(typeof(T));
var body = Expression.Convert(Expression.PropertyOrField(parameter, propertyName), typeof(TResult));
var expression = Expression.Lambda<Func<T, TResult>>(body, parameter);
return _source.Select(expression);
}
}
public static FirstPropertyWithTypeSelector<T> SelectFirstProperty(this IQueryable<T> source)
{
return new FirstPropertyWithTypeSelector<T>(source);
}
现在你可以打电话:
personList.AsQueryable()
.SelectFirstProperty().OfType<int>()
.ToList();