ChatGPT解决这个技术问题 Extra ChatGPT

如何从存储过程中选择

我有一个返回行的存储过程:

CREATE PROCEDURE MyProc
AS
BEGIN
    SELECT * FROM MyTable
END

我的实际过程有点复杂,这就是为什么需要存储过程。

是否可以通过调用此过程来选择输出?

就像是:

SELECT * FROM (EXEC MyProc) AS TEMP

我需要使用 SELECT TOP XROW_NUMBER 和附加的 WHERE 子句来分页我的数据,并且我真的不想将这些值作为参数传递。

我不确定您打算在这里做什么,因为当您执行该过程时,您正在取回这些行。您是否想在 SELECT 语句中执行该过程,以便将其绑定到可分页对象?
您不想将值作为参数传递是否有特殊原因?按照您建议的方式执行此操作有点效率低下-您将选择比您需要的更多的数据,然后不会全部使用。
如果有人想将 sp 输出插入到 TABLE 变量中,则您的 sp 中有很多列,请按 Ctrl+T 将结果输出为文本,然后复制第一列行并从那里删除多余的空格,您将得到您的列名很容易。要返回网格输出,请按 Ctrl+D
注意所有进入这里的人: atm 下面只提供了大约三个答案:1. 使用函数(接受的答案),2.使用表变量或临时表和 INSERT EXEC iff your sproc isn't too complicated3. cheat and use OPENQUERY。但是每个答案都重复了惊人的次数,其中许多都获得了惊人的赞成票。你已经被警告过了。 🙈

C
Charles Bretana

你可以

创建一个表变量来保存存储过程中的结果集,然后将存储过程的输出插入到表变量中,然后像使用任何其他表一样使用表变量...

... sql ....

Declare @T Table ([column definitions here])
Insert @T Exec storedProcname params 
Select * from @T Where ...

INSERT #TINSERT @T 的问题是不能嵌套 INSERT EXEC 语句。如果存储过程中已经有一个 INSERT EXEC,这将不起作用。
这可能是最便携的解决方案,最接近基本 SQL。它还有助于维护强大的列类型定义。应该有比上述更多的赞成票。
就 sp 重新编译而言,table variables 在这里看起来比临时表更有用。所以我同意,这个答案应该有更多的赞成票。
K
Kolappan N

您可以使用 User-defined functionview 代替过程。

一个过程可以返回多个结果集,每个结果集都有自己的模式。它不适合在 SELECT 语句中使用。


此外,如果在转换为 UDF 后您发现需要存储过程语义,您可以始终使用过程包装 UDF。
如果我们需要向多个存储过程发送参数并将它们组合成一个大存储过程怎么办?可以查看、获取参数,就像存储过程一样
@mrN 视图不带参数,但 UDF 可以。
你好,我真的需要在不将sp转换为视图或函数的情况下这样做,可以吗?
虽然您的回答是真实的陈述,但它没有回答问题......“从存储过程中选择”这肯定不是理想的,但它就是这样......@Aamir 的答案是正确的答案。无论是那个还是这个问题都需要改变......这对我来说似乎有点荒谬。
C
CMerat

您要么需要 Table-Valued function,要么将 EXEC 插入临时表:

INSERT INTO #tab EXEC MyProc

INSERT #TINSERT @T 的问题是不能嵌套 INSERT EXEC 语句。如果存储过程中已经有一个 INSERT EXEC,这将不起作用。
A
Aamir

您需要声明一个包含与您的存储过程返回的列数相同的表类型。表类型中的列的数据类型和过程返回的列的数据类型应该相同

declare @MyTableType as table
(
FIRSTCOLUMN int
,.....
)  

然后你需要在你刚刚定义的表类型中插入存储过程的结果

Insert into @MyTableType 
EXEC [dbo].[MyStoredProcedure]

最后只需从您的表格类型中选择

Select * from @MyTableType

这对我来说是最好的解决方案,因为您不需要指定服务器名称、连接字符串或必须配置任何链接的服务器以使其工作 - 这是我不想做的事情只是为了一些数据回来。谢谢!好棒的答案!
很好的答案ღ❤ೋღ❤ღೋ❤ღ
当存储过程太难时 - 这种方法不起作用,例如,当存储过程使用两个临时表时。
P
Paul

您必须阅读 OPENROWSETOPENQUERY

SELECT  * 
INTO    #tmp FROM    
OPENQUERY(YOURSERVERNAME, 'EXEC MyProc @parameters')

如何动态获取 YOURSERVERNAME?你不能指望总是知道。这不会每隔一个星期二休息吗?因此,如果我有 100 台不同名称的服务器...
如果我的数据库未配置为允许这样做怎么办?
尝试@@servername 动态获取它
s
slavoo

没有必要使用临时表。

这是我的解决方案

