阅读导航

1. TVF(-值行数Table-Valued Functions)

        a. 创建TVF

        b. 使用TVF的低性能T-SQL

        c. 使用临时表代替TVF

2. 标量函数

3. 替代标量函数

    1). 临时表

    2). 持久化确定的计算列

    3). 使用计划更新工作

        a. 创建标量函数

        b. 使用临时表替换标量函数

        c. 使用持久化确定的计算列

        d. 使用计划工作代替标量函数

上一篇介绍了关于“临时表、表变量和Union优化”这次转向关注定义函数——也就是表-值函数、标量函数。

UDF(用户定义函数,User defined Function)对于集中精力处理业务逻辑很方便,因为可以在UDF中指定一组业务逻辑,其中可以设计多个存储过程和一些特定的查询语句。但是,由于UDF对CPU的大量请求可能导致性能下降

1. TVF(-值行数Table-Valued Functions)

一般情况,当使用TVF与一个对象内联接,如果该对象没有索引将会导致TVF像索引扫描或表扫描一样做扫描操作。

作为一个选择,可以创建临时表,临时表上创建适当的聚集索引或非聚集索引。

详情如下:

  • 创建适当的临时表。
  • 根据T-SQL创建适当的聚集索引和非聚集索引。
  • 将TVF的数据插入到临时表中。
  • 用临时表和相关的列替换每一个TVF。
  • 在查询语句执行结束后,删除临时表。

                注意,临时表的性能提升是超过表参数,在上一篇博客中提到的,表参数不支持索引。

例子:

a. 创建TVF:

   1: use [MyDemo]

   2: go

   3: alter FUNCTION Dep_Salaries1

   4: (

   5: @empid int

   6: )

   7: RETURNS @table table

   8: (

   9: Department int,

  10: Salary_Max int,

  11: Salary_Min int

  12: )

  13: AS

  14: BEGIN

  15: declare @Department int = (select S.deptid from Employees s where s.empid=@empid)

  16: insert into @table

  17: SELECT S.deptid , max (Salary) , MIN(Salary) FROM Employees s inner join Departments T ON S.deptid =T.deptid group by S.deptid having S.deptid =@Department

  18: RETURN

  19: END

  20: GO

b. 使用TVF的低性能T-SQL:

   1: alter procedure Unperformant_SP1

   2: @empid int

   3: as

   4: begin

   5: select T.deptid as department_name , s.* from Dep_Salaries1 (@empid )S inner join Departments T ON S.Department =T.deptid

   6: end

c. 使用临时表代替TVF:

   1: go

   2: alter procedure Performant_SP1

   3: @empid int

   4: as

   5: begin

   6: create table #table

   7: (

   8: Department int,

   9: Salary_Max int,

  10: Salary_Min int

  11: )

  12: create clustered index #table_index1 on #table (Department)

  13: insert into #table select * from Dep_Salaries1 (@empid )

  14: select T.deptid as department_name , s.* from #table S inner join Departments T ON S.Department =T.deptid

  15: end

在使用具体的查询和数据时,还是应该进行必要的性能测试,发现最适合自己情况的解决方案。

2. 标量函数

标量函数,对于确定存储过程或特定查询语句的聚合值、累计值、差分值非常方便的,但是对性能是有损失的,尤其使用大数据,标量函数将执行每一个记录。

3. 替代标量函数

1). 临时表

使用临时表,但是这个解决方案有一点不同于TVF的情况,这里希望完全放弃标量函数并且也不去直接使用内部T-SQL代码。

2). 持久化确定的计算列

持久化确定的计算列值不是每次选择都重新计算该列,而只是在创建时计算一次。因此,这时可以添加不同的T-SQL语句提高性能,因为这样可以减少进程的开销。

这个功能可以通过下面步骤添加:

  • 增加一个新的计算列存储标量函数的结果。
  • 启用这个计算列的持久化功能。
  • 在列(不管是主键列还是包含列)上设置适当的索引。

