本文转自:http://diaosbook.com/Post/2012/9/21/linq-paging-in-entity-framework

我们知道,内存分页效率很低。并且,如果是WebForm的项目,页面上会有巨型ViewState,这必然不好。我自己博客用的是一个存储过程做的分页,用到现在都挺好,没有任何效率问题。后来想到,既然项目里有Entity Framework,那为什么不利用EF完成分页呢~

稍做研究之后发现,EF分页其实很简单。不过一样写文章了,光贴代码是不负责的,还是得稍微介绍一下相关的知识。

一、页数计算

关于分页的基本原理,网上有很多文章,我就不多叙述了。但我发现很多介绍分页的文章里,计算页码公式都掐掉了,广为流传的版本是:

totalPage = totalRecord / pageSize +
稍微推敲一下就会发现,这个公式在totalRecord和pageSize正好整除时,会多一页。比如10条记录,每页5条,应该是2页的,但结果是3。然而,不写+1又不行,因为要考虑到记录总数小于页尺寸的情况。现在,许多网站的分页都有这个问题。而我自己用的是下面这个公式,无论如何都不会掐掉的:
totalPage =(totalRecord + pageSize -)/ pageSize
注意,C#里的“/”是整除。所以这两个公式不是数学意义上的计算公式,呵呵。

二、分页存储过程的原理

(题外话:SQL Server 2012有了新的分页语法,本文只针对SQL Server 2008 R2及以往的版本)

任何分页存储过程的核心语句最后都会拼接为类似下面的这种SQL语句:

SELECT TOP(10)-- pageSize
*
FROM (
SELECT *,
ROW_NUMBER() OVER(ORDER BY [Temp].[PubDate] DESC) AS
[row_number]
FROM [dbo].[Post] AS [Temp]) AS [Temp]
WHERE [Temp].[row_number]>10--(pageIndex -1)* pageSize
ORDER BY
[Temp].[PubDate] DESC
这段SQL获取的是Post表,按PubDate逆序排列后,第二页的数据。

ROW_NUMBER()函数的作用是给查询结果生成行号,这是为了给后面的where语句用的。这个函数的具体介绍请看MSDN:http://msdn.microsoft.com/en-us/library/ms186734.aspx

三、LINQ to SQL分页

我们的目标很明确,就是最终要将LINQ语句翻译到与上面相符的SQL语句。在LINQ to SQL中,Enumerable.Take<TSource>最终会翻译为TOP(n),而Enumerable.Skip<TSource>和Take结合起来使用,就会翻译成ROW_NUMBER()…OVER…的语句。

所以,上面贴出的SQL代码的等价LINQ是:

Enumerable.OrderByDescending(p => p.PubDate).Skip().Take();
 

四、在Entity Framework中对某表分页

用EF对某表做完映射后,我们可以封装一个分页方法,比如:

private static List<Post> GetPostList(int pageIndex, int pageSize)
{
int startRow = (pageIndex - ) * pageSize;
using (var db = new EdiBlogEntities())
{
var query = db.Post.OrderByDescending(p => p.PubDate).Skip(startRow).Take(pageSize);
return query.ToList();
}
}
对于方法的返回类型,我选择List是因为调用它的程序不需要访问导航属性。如果程序中有访问导航属性的需要,就不能关闭当前的database context,应该要返回IEnumerable<T>类型。

测试代码:

static void Main(string[] args)
{
int pageIndex = ;
int pageSize = ;
int totalPage = (GetPostCount() + pageSize - ) / pageSize; Console.WriteLine("List of Posts. Page {0} / {1}, showing {2} rows per page.", pageIndex, totalPage, pageSize);
Console.WriteLine("---------------------------------------------------------");
foreach (var item in GetPostList(pageIndex, pageSize))
{
Console.WriteLine("{0}, {1}", item.Title, item.PubDate);
}
} private static int GetPostCount()
{
using (var db = new EdiBlogEntities())
{
return db.Post.Count();
}
}
 

运行结果:

和之前的存储过程分页是一致的:

