- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我有混合 EF 和普通 SQL 调用的代码。整个事情在Azure上运行,所以我们使用ReliableSqlConnection 。我们正在使用 TransactionScope,但没有分布式事务管理器(又是 Azure)。因此我必须将 ReliableSqlConnection 传递给每个 SQL 调用。
现在的问题是如何将 ReliableSqlConnection 传递到 EF 调用中?如果发现这篇文章:
How to use ADO.net Entity Framework with an existing SqlConnection?
这会导致以下代码:
MetadataWorkspace workspace = new MetadataWorkspace(
new string[] { "res://*/" },
new Assembly[] { Assembly.GetExecutingAssembly() });
using (var scope = new TransactionScope())
using (var conn = DatabaseUtil.GetConnection())
using (EntityConnection entityConnection = new EntityConnection(workspace, (DbConnection)conn))
using (var db = new UniversalModelEntities(entityConnection))
{
//Do EF things
//Call other SQL commands
return db.SaveChanges();
}
但是我无法将 ReliableSqlConnection 转换为 DbConnection,UniversalModelEntities 也不接受 EntityConnection。
最佳答案
问题是 ReliableSqlConnection
实现 IDbConnection
接口(interface),但 EF 上下文构造函数都接受 DbConnection
(不是接口(interface))。我不知道他们为什么做出这样的决定,也许他们背后有合理的理由,也许这只是糟糕的设计决定。然而,你必须忍受这一点。请注意,使用 ReliableSqlConnection.Open()
或 ReliableSqlConnection.Current
返回的任何内容都不是一个选项 - 它会起作用,但您将只使用常规连接,然后无需重试逻辑,基本上绕过了整个 ReliableSqlConnection 类的目的。相反,您可以尝试围绕 ReliableSqlConnection 创建一个包装器,如下所示:
public class ReliableSqlConnectionWrapper : DbConnection {
private readonly ReliableSqlConnection _connection;
public ReliableSqlConnectionWrapper(ReliableSqlConnection connection) {
_connection = connection;
}
protected override DbTransaction BeginDbTransaction(System.Data.IsolationLevel isolationLevel) {
return (DbTransaction) _connection.BeginTransaction();
}
public override void Close() {
_connection.Close();
}
public override void ChangeDatabase(string databaseName) {
_connection.ChangeDatabase(databaseName);
}
public override void Open() {
_connection.Open();
}
public override string ConnectionString
{
get { return _connection.ConnectionString; }
set { _connection.ConnectionString = value; }
}
public override string Database
{
get { return _connection.Database; }
}
public override ConnectionState State
{
get { return _connection.State; }
}
public override string DataSource
{
get { return _connection.Current?.DataSource; }
}
public override string ServerVersion
{
get { return _connection.Current?.ServerVersion; }
}
protected override DbCommand CreateDbCommand() {
return _connection.CreateCommand();
}
protected override DbProviderFactory DbProviderFactory {
get { return SqlClientFactory.Instance; }
}
}
这里,我们按照 EF 的要求继承了 DbConnection
,并将所有逻辑转发到底层 ReliableSqlConnection
实例。请注意,您可能需要重写 DbConnection
中的更多方法(例如 Dispose
) - 此处我仅展示如何仅重写必需的(抽象)成员。
包装器的替代选项是复制 ReliableSqlConnection
类的源代码并修改它以继承 DbConnection
。
然后,在 EF 上下文中,您需要添加一个接受 DbConnection 的构造函数:
public UniversalModelEntities(DbConnection connection, bool contextOwnsConnection) : base(connection, contextOwnsConnection) {}
这只是使用相同的参数调用基类构造函数。第二个参数 (contextOwnsConnection
) 定义上下文是否能够管理此连接,例如在释放上下文时关闭它。
如果您使用 EF 数据库优先方法 - 编辑为您的上下文生成代码的 EF 模板,并在其中添加此构造函数。
完成这一切之后,您可以执行以下操作:
using (var scope = new TransactionScope()) {
using (var conn = new ReliableSqlConnection("")) {
using (var ctx = new UniversalModelEntities(new ReliableSqlConnectionWrapper(conn), false)) {
}
}
}
经过一番调查,我得出的结论是,上述方法可能很难实现,因为连接包装器与 Entity Framework 不太兼容。考虑更简单的替代方案 - 使用 DbCommandInterceptor 并通过提供 ReliableSqlConnection 的同一 transient 故障处理程序库提供的扩展方法重用重试逻辑:
public class RetryInterceptor : DbCommandInterceptor {
public override void NonQueryExecuting(DbCommand command, DbCommandInterceptionContext<int> interceptionContext) {
interceptionContext.Result = ((SqlCommand)command).ExecuteNonQueryWithRetry();
}
public override void ReaderExecuting(DbCommand command, DbCommandInterceptionContext<DbDataReader> interceptionContext) {
interceptionContext.Result = ((SqlCommand)command).ExecuteReaderWithRetry();
}
public override void ScalarExecuting(DbCommand command, DbCommandInterceptionContext<object> interceptionContext) {
interceptionContext.Result = ((SqlCommand)command).ExecuteScalarWithRetry();
}
}
因此,我们拦截命令并将其执行转发到 transient 故障处理程序 block 。然后在main方法中:
static void Main() {
// don't forget to add interceptor
DbInterception.Add(new RetryInterceptor());
MetadataWorkspace workspace = new MetadataWorkspace(
new string[] {"res://*/"},
new[] {Assembly.GetExecutingAssembly()});
// for example
var strategy = new FixedInterval("fixed", 10, TimeSpan.FromSeconds(3));
var manager = new RetryManager(new[] {strategy}, "fixed");
RetryManager.SetDefault(manager);
using (var scope = new TransactionScope()) {
using (var conn = new ReliableSqlConnection("data source=(LocalDb)\\v11.0;initial catalog=TestDB;integrated security=True;MultipleActiveResultSets=True;App=EntityFramework")) {
// pass Current - we don't need retry logic from ReliableSqlConnection any more
using (var ctx = new TestDBEntities(new EntityConnection(workspace, conn.Current), false)) {
// some sample code I used for testing
var code = new Code();
code.Name = "some code";
ctx.Codes.Add(code);
ctx.SaveChanges();
scope.Complete();
}
}
}
}
关于c# - 对 EntityFramework 和 "normal"SQL 调用使用相同的 SqlConnection,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36940240/
全部: 我们使用 ASP.NET Web 窗体开始了我们的项目。 我们的 ASP.NET C# web 应用程序在以下环境中使用 .NET Framework 4 ASP.NET Web Forms.
我在尝试为 Entity Framework Core 添加迁移到 Code First 项目时遇到错误,这里是详细信息... 我创建了一个新的 ASP.Net Core Web 项目(VS 2017
Error 1 Copying file bin\EntityFramework.SqlServer.xml to obj\Debug\Package\PackageTmp\bin\Entit
我有一个WPF应用程序,我使用了MVVM体系结构,数据库sql server compact 4。 首先是Entityframework.sqlserverCompact数据库,我使用了通用存储库模式
例如,将 products.Where(p => p.LastModifiedOn > someDate) 总是以相同的订单返回产品(无论如何订购,只要每次调用相同的代码时都是一致的)?或者我应该添加
好吧,假设我有一个用户表,然后是一个记录所有登录尝试的表(Id(计数器)、userId、时间以及是否成功) 现在在我的实体模型中,我希望它成为 2 个关联的一个,包含所有成功尝试的时间和一个包含所有不
我正在努力处理如何最好地跨多个 Visual Studio 项目分解域模型。我正在使用 EntityFramework 4 和企业类型模式来合理化一个可以重用于各种应用程序的系统。最终我想得到一组可以
是否可以使用 POCO 和 EF 处理以下情况?我想要一个 Vehicles 表,有一些 bool 字段,如 IsFast、IsVan、IsTruck,然后有一个 Vehicles 类,有一个 Fas
当我通过传递比较泛型类型 TId 的谓词调用 DbSet.FirstOrDefault() 时,出现以下异常: unable to create a constant value of type 's
我正在使用 EF5 将一些数据从一个数据库迁移到另一个数据库。我通常会使用 SQL 来做这样的事情,但是我需要其他功能(比如在 MembershipProvider 中创建用户)并且希望在 EF 中完
我创建了一个单独的项目并将我的模型复制到这个项目中。我引用了 using System.ComponentModel.DataAnnotations 我的模型定义如下: public class F
上下文: 我有两个类在 SQL 中有两个对应的表。 系统信息 EF 版本: SQL 版本:2012 类 public class Employee { public Employee(
我有一个定义了主从关系的 Entity Framework 对象。详细信息对象集合有一个导航属性。 在稍后的代码中,我尝试使用 AutoMapper 将其中一个主对象映射到数据传输对象。但是,数据传输
有没有办法让这个通用 Context.SalesEntity.Where(t=>t.id==3).Delete(); 有点像 private void DoWork(Expression> predi
来自 NHibernate 背景 - 我有一些审计列来反射(reflect)谁最后编辑了一条记录,以及该编辑发生的时间。 我已经覆盖了 SaveChanges 以寻找一个公共(public)基类,并使
我是 EntityFramework 6 的新手,一直在 POCO 中遇到问题。 以下是我的代码 Domain.cs public abstract class BaseEntity { [K
我刚在单元测试项目中遇到一个问题,坚持认为它需要 EntityFramework 的引用,我确信它不需要它。其他项目正在引用单元测试项目正在引用/测试的项目/扩展方法,并且在不引用 EntityFra
下面的代码应该做什么?在此代码中添加包含表的目的是什么,它应该对与 PersonId 相关的 JobType 进行级联删除,还是只删除具有指定 PersonId 的 Person ? db.tblPe
我试图了解延迟加载在 Entity Framework 6 中是如何工作的。在我看来,延迟加载只适用于 DbContext,其中“to-be-loaded”属性已经存在访问过。请查看下面的示例 - 测
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: Entity Framework Validation EntityFramework有什么好的灵活的验证框
我是一名优秀的程序员,十分优秀!