但是要注意持久化功能还是有一些限制,如:

                   i. 计算列不应该使用任何其他记录的聚合功能。

                   ii. 计算列不应该使用调用外部系统过程的功能。

                   iii. 计算列不应该使用任何其他表的其他字段的功能。

                   iv. 计算列生成最好是使用系统提供的功能,例如:Convert、Cast、Replace等等,并且开发者不能创建UDF,因为UDF通常和该功能相矛盾。

这仅仅是适用于持久化的功能,但是可以添加计算列索引,应该通过确定计算数据的精确类型(如,INT、 Bigint、 DateTime和decimal)精确列的类型。如果数据类型不精确,可以添加这些列为索引的包含列的一部分,但不是主键列的一部分。

3). 使用计划更新工作

如果不可能使用持久化确定的计算列,可以创建普通列并同时创建计划更新工作,更新这些列的标量函数输出,然后用T-SQL代替标量函数并且在T-SQL中使用这些列。具体如下:

a. 创建标量函数:

   1: use [Workshops]

   2: go

   3: create FUNCTION Salary_Tax

   4: (

   5: @empid int

   6: )

   7: RETURNS float

   8: AS

   9: BEGIN

  10: declare @salary int = (select (S.salary-100) from Employees s where s.empid=@empid)

  11: RETURN @salary

  12: END

  13: GO

  14: --性能低些的标量函数

  15: Select empid ,dbo.Salary_Tax (empid) as 'SalaryWithTax' from Employees

b. 使用临时表替换标量函数:

   1: Create Table #temp (Empid int primary key clustered , Salary_Tax float)

   2: Create nonclustered index #temp_Index1 on #temp (Empid ) include (Salary_Tax )

   3: insert into #temp select Empid ,(Salary-100) as salary_Tax from Employees

   4: select * from #temp

c. 使用持久化确定的计算列:

   1: ALTER TABLE dbo.Employees ADD Salary_Tax AS Salary-100 PERSISTED

   2: Create nonclustered index Employees_Index1 on Employees (Empid, Salary_Tax )

   3: select empid ,Salary_Tax from Employees

d. 使用计划工作代替标量函数:

   1: ALTER TABLE dbo.Employees ADD Salary_Tax1 float, update_flag bit

   2: ALTER TABLE dbo.Employees ADD CONSTRAINT DF_Employees_update_flag DEFAULT 0 FOR update_flag

   3: Schedule the below DML update by an appropriate frequency according to your workload

   4: Update Employees set Salary_Tax1=Salary-100 WHERE UPDATE_Flag=0

   5: Then you can include the below select query within your stored procedure.

   6: select empid , Salary_Tax1 from Employees

上班时间到了!

期待下一篇吧!

任何的优化的不是绝对的,只有适应自己环境才是最好的,性能测试是必要。

SQL Server 性能优化之——T-SQL TVF和标量函数的更多相关文章

  1. 【SQL Server性能优化】运用SQL Server的全文检索来提高模糊匹配的效率

    原文:[SQL Server性能优化]运用SQL Server的全文检索来提高模糊匹配的效率 今天去面试,这个公司的业务需要模糊查询数据,之前他们通过mongodb来存储数据,但他们说会有丢数据的问题 ...

  2. SQL Server 性能优化(一)——简介

    原文:SQL Server 性能优化(一)--简介 一.性能优化的理由: 听起来有点多余,但是还是详细说一下: 1.节省成本:这里的成本不一定是钱,但是基本上可以变相认为是节省钱.性能上去了,本来要投 ...

  3. SQL Server 性能优化之——系统化方法提高性能

    SQL Server 性能优化之——系统化方法提高性能 阅读导航 1. 概述 2. 规范逻辑数据库设计 3. 使用高效索引设计 4. 使用高效的查询设计 5. 使用技术分析低性能 6. 总结 1. 概 ...

  4. SQL Server性能优化与管理的艺术 附件下载地址

    首先感谢读者们对鄙人的支持,购买了<SQL Server性能优化与管理的艺术>,由于之前出版社的一些疏忽,附件没有上传成功,再次本人深表歉意. 请需要下载附件的读者从下面链接下载,谢谢: ...

  5. SQL Server性能优化(6)查询语句建议

    1. 如果对数据不是工业级的访问(允许脏读),在select里添加 with(nolock) ID FROM Measure_heat WITH (nolock) 2. 限制结果集的数据量,如使用TO ...

  6. SQL SERVER性能优化综述

    SQL SERVER性能优化综述 一个系统的性能的提高,不单单是试运行或者维护阶段的性能调优的任务,也不单单是开发阶段的事情,而是在整个软件生命周期都需要注意,进行有效工作才能达到的.所以我希望按照软 ...

  7. SQL Server 性能优化之RML Utilities:快速入门(Quick Start)(1)

      SQL Server 性能优化之RML Utilities:快速入门(Quick Start)(1) 安装Quick Start工具 RML(Replay Markup Language)是MS ...

  8. 【SQL Server性能优化】删除大量数据的方法比较

    原文:[SQL Server性能优化]删除大量数据的方法比较 如果你要删除表中的大量数据,这个大量一般是指删除大于10%的记录,那么如何删除,效率才会比较高呢? 而如何删除才会对系统的影响相对较小呢? ...

  9. SQL Server 性能优化之——重复索引

    原文 http://www.cnblogs.com/BoyceYang/archive/2013/06/16/3139006.html 阅读导航 1. 概述 2. 什么是重复索引 3. 查找重复索引 ...