[转]在Entity Framework中使用LINQ语句分页的更多相关文章

  1. 在Entity Framework 中执行T-sql语句

    从Entity Framework  4开始在ObjectContext对象上提供了2个方法可以直接执行SQL语句:ExecuteStoreQuery<T> 和 ExecuteStoreC ...

  2. Entity Framework中执行Sql语句

           如果想在EF框架中执行Sql语句,其实很简单,EF里面已经提供了相关的方法(此处使用的EF为EF4.1版本).        EF中提供了两个方法,一个是执行查询的Sql语句SqlQue ...

  3. 在Linq to sql 和 Entity framework 中使用lambda表达式实现left join

    在Linq to sql 和 Entity framework 中使用lambda表达式实现left join 我们知道lambda表达式在Linq to sql 和 Entity framework ...

  4. Entity framework 中Where、First、Count等查询函数使用时要注意

    在.Net开发中,Entity framework是微软ORM架构的最佳官方工具.我们可以使用Lambda表达式在Entity framework中DbSet<T>类上直接做查询(比如使用 ...

  5. Entity Framework中的几种加载方式

            在Entity Framework中有三种加载的方式,分别是延迟加载,自动加载和显示加载.下面用一个例子来说明:现在有两个表,一个是资料表(Reference),另外一个表是资料分类表 ...

  6. Entity Framework中DbContext结合TransactionScope提交事务的正确方式

    问: I would like know what is the best possible way to implement transactions with DBContext. In part ...

  7. Entity Framework 教程——Entity Framework中的实体类型

    Entity Framework中的实体类型 : 在之前的章节中我们介绍过从已有的数据库中创建EDM,它包含数据库中每个表所对应的实体.在EF 5.0/6.0中,存在POCO 实体和动态代理实体两种. ...

  8. 关于Entity Framework中的Attached报错相关解决方案的总结

    关于Entity Framework中的Attached报错的问题,我这里分为以下几种类型,每种类型我都给出相应的解决方案,希望能给大家带来一些的帮助,当然作为读者的您如果觉得有不同的意见或更好的方法 ...

  9. 关于Entity Framework中的Attached报错的完美解决方案终极版

    之前发表过一篇文章题为<关于Entity Framework中的Attached报错的完美解决方案>,那篇文章确实能解决单个实体在进行更新.删除时Attached的报错,注意我这里说的单个 ...

随机推荐

  1. 【JAVA、C++】LeetCode 011 Container With Most Water

    Given n non-negative integers a1, a2, ..., an, where each represents a point at coordinate (i, ai). ...

  2. August 1st, 2016, Week 32nd Monday

    Laughing is the most touching mask. 笑容是最动人的面具. I used to be very weclome in those I had met. And the ...

  3. win7下IIS配置MVC项目

    第一步:添加MVC程序映射 1.双击打开,如下图: 2. 点击界面右边操作中的:添加脚本映射 请求路径:* 可执行文件路径:C:\Windows\Microsoft.NET\Framework\v4. ...

  4. HTTP协议的几个概念

    http://www.rupeng.com/favicon.ico 可获取当前图标favicon.ico服务器是指定文件名,不能更改一.HTTP协议的几个概念1.连接(Connection):浏览器和 ...

  5. svn update 每更新一项就输出一行信息,使用首字符来报告执行的动作 这些字符的含义是:

    A 已添加 D 已删除 U 已更新 C 合并冲突 G 合并成功 例子: [root@ok 资料库]# svn ci -m "" Sending 资料库/简历 Transmittin ...

  6. Java 对时间和日期的相关处理

    1. 获取当前系统时间和日期并格式化输出 import java.util.Date; import java.text.SimpleDateFormat; public class NowStrin ...

  7. android之WakeLock机制浅析

    转自:http://blog.sina.com.cn/s/blog_4ad7c2540101n2k2.html 应用程序耗电的实质,是所启用的硬件在消耗电量.  手机的耗电单元 CPU: 应用处理器( ...

  8. iOS开发网络篇—JSON介绍

    一.什么是JSON JSON是一种轻量级的数据格式,一般用于数据交互 服务器返回给客户端的数据,一般都是JSON格式或者XML格式(文件下载除外) JSON的格式很像OC中的字典和数组 {" ...

  9. sql server 本地复制订阅 实现数据库服务器 读写分离

    再前段echosong 写了一遍关于mysql 数据同步实现业务读写分离的文章,今天咱们来看下SQL Server的复制订阅实现数据的读写分离 比起mysql的复制,SQL server 复制相对强大 ...

  10. js:语言精髓笔记5----语言分类

    计算模型:源于对计算过程的不同认识: 1.基于不同计算模型一般分为://教科书的一般分类 命令式语言: 函数式语言: 逻辑式语言: 面向对象程序设计语言: 2.基于程序本质分类:  //编程的经典法则 ...