设为首页收藏本站

全球主机交流论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

扫一扫,访问微社区

搜索
热搜: discuz
查看: 359|回复: 0
打印 上一主题 下一主题

SQL Server 2008:表值参数的创建和使用

[复制链接]
  • TA的每日心情
    无聊
    2017-9-11 09:33
  • 签到天数: 183 天

    [LV.7]常住居民III

    跳转到指定楼层
    楼主
    发表于 2015-6-15 17:22:53 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式

    表值参数(Table-valued parameter)是SQL Server 2008的一个新特性,在以前的版本中,没有办法把表变量当作一个参数传递给存储过程。微软在SQL Server2008中引入了表值参数的特性,可以实现这项功能。

      表值参数有两大优点:一是它不需要为初始的数据加锁,二是它不会导致语句重新编译。

      表值参数的创建和使用包括以下步骤:

      1) 创建表类型

      2) 创建一个可将表类型作为参数来接受的存储过程或函数

      3) 创建表变量并插入数据

      4) 调用该存储过程和函数,并将表变量作为参数传递。

      下面,我们来一步步分解这个创建和使用的过程。首先,我们用以下的DDL SQL语句来创建一个名为“TestDB”的测试数据库:


    USE [master]
    GO
    IF EXISTS (SELECT name FROM sys.databases WHERE name = N'TestDB')
    DROP DATABASE TestDB
    GO
    Create database TestDB
    go

      接下来我们使用以下的DDL SQL语句来创建一个名为TestLocationTable的表:

      然后,使用以下的DML SQL语句将数据添加到我们上面创建的表中:



    USE [TestDB]
    GO
    insert into TestLocationTable ( Id, shortname, Name) select 1, 'NA1', 'NewYork'
    insert into TestLocationTable ( Id, shortname, Name) select 2, 'NA2', 'NewYork'
    insert into TestLocationTable ( Id, shortname, Name) select 3, 'NA3', 'NewYork'
    insert into TestLocationTable ( Id, shortname, Name) select 4, 'EU1', 'London'
    insert into TestLocationTable ( Id, shortname, Name) select 5, 'EU2', 'London'
    insert into TestLocationTable ( Id, shortname, Name) select 6, 'AS1', 'Tokyo'
    insert into TestLocationTable ( Id, shortname, Name) select 7, 'AS2', 'HongKong'
    go


      下一步,我们要创建一个和TestLocationTable表具有相似表结构的表类型(TABLE TYPE),语句如下:

      紧接着,我们要创建一个可以将表类型作为一个参数来接受的存储过程,使用的语句如下:


    USE [TestDB]
    GO
    IF EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[dbo].
    [usp_InsertProdLocation]') AND type in (N'P', N'PC'))
    DROP PROCEDURE [dbo].[usp_selectProdLocation]
    GO
    CREATE PROCEDURE usp_InsertProdLocation
    @TVP OfficeLocation_Tabetype READONLY
    AS
    SET NOCOUNT ON
    INSERT INTO TestLocationTable Select ID, shortname, name from @TVP
    where convert(varchar(10),id)+shortname+name not in (select
    convert(varchar(10),id)+shortname+name from TestLocationTable)
    GO

      这个存储过程将表变量作为导入值接收,并且只插入TestLocationTable中没有的数据。现在,我们可以尝试创建一个表变量,并执行上面创建的存储过程usp_InsertProdLocation,语句如下:


    use TestDB
    go
    DECLARE @TV AS [OfficeLocation_Tabetype]
    INSERT INTO @TV (Id, Shortname, Name) SELECT 12, 'ME1', 'Dubai'
    INSERT INTO @TV (Id, Shortname, Name) SELECT 13, 'ME2', 'Tehran'
    INSERT INTO @TV (Id, Shortname, Name) SELECT 17, 'EA1', 'Bombay'
    INSERT INTO @TV (Id, Shortname, Name) SELECT 18, 'EA2', 'Karachi'
    INSERT INTO @TV (Id, Shortname, Name) SELECT 3, 'NA3', 'NewYork'
    INSERT INTO @TV (Id, Shortname, Name) SELECT 4, 'EU1', 'London'
    exec usp_InsertProdLocation @TV
    go

      这时候,我们可以使用以下的TSQL语句从表TestLocationTable查询所有的数据:


    use TestDB
    go
    select * from TestLocationTable
    go

      查询结果如下所示:


    Id, shortname, name
    1, NA1, NewYork
    2, NA2, NewYork
    3, NA3, NewYork
    4, EU1, London
    5, EU2, London
    6, AS1, Tokyo
    7, AS2, HongKong
    12, ME1, Dubai
    13, ME2, Tehran
    17, EA1, Bombay
    18, EA2, Karachi
    (11 row(s) affected)

      从返回的结果,我们可以看到存储过程usp_InsertProdLocation 插入了表变量@TV中和表TestLocationTable所有不匹配的行。

      我们还可以将表变量传递给一个函数。下面我们创建一个简单的函数,语句如下:

      现在,我们通过创建一个表变量并将该变量作为一个参数传递给已创建的函数以调用该函数,语句如下:


     USE [TestDB]
    GO
    DECLARE @TV AS [OfficeLocation_Tabetype]
    INSERT INTO @TV (Id, Shortname, Name) SELECT 12,'ME1','Dubai'
    INSERT INTO @TV (Id, Shortname, Name) SELECT 13,'ME2','Tehran'
    INSERT INTO @TV (Id, Shortname, Name) SELECT 17,'EA1','Bombay'
    INSERT INTO @TV (Id, Shortname, Name) SELECT 18,'EA2','Karachi'
    INSERT INTO @TV (Id, Shortname, Name) SELECT 3,'NA3','NewYork'
    INSERT INTO @TV (Id, Shortname, Name) SELECT 4,'EU1','London'
    select dbo.myfunction(@TV)
    go

      执行结果如下:


     (1 row(s) affected)
    (1 row(s) affected)
    (1 row(s) affected)
    (1 row(s) affected)
    (1 row(s) affected)
    (1 row(s) affected)
    -----------
    6

      注:上面所演示的脚本都是在SQL Server 2008 CTP6版本上进行编写并经过测试的。


    分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友 微信微信
    收藏收藏
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|小黑屋|Archiver|手机版|中国U网    

    GMT+8, 2024-5-19 17:04 , Processed in 0.068046 second(s), 21 queries .

    Powered by Discuz! X3.2

    © 2001-2013 Comsenz Inc.

    快速回复 返回顶部 返回列表