3

具体来说,我可以从视图中的当前数据库调用 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 根据环境(开发、测试、产品)连接到不同的数据库。

4

1 回答 1

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 数据库中的存储过程。这样做的缺点是在任何给定时间只能激活/重定向一个同义词。

于 2012-05-16T21:27:02.877 回答