69 votes

Comment définir une valeur pour une variable en utilisant 'execute' dans t-sql ?

DECLARE @dbName nvarchar(128) = 'myDb'
DECLARE @siteId int 
exec ('SELECT TOP 1 @siteId = Id FROM ' + @dbName + '..myTbl')  
select @siteId

Lorsque j'exécute le script ci-dessus, j'obtiens l'erreur suivante

Msg 137, Level 15, State 1, Line 1
Must declare the scalar variable "@siteId".

(1 row(s) affected)

Pourquoi et comment y remédier ?

Merci.

129voto

Mikael Eriksson Points 77190

Vous pouvez utiliser les paramètres de sortie avec sp_executesql.

DECLARE @dbName nvarchar(128) = 'myDb'
DECLARE @siteId int 
DECLARE @SQL nvarchar(max) = N'SELECT TOP 1 @siteId = Id FROM ' + quotename(@dbName) + N'..myTbl'
exec sp_executesql @SQL, N'@siteId int out', @siteId out
select @siteId

2 votes

+1 C'est la façon la plus agréable. J'ai vu un nom de base de données paramétré et j'ai rejeté sp_executesql...

0 votes

Cool, j'ai trouvé ce que je cherchais !

5 votes

Je ne suis pas en mesure d'utiliser @siteId par la suite... il renvoie NULL. La ligne d'exécution fonctionne bien...

26voto

gbn Points 197263

Le SQL dynamique a un champ d'application différent de celui du SQL externe, qui appelle : @siteid n'est donc pas reconnu.

Vous devrez utiliser une table temporaire/une variable de table en dehors du SQL dynamique :

DECLARE @dbName nvarchar(128) = 'myDb'
DECLARE @siteId TABLE (siteid int)

INSERT @siteId
exec ('SELECT TOP 1 Id FROM ' + @dbName + '..myTbl')  

select * FROM @siteId

Note : TOP sans ORDER BY est inutile . Il n'y a pas d'ordre naturel, implicite ou intrinsèque dans un tableau. Tout ordre n'est garanti que par l'ordre extérieur ORDER BY

3 votes

"TOP sans ORDER BY n'a aucun sens" - "Aucun sens" est probablement un peu fort ici, peut-être que tout ce que je veux est un enregistrement et je ne me soucie pas duquel, par exemple, quel restaurant dois-je manger ce soir ?

0 votes

@RyanfaeScotland : peut-être, mais alors il n'y a pas de ANY() fucntion en SQL qui donne des rangs arbitraires. Sinon, je voudrais que ma requête se comporte de la même manière à chaque fois, au lieu de dépendre d'un ordre supposé.

0 votes

Votre remarque n'est tout simplement pas vraie dans la plupart des scénarios pratiques. A maintenu Un index en grappe sur la table permettra d'assurer un ordre cohérent et prévisible des résultats de la table.

1voto

Uğur Hamurpet Points 11

Vous pouvez essayer comme ci-dessous

DECLARE @sqlCommand NVARCHAR(4000)
DECLARE @ID INT
DECLARE @Name NVARCHAR(100)
SET @ID = 4
SET @sqlCommand = 'SELECT @Name = [Name]
FROM [AdventureWorks2014].[HumanResources].[Department]
WHERE DepartmentID = @ID'
EXEC sp_executesql @sqlCommand, N'@ID INT, @Name NVARCHAR(100) OUTPUT',
@ID = @ID, @Name = @Name OUTPUT
SELECT @Name ReturnedName

Source : blog.sqlauthority.com

0voto

Andrew Foster Points 1
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
-- =============================================
-- Author:      Andrew Foster
-- Create date: 28 Mar 2013
-- Description: Allows the dynamic pull of any column value up to 255 chars from regUsers table
-- =============================================
ALTER PROCEDURE dbo.PullTableColumn
(
    @columnName varchar(255),
    @id int
)
AS
BEGIN
    -- SET NOCOUNT ON added to prevent extra result sets from
    -- interfering with SELECT statements.
    SET NOCOUNT ON;

    DECLARE @columnVal TABLE (columnVal nvarchar(255));

    DECLARE @sql nvarchar(max);
    SET @sql = 'SELECT ' + @columnName + ' FROM regUsers WHERE id=' + CAST(@id AS varchar(10));
    INSERT @columnVal EXEC sp_executesql @sql;

    SELECT * FROM @columnVal;
END
GO

0voto

anuj sharma Points 1

Une légère modification de la requête d'exécution résoudra le problème :

DECLARE @dbName nvarchar(128) = 'myDb'
DECLARE @siteId int 
exec ('SELECT TOP 1 **''@siteId''** = Id FROM ' + @dbName + '..myTbl')  
select @siteId

Prograide.com

Prograide est une communauté de développeurs qui cherche à élargir la connaissance de la programmation au-delà de l'anglais.
Pour cela nous avons les plus grands doutes résolus en français et vous pouvez aussi poser vos propres questions ou résoudre celles des autres.

Powered by:

X