(六)高效合理的使用LINQ

1、DataContext中的两个属性

为了能够使用DataContext进行数据提交,在DataContext进行数据查询和操作的过程中,内部会进行数据状态的保持和追踪,这会带来一些额外的开销。如果用户仅需要进行数据读取的话,可以通过将DataContext的ObjectTrackingEnabled属性设为false(默认值为true)来避免这些维护数据状态带来的额外开销。

此外,在DataContext中还有一个DefferedLoadingEnabled属性,用于控制数据实体之间的引用(对应到SharePoint列表中即查阅项)是否加载。如果将此属性设置为false(默认值为true),那么在返回结果之后,所有的查阅项引用都不会加载,值为null。因此,如果不需要使用其中查阅项类型内容的话,可以将此属性设置为false,可以进一步提高查询速度。

2、LINQ中的运算

在针对其他数据源进行查询的时候,LINQ可以进行多种运算,但是由于在SharePoint中,查询最终是使用CAML进行的,因此有些运算在LINQ to SharePoint中是不支持的,或者在执行效率上有所区别。

高效查询(Efficient Queries)

在LINQ to SharePoint中,凡是可以直接被转换成CAML查询的LINQ查询,都被称为高效查询。系统会将CAML查询的结果直接转换成相应的数据实体。

不支持的查询

在LINQ中,不支持查询嵌套,即将一个查询的结果作为另外一个查询的查询条件(在传统的T-SQL中,也就是select语句的嵌套)。

此外,在LINQ to SharePoint中,对两个查询结果的集合操作也是不被支持的,例如下面的程序会抛出异常:

   1: var orders1 = from o in ctx.InteralOrder

   2:               select o;

   3: var orders2 = from o in ctx.ExternalOrder

   4:               select o;

   5: var total = orders.Union(orders2);  // Exception Here!

但是通过使用ToList方法将结果转换成List之后,就可以进行集合运算了:

   1: var orders1 = from o in ctx.InteralOrder

   2:               select o;

   3: var orders2 = from o in ctx.ExternalOrder

   4:               select o;

   5: var total = orders.ToList().Union(orders2.ToList());  //OK Now.

半高效查询(Semi-Efficient Queries)

在有些情况下,查询条件可以被直接转换成CAML进行,但是在返回结果的时候,却无法转换成相应的CAML。考虑下面两个例子:

   1: // 查询1:

   2: var chapters = from chp in ctx.Chapters

   3:                where chp.Writer == "Erucy"

   4:                select new {chp.Name, chp.WordCount};

   5:  

   6: // 查询2:

   7: var chapters = from chp in ctx.Chapters

   8:                where chp.Writer == "Erucy"

   9:                select new {chp.Name, Pay = chp.WordCount * 0.05};

在这两个查询中,查询条件可以直接转换成:

   1: <Where>

   2:   <Eq>

   3:     <FieldRef Name='Writer'/>

   4:     <Value Type='Text'>Erucy</Value>

   5:   </Eq>

   6: </Where>

在查询1中,查询结果可以直接用CAML的ViewFields来表示:

   1: <ViewFields>

   2:   <FieldRef Name='Name' />

   3:   <FieldRef Name='WordCount' />

   4: </ViewFields>

于是,LINQ to SharePoint便可以直接将返回结果转换成相应的数据实体对象。

但是在查询2中,返回结果的“WordCount * 0.05”部分由于涉及到了运算,无法直接使用CAML进行描述,因此LINQ to SharePoint在返回这类查询结果的时候,需要在CAML查询结果的基础之上,再利用LINQ to Objects进行一次结果转换,但是在转换过程中不需要再涉及查询操作。这种查询实际上相当于如下代码:

   1: var camlResult = from chp in ctx.Chapters

   2:                  where chp.Writer == "Erucy"

   3:                  select new {chp.Name, chp.WordCount};

   4: var chapters = from chp in camlResult

   5:                select new {chp.Name, Pay = chp.WordCount * 0.05};

