C# tsql过程返回值asp.net

C# tsql过程返回值asp.net,c#,sql-server,tsql,stored-procedures,C#,Sql Server,Tsql,Stored Procedures,我要做的是:我创建了一个T-SQL存储过程,它从多个表返回一个列 CREATE PROCEDURE [dbo].[Search_] (@am VARCHAR(12)) AS BEGIN SET NOCOUNT ON; DECLARE @DBName VARCHAR(128) DECLARE Tbl CURSOR READ_ONLY FOR SELECT LEFT(TABLE_NAME, 27) AS Tbl FROM

我要做的是:我创建了一个T-SQL存储过程,它从多个表返回一个列

CREATE PROCEDURE [dbo].[Search_] 
    (@am VARCHAR(12)) 
AS
BEGIN
    SET NOCOUNT ON;

    DECLARE @DBName VARCHAR(128)

    DECLARE Tbl CURSOR READ_ONLY FOR
         SELECT LEFT(TABLE_NAME, 27) AS Tbl 
         FROM ap.INFORMATION_SCHEMA.TABLES 
         WHERE TABLE_NAME LIKE LEFT('STORED_PRODUCTS_FR_[0-9]_%', 27)
           AND SUBSTRING(TABLE_NAME, 20, 2) LIKE '%[0-9]%'
           AND SUBSTRING(TABLE_NAME, 23, 1) = '_'
         ORDER BY
             CONVERT(INT, SUBSTRING(TABLE_NAME, 24, 4)) ASC,
             CONVERT(INT, SUBSTRING(TABLE_NAME, 21, 2)) ASC

    OPEN Tbl

    FETCH NEXT FROM Tbl INTO @DBName

    WHILE @@FETCH_STATUS = 0
    BEGIN
        DECLARE @SQL VARCHAR(100)
        SET @sql = 'SELECT [column 3] FROM ' + @DBName + ' WHERE [COLUMN 4] = ' + @am + '';

        EXEC(@sql)

        FETCH NEXT FROM Tbl INTO @DBName
    END
END

CLOSE Tbl
DEALLOCATE Tbl
然后在asp.net中,我想填充一个gridview

try
{
    SqlDataAdapter dt = new SqlDataAdapter();

    string message = string.Empty;
    con.Open();

    cmd.CommandText = "Search_";
    cmd.CommandType = CommandType.StoredProcedure;
    cmd.Parameters.AddWithValue("@am", am.Text);   
    cmd.Connection = con;

    dt.SelectCommand = cmd;
    DataTable dTable = new DataTable();
    dt.Fill(dTable);

    GridView1.DataSource = dTable;
    GridView1.DataBind();
}
catch (Exception ex)
{
    throw ex;
}
它只返回存储过程的一行(第一行)

我做错了什么?这些值似乎正在覆盖它们自己


任何人的任何帮助…

您的存储过程都会返回两行,但由于每一行都是在不同的select语句中选择的,
DataAdapter
只能使用返回的第一行来填充
数据表。如果在哪里填充
数据集
,它将包含2个
数据表
实例

请注意,您的存储过程表明您的数据库设计中存在缺陷-对于数据库来说,具有多个存储相同类型实体的表是一种糟糕的设计。然而,我不确定这里的情况是否如此

无论如何,要通过快速修补程序解决此问题,您可以将过程更改为以下内容:

CREATE PROCEDURE [dbo].[Search_] (@am varchar(12)) 
AS
BEGIN
    SET NOCOUNT ON;
    DECLARE @DBName varchar(128)
    declare @SQL varchar(max) = ''

    DECLARE Tbl CURSOR READ_ONLY FOR
    select LEFT(TABLE_NAME,27) as Tbl from ap.INFORMATION_SCHEMA.TABLES 
    where TABLE_NAME like left('STORED_PRODUCTS_FR_[0-9]_%',27)
    and SUBSTRING(TABLE_NAME , 20,2) like '%[0-9]%'
    and SUBSTRING(TABLE_NAME , 23,1) = '_'
    order by CONVERT(int , substring(TABLE_NAME,24,4)) asc ,
    CONVERT(int , substring(TABLE_NAME,21,2)) asc   

    OPEN Tbl

    FETCH NEXT FROM Tbl INTO @DBName
    WHILE @@FETCH_STATUS = 0
    BEGIN
        set @sql += 'union all select [column 3] from '+@DBName+' WHERE [COLUMN 4] ='+@am+' ';
        FETCH NEXT FROM Tbl INTO @DBName
    END

    CLOSE Tbl
    DEALLOCATE Tbl

    SET @sql = STUFF(@Sql, 1, 10, '') -- Remove the first union all
    exec(@sql)
END

在脚本中,最后一个
end
应该放在
DEALLOCATE Tbl
之后,用于调用返回多个记录集的sp以更好地使用DataSet或DataReader:

var dataset = new DataSet();
using(var cnnn = new SqlConnection(cnnString))
{
    var adapter = new SqlDataAdapter();
    adapter.SelectCommand = new SqlCommand("dbo.Search_", cnnn);
    adapter.SelectCommand.CommandType = CommandType.StoredProcedure;
    adapter.Fill(dataset);
}
dataset.Tables => gives u all the results of exec(@sql)
由于每个结果集都重新运行相同的列名和结构,因此可以将它们合并到一个tb中:

var allTbs = dataset.Tables[0].Copy();
for(int i = 1; i < dataset.Tables.Count; i++)
    allTbs.Merge(dataset.Tables[i]);
var allTbs=dataset.Tables[0].Copy();
对于(int i=1;i
您的SP是否可以在SQL Server中正常工作?可以。它会按预期返回2行。请在SQL management studio中运行存储过程,查看数据是如何返回的。可能有两个表有一行,而不是一个表有两行。此代码将为每个结果返回一个DataTable,并最终仅将第一个表及其唯一结果绑定到DataGrid中。我发布的不是2行,而是3行。我没有注意到最后一行。前两行有值。最后一个名为“Return Value”,值为0。我刚刚测试了它,它就工作了..上面的解决方案非常有效..感谢您的帮助;-)