प्रश्न के लिए कुछ साल देर हो गई, लेकिन मुझे कुछ त्वरित और गंदे कोड पीढ़ी के लिए कुछ इस तरह की आवश्यकता थी। मेरा मानना है कि जैसा कि दूसरों ने कहा है कि अस्थायी तालिका को सामने से परिभाषित करना आसान है, लेकिन इस पद्धति को सरल संग्रहीत कार्यविधि प्रश्नों या sql मूर्तियों के लिए काम करना चाहिए।
यह थोड़ा जटिल होगा, लेकिन यह यहां के योगदानकर्ताओं से उधार लेता है और साथ ही डीबीए स्टैक एक्सचेंज से पॉल व्हाइट के समाधान संग्रहीत कार्यविधि परिणाम कॉलम-प्रकार प्राप्त करें । फिर से, इस दृष्टिकोण और उदाहरण को दोहराने के लिए एक बहु उपयोगकर्ता वातावरण में प्रक्रियाओं के लिए डिज़ाइन नहीं किया गया है। इस मामले में एक कोड पीढ़ी टेम्पलेट प्रक्रिया द्वारा संदर्भ के लिए एक वैश्विक अस्थायी तालिका में थोड़े समय के लिए तालिका परिभाषा निर्धारित की जा रही है।
मैंने इसका पूरी तरह से परीक्षण नहीं किया है, इसलिए हो सकता है कि आप पॉल व्हाइट के जवाब में एमएसडीएन लिंक पर जाना चाहें। यह SQL 2012 और उच्चतर पर लागू होता है।
पहले संग्रहीत कार्यविधि का उपयोग करें sp_describe_first_result_set जो ओरेकल के विवरण से मिलता जुलता है।
यह पहले परिणाम सेट की पहली पंक्ति का मूल्यांकन करेगा, यदि आपकी संग्रहीत प्रक्रिया या कथन कई प्रश्न देता है तो यह केवल पहले परिणाम का वर्णन करेगा।
मैंने उन कार्यों को तोड़ने के लिए एक संग्रहित खरीद बनाई, जो एक एकल फ़ील्ड लौटाता है जिससे अस्थायी तालिका परिभाषा बनाने के लिए चयन किया जाता है।
CREATE OR ALTER PROCEDURE [dbo].[sp_GetTableDefinitionFromSqlBatch_DescribeFirstResultSet]
(
@sql NVARCHAR(4000)
,@table_name VARCHAR(100)
,@TableDefinition NVARCHAR(MAX) OUTPUT
)
AS
BEGIN
SET NOCOUNT ON
DECLARE @TempTableDefinition NVARCHAR(MAX)
DECLARE @NewLine NVARCHAR(4) = CHAR(13)+CHAR(10)
DECLARE @ResultDefinition TABLE ( --The View Definition per MSDN
is_hidden bit NOT NULL
, column_ordinal int NOT NULL
, [name] sysname NULL
, is_nullable bit NOT NULL
, system_type_id int NOT NULL
, system_type_name nvarchar(256) NULL
, max_length smallint NOT NULL
, [precision] tinyint NOT NULL
, scale tinyint NOT NULL
, collation_name sysname NULL
, user_type_id int NULL
, user_type_database sysname NULL
, user_type_schema sysname NULL
, user_type_name sysname NULL
, assembly_qualified_type_name nvarchar(4000)
, xml_collection_id int NULL
, xml_collection_database sysname NULL
, xml_collection_schema sysname NULL
, xml_collection_name sysname NULL
, is_xml_document bit NOT NULL
, is_case_sensitive bit NOT NULL
, is_fixed_length_clr_type bit NOT NULL
, source_server sysname NULL
, source_database sysname NULL
, source_schema sysname NULL
, source_table sysname NULL
, source_column sysname NULL
, is_identity_column bit NULL
, is_part_of_unique_key bit NULL
, is_updateable bit NULL
, is_computed_column bit NULL
, is_sparse_column_set bit NULL
, ordinal_in_order_by_list smallint NULL
, order_by_is_descending smallint NULL
, order_by_list_length smallint NULL
, tds_type_id int NOT NULL
, tds_length int NOT NULL
, tds_collation_id int NULL
, tds_collation_sort_id tinyint NULL
)
--Insert the description into table variable
INSERT @ResultDefinition
EXEC sp_describe_first_result_set @sql
--Now Build the string to create the table via union select statement
;WITH STMT AS (
SELECT N'CREATE TABLE ' + @table_name + N' (' AS TextVal
UNION ALL
SELECT
CONCAT(
CASE column_ordinal
WHEN 1 THEN ' ' ELSE ' , ' END --Determines if comma should precede
, QUOTENAME([name]) , ' ', system_type_name -- Column Name and SQL TYPE
,CASE is_nullable
WHEN 0 THEN ' NOT NULL' ELSE ' NULL' END --NULLABLE CONSTRAINT
) AS TextVal
FROM @ResultDefinition WHERE is_hidden = 0 -- May not be needed
UNION ALL
SELECT N');' + @NewLine
)
--Now Combine the rows to a single String
SELECT @TempTableDefinition = COALESCE (@TempTableDefinition + @NewLine + TextVal, TextVal) FROM STMT
SELECT @TableDefinition = @TempTableDefinition
END
पहेली यह है कि आपको एक वैश्विक तालिका का उपयोग करने की आवश्यकता है, लेकिन आपको इसे अद्वितीय बनाने की आवश्यकता है ताकि आप टकराव की चिंता किए बिना इसे बार-बार छोड़ सकें और बना सकें।
उदाहरण में मैंने एक गाइड (FE264BF5_9C32_438F_8462_8A5DC8DEE49E) का उपयोग किया, वैश्विक चर के लिए हाइफ़न को अंडरस्कोर के साथ बदलना
DECLARE @sql NVARCHAR(4000) = N'SELECT @@SERVERNAME as ServerName, GETDATE() AS Today;'
DECLARE @GlobalTempTable VARCHAR(100) = N'##FE264BF5_9C32_438F_8462_8A5DC8DEE49E_MyTempTable'
--@sql can be a stored procedure name like dbo.foo without parameters
DECLARE @TableDef NVARCHAR(MAX)
DROP TABLE IF EXISTS #MyTempTable
DROP TABLE IF EXISTS ##FE264BF5_9C32_438F_8462_8A5DC8DEE49E_MyTempTable
EXEC [dbo].[sp_GetTableDefinitionFromSqlBatch_DescribeFirstResultSet]
@sql, @GlobalTempTable, @TableDef OUTPUT
--Creates the global table ##FE264BF5_9C32_438F_8462_8A5DC8DEE49E_MyTempTable
EXEC sp_executesql @TableDef
--Now Call the stored procedure, SQL Statement with Params etc.
INSERT ##FE264BF5_9C32_438F_8462_8A5DC8DEE49E_MyTempTable
EXEC sp_executesql @sql
--Select the results into your undefined Temp Table from the Global Table
SELECT *
INTO #MyTempTable
FROM ##FE264BF5_9C32_438F_8462_8A5DC8DEE49E_MyTempTable
SELECT * FROM #MyTempTable
DROP TABLE IF EXISTS #MyTempTable
DROP TABLE IF EXISTS ##FE264BF5_9C32_438F_8462_8A5DC8DEE49E_MyTempTable
फिर, मैंने केवल इसे सरल संग्रहित प्रक्रिया प्रश्नों और सरल प्रश्नों के साथ परीक्षण किया है ताकि आपका माइलेज अलग-अलग हो सके। आशा है कि यह किसी की मदद करता है।