-2

[前言]

当我将我的 api 从 .netcore2.1 升级到 .netcore3.1 时,我发现我需要大幅更改我的实体框架代码。

我没有意识到我的远程数据库中的存储过程中有一个错误,这在我的 .netcore2.1 代码中没有表达出来。

我遇到的错误消息并没有将我指向检查存储过程内部的方向。如果搜索错误消息对某人有帮助,我会留下我的问题的最终形式。

[问题]

我有以下代码来调用返回整数的存储过程

该代码在服务器端单元测试中工作,但在通过 api 或通过 SwaggerUI 调用时失败

var obj = connect.Ints.FromSqlInterpolated<intDto>(@$"Set NOCOUNT ON
          declare @num int 
          exec @num = spGetDefaultID {userName}  
          select @num as num").ToList();

id = (int)obj.First().num;

connect 是我的 DbContext 包含

public DbSet<intDto> Ints { get; set; }  // I actually don't want a table

public class intDto
{
    public int num { get; set; }
}

我按照建议创建了一个虚拟视图,这样我就不会在我的数据库中得到不需要的表。

并且有

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        base.OnModelCreating(modelBuilder);
        modelBuilder.Entity<intDto>).HasNoKey().ToView("view_name_that_doesnt_exist");

调用堆栈是

System.InvalidOperationException: The required column 'num' was not present in the results of a 'FromSql' operation.
at Microsoft.EntityFrameworkCore.Query.Internal.BufferedDataReader.BufferedDataRecord.InitializeFields()
at Microsoft.EntityFrameworkCore.Query.Internal.BufferedDataReader.BufferedDataRecord.Initialize(DbDataReader reader, IReadOnlyList`1 columns)
at Microsoft.EntityFrameworkCore.Query.Internal.BufferedDataReader.Initialize(IReadOnlyList`1 columns) 
at Microsoft.EntityFrameworkCore.Storage.RelationalCommand.ExecuteReader(RelationalCommandParameterObject parameterObject)
at Microsoft.EntityFrameworkCore.Query.Internal.QueryingEnumerable`1.Enumerator.InitializeReader(DbContext _, Boolean result)
at Microsoft.EntityFrameworkCore.Storage.ExecutionStrategy.ExecuteImplementation[TState,TResult](Func`3 operation, Func`3 verifySucceeded, TState state)
at Microsoft.EntityFrameworkCore.Storage.ExecutionStrategy.Execute[TState,TResult](TState state, Func`3 operation, Func`3 verifySucceeded)
at Microsoft.EntityFrameworkCore.Query.Internal.QueryingEnumerable`1.Enumerator.MoveNext()
at System.Collections.Generic.List`1..ctor(IEnumerable`1 collection)
at System.Linq.Enumerable.ToList[TSource](IEnumerable`1 source)   
at MyAPi.Job_Management.JobDataRead.GetCartIdForUser(ApiDbContext connect, String userName) 

注意我已经大量编辑了这个问题,因为有一次我让 api 工作,但不再工作。

我发现我必须使用 .ToList() 否则出现错误

System.InvalidOperationException:FromSqlRaw 或 FromSqlInterpolated 是使用不可组合的 SQL 调用的,并且查询在其上构成。考虑AsEnumerable在 FromSqlRaw 或 FromSqlInterpolated 方法之后调用以在客户端执行组合。

[更新]

我弹出以下代码并没有失败

 var obj = db.Ints.FromSqlInterpolated(@$"Set NOCOUNT on
                                       select count(*) as num from people 
                                       where email like  {name} ").ToList();

所以下一步是尝试制作一个新的存储过程

或者,也许它是数据库版本..

有趣的是单元测试失败,除非我有

Microsoft.EntityFrameworkCore.Relational.dll

在正在测试的项目中引用。即使我没有明确使用它。

这种情况下的错误消息是

System.MissingMethodException: 'Method not found:'System.Linq.IQueriable '1<!!O> 
Microsoft.EntityFrameworkCore.RelationalQueryableExtensions.
FromSqlInterpolated(Microsoft.EntityFrameworkCOre.DbSet'1<!!0>, 
System.FormattableString)'

我尝试将我的代码更改为

var obj = connect.Ints.FromSqlInterpolated<intDto>(@$"declare @num int 
          exec @num = spGetDefaultID {userName}").ToList();

但是我的单元测试出错了

The underlying reader doesn't have as many fields as expected

[更新]

远程存储过程使用结果集返回数据。

create PROCEDURE [dbo].[spGetDefaultID] 
@email varchar(300)
AS
BEGIN
    SET NOCOUNT ON;
    declare @cartID int
    select @CartID= id from people where email = @email   /* simplified */
    select @CartID /* this line was not present in my local database */
    return @CartID  
END
4

2 回答 2

2

当我尝试按照 Zohar 的建议使用 ssms 调用存储过程时,我很惊讶地看到 2 个结果输出

当我删除了不必要的

选择@CartID

问题解决了。

于 2020-04-02T17:59:24.210 回答
0

1.你需要修改你的存储过程。您需要一个输出参数。它应该看起来。像这样:

CREATE PROCEDURE [dbo].[spGetDefaultID] (
                 @Username VARCHAR(100)
                ,@num_output int OUTPUT)
AS 
BEGIN
  SELECT @num_output = count(*)
  FROM [Person].[Person]
  WHERE PersonType LIKE (@Username)
RETURN
END

2.您需要修改一行代码。语法与您选择的语法略有不同。您需要 OUTPUT 语句。

SET NOCOUNT ON
DECLARE @num INT
exec spGetDefaultID {UserName}, @num_output=@num OUTPUT;
SELECT @num AS num

让您感到困惑的部分之一可能是 SQL 正在返回一个值,即使您像一开始那样编写它,但它没有将结果存储在 @num 参数中。

我希望这能解决你的问题。有关如何从存储过程返回数据的更多信息,请参阅 Microsoft Docs。

于 2020-04-02T09:41:38.483 回答