除去这种简单运算之外,在返回结果的时候,LINQ查询中所使用到的Distinct、Average、Sum、Max、Min等操作都属于这一类型。由于这种查询需要两个阶段来完成,因此也被称为两阶段查询(Two-Stage Queries)。

低效查询(Inefficient Queries)

在上面的例子中,仅是在返回结果的时候需要进行额外的转换,而查询本身仍然可以使用CAML进行。而在有些情况下,查询完全无法转换成CAML格式,而这个时候,LINQ to SharePoint会首先获取到列表中所有内容,将其转换为类型,在此基础上再进行LINQ to Objects的查询,使得SharePoint数据查询完全退化成内存中的数据查询。这种查询也需要使用两个阶段来完成操作,但由于需要获取列表中的全部数据,效率较低。

这种查询可以实现一些CAML无法实现的功能。在CAML查询中,只支持字段栏与具体数值之间的比较运算,但不支持字段栏与字段栏之间的运算,例如下面的这个查询:

   1: var okOrders = from order in ctx.Orders

   2:                where order.Sale > order.Quota

   3:                select order;

在实际查询的时候,由于CAML不支持在两个字段之间进行比较,因此这个查询最终会退化为在内存中的对象查询:

   1: var okOrders = from order in ctx.Orders.ToList()

   2:                where order.Sale > order.Quota

   3:                select order;

(ToList方法会将EntityList退化成普通的List,将Orders列表中的所有数据都载入内存,再进行查询)

3、LINQ to SharePoint所不支持的场景

虽然LINQ to SharePoint在进行数据查询、数据操作的时候提供了无比的便利性,但是并非是所有场景都可以使用LINQ。除去上面提到的不支持的运算外,下面列出了几种常见的默认情况下不能使用LINQ to SharePoint的场景:

  • 无法查询外部列表(External List)和外部内容类型(External Content Type)。SPMetal工具不会为这些类型的数据生成数据实体类。不过查询外部数据可以使用相应的其他类型的LINQ Provider,如LINQ to SQL、LINQ to XML等。
  • 在Web上只能查询当前网站的内容。由于在DataContext中某些地方用到了当前的上下文信息(HttpContext),因此在Web上(比如Web部件),LINQ to SharePoint默认情况下只能对当前程序所在网站的内容进行查询。
  • 不支持匿名网站。LINQ to SharePoint目前仍不支持网站的匿名访问(即使网站开启了匿名访问支持)。