随机推荐

  1. EFM32外设模块—USART V1.00

    http://wenku.baidu.com/link?url=hx-pumUzdpS-AbD1OhEW11Jl6H8wex2DNsv4IcZwrgL-drwuUzZ6E1L64fCnAfdUOObK ...

  2. 关于SharePoint 2010中不能使用AjaxControlToolkit的解决办法

    因为项目中有一个需求需要使用calendar控件,而且样式要和Reporting Service中的尽量一致,搜索了很久发现还是微软的AjaxControlToolkit提供的CalendarExte ...

  3. Asp.net Mvc 身份验证、异常处理、权限验证(拦截器)实现代码

    本问主要介绍asp.net的身份验证机制及asp.net MVC拦截器在项目中的运用.现在让我们来模拟一个简单的流程:用户登录>权限验证>异常处理 1.用户登录 验证用户是否登录成功步骤直 ...

  4. HttpWebRequest类

    HttpWebRequest类与HttpRequest类的区别. HttpRequest类的对象用于服务器端,获取客户端传来的请求的信息,包括HTTP报文传送过来的所有信息.而HttpWebReque ...

  5. 深入研究C语言 第四篇

    这里更多探究的是指针的机制. 用debug对下面程序进行分析,记录每一条C语句运行后,相关内存单元的值. 程序a.c 注意理解指针机制 我们编写如下代码: 编译加载进debug查看: 我们先看其反汇编 ...

  6. android 常见面试题以及答案

    http://blog.csdn.net/bobo1808/article/details/6783344 1.    请描述下Activity的生命周期.2.    如果后台的Activity由于某 ...

  7. 【转】Beanstalkd 队列简易使用

    Beanstalkd一个高性能分布式内存队列系统   之前在微博上调查过大家正在使用的分布式内存队列系统,反馈有Memcacheq,Fqueue, RabbitMQ, Beanstalkd以及link ...

  8. Java学习笔记 04 类和对象

    一.类和对象的概念 类 >>具有相同属性和行为的一类实体 对象 >>实物存在的实体.通常会将对象划分为两个部分,即静态部分和动态部分.静态部分指的是不能动的部分,被称为属性,任 ...

  9. Xcode 升级成Xcode 8 版本以后,出现 Signing for "sqlite3--test" requires a development team. 问题的解决

    升级xcode到8版本以后,工程文件会出现以下提示 解决办法就是,点击Team,添加自己的appid,然后选择自己的appid即可, 注意: 这里不需要开发者账号,自己的 apple id,就可以”:

  10. IOS绘制渐变背景色折线图的一种尝试

    1.绘制折线图 上次在群里看到一个折线图划的很漂亮,自己想实现一个这样的 ,但是一直没什么头绪,不知道怎么做,就开始在网上查找划线,绘 制渐变色这一块的内容,用最笨的方式,自己尝试的写了一些,也没 有 ...