【发布时间】:2015-12-21 16:30:58
【问题描述】:
我想创建一个这样的存储库方法:
public List<SelectListItem> AllAsSelectListItems(
Expression<Func<T, string>> valueProperty,
Expression<Func<T, string>> textProperty,
string selectedValue = "")
{
// what goes here? I am having serious trouble with this bit!
}
这将使我可以这样称呼它:
List<SelectListItem> selectListItems = PersonRepository.AllAsSelectListItems(
m => m.ID,
m => m.Name,
selectedIDAsString
);
而且,selectedValue 参数为“1”,它应该产生如下结果:
List<SelectListItem>(){
{Value: "1", Text: "Ted", Selected: true},
{Value: "2", Text: "Sam", Selected: false},
{Value: "3", Text: "Tracy", Selected: false}
};
我在使用通用 AllAsSelectListItems() 方法时遇到问题。
到目前为止,您可以在下面的代码中看到我的尝试。但这并不理想。
我使用了硬编码字符串来使用T 属性填充SelectListItem 属性。我认为表达式树是解决方案,但我正在努力正确编码。
另外,分配ID 属性会破坏它,因为它是int 而不是string。
最后,我还在努力将selectedValue 参数与SelectListItem.Value 属性进行比较。
人员类
public class Person
{
public int ID {get;set;}
public string Name {get;set;}
}
控制器
public class PersonController : Controller
{
public IPersonRepository Repository {get;set;}
public PersonController(IPersonRepository repository)
{
Repository = repository;
}
public ActionResult SelectPerson(int selectedID)
{
string selectedIDAsString = selectedID.ToString();
var selectListItems = Repository.AllAsSelectListItems(
m => m.ID,
m => m.Name,
selectedIDAsString
);
return View(selectListItems);
}
}
存储库
public class PersonRepository : Repository
{
// various specialised methods
}
public class Repository<T> : IRepository<T> where T : DBEntity
{
private ApplicationDbContext db = null;
private DbSet<T> table = null;
public RepositoryBase()
{
this.db = new ApplicationDbContext();
table = db.Set<T>();
}
public RepositoryBase(ApplicationDbContext db)
{
this.db = db;
table = db.Set<T>();
}
protected virtual IQueryable<T> AllAsQueryable(
params Expression<Func<T, object>>[] includeExpressions)
{
return includeExpressions.Aggregate<Expression<Func<T, object>>, IQueryable<T>>
(table, (current, expression) => current.Include(expression));
}
public List<SelectListItem> AllAsSelectListItems(
Expression<Func<T, string>> valueProperty,
Expression<Func<T, string>> textProperty,
string selectedValue = "")
{
// temp hard coded values until we learn how to use the expression parameters properly
string valuePropertyHardCoded = "Name";
string textPropertyHardCoded = "Name";
Type currentType = typeof(T);
var itemParam = Expression.Parameter(currentType, "x");
var valueMember = Expression.PropertyOrField(itemParam, valuePropertyHardCoded);
var textMember = Expression.PropertyOrField(itemParam, textPropertyHardCoded);
var selector = Expression.MemberInit(Expression.New(typeof(SelectListItem)),
Expression.Bind(typeof(SelectListItem).GetMember("Value").Single(), valueMember),
Expression.Bind(typeof(SelectListItem).GetMember("Text").Single(), textMember)
);
var lambda = Expression.Lambda<Func<T, SelectListItem>>(
selector, itemParam);
return AllAsQueryable().Select(lambda.Compile()).ToList();
}
}
【问题讨论】:
-
我有点迷茫,AllAsSelectListItemsSpecifyProperties你是怎么写的AllAsSelectListItems?还是其他一些可行的方法?
-
抱歉,这是复制粘贴时的拼写错误。感谢您指出。应该是
AllAsSelectListItems()。现已更新。 -
string selectedValue = ""参数的目的是什么? -
与每个
SelectListItem.Value进行比较。如果它们相同,我们将SelectListItem.Selected标记为true。
标签: c# linq lambda linq-to-entities expression-trees