我已经在 LINQ To SQL 中使用 .Skip() 和 .Take() 扩展方法有一段时间了,没有任何问题,但在我使用它们的所有情况下,它始终用于单个表 - 例如:
database.Users.Select(c => c).Skip(10).Take(10);
我的问题是我现在正在从多个表中投影一组结果,并且我想在整个集合上进行分页(并且仍然可以在数据库中获得分页的好处)。
我的实体模型如下所示:
一个活动[有很多]组,一个组[有很多]联系人
这是通过数据库中的关系建模的,例如
活动 -> CampaignToGroupMapping -> 组 -> GroupToContactMapping -> 联系人
我需要生成一个包含活动详细信息的数据结构,以及通过 CampaignToGroupMapping 与活动关联的每个联系人的列表,即
Campaign
CampaignName
CampaignFrom
CampaignDate
Recipients
Recipient 1
Recipient 2
Recipient n...
我曾尝试使用 .SelectMany 编写一个 LINQ 查询,将每个组中的一组联系人投影到一个线性数据集中,希望我可以从中 .Skip() .Take() 。
我的尝试是:
var schedule = (from c in database.Campaigns
where c.ID == highestPriority.CampaignID
select new PieceOfCampaignSchedule
{
ID = c.ID,
UserID = c.UserID,
Name = c.Name,
Recipients = c.CampaignGroupsMappings.SelectMany(d => d.ContactGroup.ContactGroupMappings.Select(e => new ContactData() { /*Contact Data*/ }).Skip(c.TotalSent).Take(totalRequired)).ToList()
}).SingleOrDefault();
问题是分页(关于 Skip() 和 Take())发生在每个组,而不是整个数据集。
这意味着,如果我为参数totalRequired (传递给 .Take())使用值 200,并且我有 3 个与此活动相关联的组,则每组将需要 200 - 而不是从与活动。
在 SQL 中,我可以通过如下查询来实现:
select * from
(
select [t1].EmailAddress, ROW_NUMBER() over(order by CampaignID desc) as [RowNumber] from contacts as [t1]
inner join contactgroupmapping as [t2] on [t1].ID = [t2].ContactID
inner join campaigngroupsmapping as [t3] on [t3].ContactGroupID = [t2].GroupID
where [t3].CampaignID = @HighestPriorityCampaignID
) as [Results] where [Results].[RowNumber] between 500 and 3000
使用此查询,我正在对与特定活动相关联的每个组的联系人组合集进行分页。所以我的问题是,如何使用 LINQ To SQL 语法来实现这一点?