我可以从视图中动态调用存储过程吗?

时间:2012-05-16 21:15:38

标签: sql-server stored-procedures sql-server-2008-r2 dynamic-sql sql-view

具体来说,我可以在视图中从当前数据库调用proc。我已经知道openrowset hack了,所以这有用,例如:

create view MyView as
    select *
    from openrowset (
        'sqloledb',
        'server=(local);trusted_connection=yes;',
        'exec MyDatabase.dbo.MyStoredProcedure' -- Works fine
    )

但是我希望能够从当前的数据库中调用proc,而不需要对名称进行硬编码:

create view MyView as
    select *
    from openrowset (
        'sqloledb',
        'server=(local);trusted_connection=yes;',
        'exec ' + db_name() + '.dbo.MyStoredProcedure' -- Don't want to hard-code DB name
    )

不幸的是,这不起作用,因为openrowset需要文字字符串,而不是任何类型的变量。

无论出于安全性和性能考虑,是否有解决方法?它将使遗留系统的维护更加可忍受,因为此视图将调用的proc根据环境(dev,test,prod)连接到不同的数据库。

1 个答案:

答案 0 :(得分:2)

不,您不能在视图中使用动态SQL。如果只有三个不同的“环境”,则可以创建三个视图,和/或根据环境使用同义词。例如,您可以有三个视图(伪/修剪):

create view dbo.devMyView as
    select * ... 'exec Dev.dbo.MyStoredProcedure'
go
create view dbo.testMyView as
    select * ... 'exec Test.dbo.MyStoredProcedure'
go
create view dbo.prodMyView as
    select * ... 'exec Prod.dbo.MyStoredProcedure'

然后,您可以在代码中使用动态SQL来指定所需的视图,或者在想要模拟每个环境时可以删除并创建同义词,例如

DROP SYNONYM dbo.MyView;
GO
CREATE SYNONYM dbo.MyView FOR dbo.devMyView;

现在引用dbo.MyView的代码最终将调用dev数据库中的存储过程。这样做的缺点是在任何给定时间只能有一个同义词有效/重定向。