SELECT  *  FROM    
OPENQUERY(YOURSERVERNAME, 'EXEC MyProc @parameters')
WHERE somefield = anyvalue

这需要您将服务器作为链接服务器添加到自身,但它就像一个魅力!谢谢!
对此有一些重要的警告:stackoverflow.com/questions/2374741/…
嗯...我收到错误“错误 7411:服务器 'YourServerName' 未配置为数据访问。”我需要改变什么?
您是否将服务器添加为链接服务器? YourServerName 是您的服务器的名称。您必须将 YourServerName 更改为您的真实服务器名称。
@马特:sp_serveroption 'MYSERVER', 'DATA ACCESS', TRUE;
佚名

您可以将输出从 sp 复制到临时表。

CREATE TABLE #GetVersionValues
(
    [Index] int,
    [Name]  sysname,
    Internal_value  int,
    Character_Value sysname
)
INSERT #GetVersionValues EXEC master.dbo.xp_msver 'WindowsVersion'
SELECT * FROM #GetVersionValues
drop TABLE #GetVersionValues

P
Phillip Senn

尝试将您的过程转换为返回表的内联函数,如下所示:

CREATE FUNCTION MyProc()
RETURNS TABLE AS
RETURN (SELECT * FROM MyTable)

然后你可以把它称为

SELECT * FROM MyProc()

您还可以选择将参数传递给函数,如下所示:

CREATE FUNCTION FuncName (@para1 para1_type, @para2 para2_type , ... ) 

并称它为

SELECT * FROM FuncName ( @para1 , @para2 )

m
marc_s

你可以用 OPENROWSET 作弊:

SELECT ...fieldlist...
FROM OPENROWSET('SQLNCLI', 'connection string', 'name of sp')
WHERE ...

当然,这仍然会每次都运行整个 SP。


SQL Server 阻止了对组件“即席分布式查询”的声明“OpenRowset/OpenDatasource”的访问,因为该组件作为该服务器安全配置的一部分被关闭。系统管理员可以使用 sp_configure 启用“即席分布式查询”的使用。有关启用“即席分布式查询”的详细信息,请在 SQL Server 联机丛书中搜索“即席分布式查询”。
A
Ali Osman Yavuz

如果“数据访问”为假,

EXEC sp_serveroption 'SQLSERVERNAME', 'DATA ACCESS', TRUE

后,

SELECT  *  FROM OPENQUERY(SQLSERVERNAME, 'EXEC DBNAME..MyProc @parameters')

有用。


T
TT.

使用 OPENQUERY,并在执行之前设置 SET FMTONLY OFF; SET NOCOUNT ON;

试试这个示例代码:

SELECT top(1)*
FROM
OPENQUERY( [Server], 'SET FMTONLY OFF; SET NOCOUNT ON; EXECUTE  [database].[dbo].[storedprocedure]  value,value ')

如果您收到错误“未为数据访问配置服务器”,请使用以下命令:

EXEC sp_serveroption 'YourServer', 'DATA ACCESS', TRUE

K
KirstieBallance

为了简单起见并使其可重新运行,我使用了系统 StoredProcedure "sp_readerrorlog" 来获取数据:

-----USING Table Variable
DECLARE @tblVar TABLE (
   LogDate DATETIME,
   ProcessInfo NVARCHAR(MAX),
   [Text] NVARCHAR(MAX)
)
INSERT INTO @tblVar Exec sp_readerrorlog
SELECT LogDate as DateOccured, ProcessInfo as pInfo, [Text] as Message FROM @tblVar



-----(OR): Using Temp Table
IF OBJECT_ID('tempdb..#temp') IS NOT NULL  DROP TABLE #temp;
CREATE TABLE #temp (
   LogDate DATETIME,
   ProcessInfo NVARCHAR(55),
   Text NVARCHAR(MAX)
)
INSERT INTO #temp EXEC sp_readerrorlog
SELECT * FROM #temp

L
Lawrence Barsanti

听起来您可能只需要使用 view。视图允许将查询表示为表,因此可以查询视图。


F
Fandango68

例如,如果您的服务器称为 SERVERX,我就是这样做的...

EXEC sp_serveroption 'SERVERX', 'DATA ACCESS', TRUE;
DECLARE @CMD VARCHAR(1000);
DECLARE @StudentID CHAR(10);
SET @StudentID = 'STUDENT01';
SET @CMD = 'SELECT * FROM OPENQUERY([SERVERX], ''SET FMTONLY OFF; SET NOCOUNT ON; EXECUTE MYDATABASE.dbo.MYSTOREDPROC ' + @StudentID + ''') WHERE SOMEFIELD = SOMEVALUE';
EXEC (@CMD);

为了检查这是否有效,我注释掉了 EXEC() 命令行并将其替换为 SELECT @CMD 以在尝试执行之前检查命令!这是为了确保所有正确数量的单引号都在正确的位置。 :-)

我希望这对某人有所帮助。