说我有一个包含物品的清单
-17“屏幕
- 100GB高清
-10788火线
- 锁定电缆
- 监控器 - 鼠标
- 键盘
- USB
我想迭代一个列表,列出从A到c,从D到F等等的项目......
I want this to run for items starting from A to items starting C
@foreach (var item in Model.Items.OrderBy(i => i.Title))
{
// Code
}
I want this to run for items starting from D to items starting F
@foreach (var item in Model.Items.OrderBy(i => i.Title))
{
// Code
}
任何帮助?
答案 0 :(得分:4)
我首先要定义一个视图模型(一如既往):
public class MyViewModel
{
public string LetterRange { get; set; }
public string[] Titles { get; set; }
}
然后是一个控制器动作,它将转换从某处获取模型,然后将其映射到视图模型。
备注:在本例中,我将把模型和视图模型之间的映射代码放在控制器动作中,但通常这应该放在一个单独的映射层中。例如,如果您使用AutoMapper可能是个好地方。
所以:
public ActionResult Index()
{
// The model could be of any form and come from anywhere but
// the important thing is that at the end of the day you will have
// a list of titles here
var model = new[]
{
"17\" Screen",
"100GB HD",
"10788 Firewire",
"Lock Cable",
"Monitor",
"Mouse",
"Keyboard",
"USB"
};
// Now let's map this domain model into a view model
// that will be adapted to the requirements of our view.
// And the requirements of this view is to group the titles
// in ranges of 3 letters of the alphabet
var viewModel = Enumerable
.Range(65, 26)
.Select((letter, index) => new
{
Letter = ((char)letter).ToString(),
Index = index
})
.GroupBy(g => g.Index / 3)
.Select(g => g.Select(x => x.Letter).ToArray())
.Select(range => new MyViewModel
{
LetterRange = string.Format("{0}-{1}", range.First(), range.Last()),
Titles = model
.Where(item => item.Length > 0 && range.Contains(item.Substring(0, 1)))
.ToArray()
})
.ToArray();
// Let's add those titles that weren't starting with an alphabet letter
var other = new MyViewModel
{
LetterRange = "Other",
Titles = model.Where(item => !viewModel.Any(x => x.Titles.Contains(item))).ToArray()
};
// and merge them into the final view model
viewModel = new[] { other }.Concat(viewModel).ToArray();
return View(viewModel);
}
现在,相应视图中的所有内容都是根据要求显示标题:
@model MyViewModel[]
@foreach (var item in Model)
{
<h2>@item.LetterRange</h2>
foreach (var title in item.Titles)
{
<div>@title</div>
}
}
结果:
在将映射逻辑重构为映射层之后,相应的控制器操作可能如下所示:
public ActionResult Index()
{
// The model could be of any form and come from anywhere but
// the important thing is that at the end of the day you will have
// a list of titles here
DomainModel[] items = ...
// Now let's map this domain model into a view model
// that will be adapted to the requirements of our view.
var viewModel = Mapper.Map<IEnumerable<DomainModel>, IEnumerable<MyViewModel>>(items);
return View(viewModel);
}
清洁干燥。
答案 1 :(得分:1)
为什么不:
var fromAtoC = Model.Items.Where(x => x.Title != null && x.Title[0] >= 'A' && x.Title[0] <= 'C');
foreach(Model.Items m in fromAtoC)
{
//Do some stuff
}
var fromDtoF = Model.Items.Where(x => x.Title != null && x.Title[0] >= 'D' && x.Title[0] <= 'F');
for(Model.Items m in fromDtoF)
{
//Do some stuff
}
答案 2 :(得分:0)
试试这个:
Model.Items.Sort((x, y)=> x.Title.CompareTo(y.Title));
foreach (var item in Model.Items.Where(x => x.Title[0] >= 'A' && x.Title[0] <= 'C')
{
//code
}
答案 3 :(得分:0)
不确定这是最优雅的解决方案,但它会起作用。我会看是否能想到更好的东西......
var itemsAtoC = Model.Items.Where.Where(i => i.StartsWith("A") || i.StartsWith("B") || i.StartsWith("C"));
foreach(var item in itemsAtoC)
{
Console.Write(item);
}
Ugggh这并没有真正涉及数字......
还在想......