SharePoint服务器端对象模型 之 使用LINQ进行数据访问操作(Part 4)的更多相关文章

  1. SharePoint 服务器端对象模型 之 使用LINQ进行数据访问操作(Part 2)

    (四)使用LINQ进行列表查询 在生成实体类之后,就可以利用LINQ的强大查询能力进行SharePoint列表数据的查询了.在传统SharePoint对象模型编程中,需要首先获取网站对象,再进行其他操 ...

  2. SharePoint服务器端对象模型 之 使用CAML进展数据查询

    SharePoint服务器端对象模型 之 使用CAML进行数据查询 一.概述 在SharePoint的开发应用中,查询是非常常用的一种手段,根据某些筛选.排序条件,获得某个列表或者某一些列表中相应的列 ...

  3. SharePoint服务器端对象模型 之 使用CAML进行数据查询

    (一)概述 在SharePoint的开发应用中,查询是非常常用的一种手段,根据某些筛选.排序条件,获得某个列表或者某一些列表中相应的列表条目的集合. 除去列表上的查询之外,在SharePoint中还大 ...

  4. SharePoint服务器端对象模型 之 使用CAML进行数据查询(Part 4)

    (五)列表查询中的阈值限制 在之前版本的SharePoint 中,如果在查询的时候没有指定返回数目,那么SharePoint将会查找该列表中所有的条目,这可能会造成在SQL表中需要返回大量的条目,极大 ...

  5. SharePoint服务器端对象模型 之 使用CAML进行数据查询(Part 2)

    (三)使用SPQuery进行列表查询 1.概述 列表查询主要是指在一个指定的列表(或文档库)中按照某些筛选.排序条件进行查询.列表查询主要使用SPQuery对象,以及SPList的GetItems方法 ...

  6. SharePoint服务器端对象模型 之 使用CAML进行数据查询(Part 3)

    (四)使用SPSiteDataQuery进行多列表查询 1.概述 前面介绍的列表查询有很多优势,但是它的一个缺点就是一次只能在一个列表中进行查询,在SharePoint中,提供了一个跨网站.跨列表查询 ...

  7. SharePoint服务器端对象模型 完结

    整个系列已完结,大概看了一眼,平均阅读量不到200.估计也没什么人看了,而且服务器端对象模型除了在某些企业开发中会用到,从2013时代开始其实已经不是SharePoint开发的最佳选择了.不过既然已经 ...

  8. SharePoint服务器端对象模型 之 序言

    对于刚刚开始接触SharePoint的开发人员,即使之前有较为丰富的ASP.NET开发经验,在面对SharePoint时候可能也很难找到入手的方向.对于任何一种开发平台而言,学习开发的过程大致会包括: ...

  9. SharePoint服务器端对象模型 之 对象模型概述(Part 2)

    (三)Url 作为一个B/S体系,在SharePoint的属性.方法参数和返回值中,大量的涉及到了Url,总的来说,涉及到的Url可以分为如下四类: 绝对路径:完整的Url,包含了协议头(http或h ...

随机推荐

  1. 在 SELECT 查询中使用集运算符

    在 SELECT 查询中使用集运算符,可以将来自两个或多个查询的结果合并到单个结果集中. 在进行集运算之前,请确保: (1)所有输入集合中,列数和列的顺序必须相同. (2)对应的列中,数据类型必须兼容 ...

  2. object-c输出对象

    有时候在xcode里打断点很不准,看到对象总是nil,还是用打log比较靠谱: NSLog(@"obj info:%@",obj);

  3. 《MVC +EasyUI 》——表单的提交

        之前用AJax给Controller传递參数,然后再调用服务端的方法对数据库进行更改,今天碰到一个新的方法,就是表单的提交.这样能够省去AJax穿參.当表单提交后.我们能够获取表单上控件中的值 ...

  4. Unity3D发布安卓报错permisson denied的解决

    没有打开SD卡的写入权限 unity提供了sd卡权限修改的参数:

  5. 多线程-BlockingQueue,Array[Linked]BlockingQueue,DelayQueue,PriorityBlockingQueue,SynchronousQueue

    阻塞场景 BlockingQueue阻塞队列,阻塞的情况主要有如下2种: 1. 当队列满了,进行入队操作阻塞 2. 当队列空了,进行出队操作阻塞 阻塞队列主要用在生产者/消费者模式中,下图展示了一个线 ...

  6. (1)Smali系列学习之Smali函数调用语句分析

    一.函数调用smali中的函数和成员变量也分为两种,分别为 direct 和 virtual.两者的区别如下: 1.direct method 是指调用private方法.2.virtual meth ...

  7. 3.Queues(队列)

    一.概述 C++队列是一种容器适配器,它给予程序员一种先进先出(FIFO)的数据结构,与stack刚好相反. 二.常用API back() 返回最后一个元素 empty() 如果队列空则返回真 fro ...

  8. Oracle之函数concat、lpad

    一.引言 程序测试需要生成大量的测试数据,且测试数据有主键,主键自增,于是决定用存储过程来实现,经过半天的查资料终于完成了,记录之,学习之 二.存储过程 格式: CREATE PROCEDURE re ...

  9. CCNA2.0笔记_OSPF v2

    OSPF(开放最短路径优先)协议概述: - 链路状态路由协议 - 无类路由协议 - 要点:RouterID.区域ID - 触发更新 .以传播 LSA 代替路由表更新 - 快速响应变更(比距离矢量路由协 ...

  10. CCNA2.0笔记_EIGRP

    EIGRP特征: •高级距离矢量路由协议 •快速收敛——路由条目不过期,拥有备份路由 •负载均衡 •无类路由 -支持 VLSM 和不连续子网,可关闭自动汇总(建议关闭) •占用带宽小 -触发更新(当拓 ...