当我在 DataContext 中手动打开连接时应遵循什么模式
Linq2Sql 不支持存储过程的表值参数 (1, <一href="https://stackoverflow.com/questions/1019414/linq-to-sql-with-stored-procedures-and-user-define-table-type-parameter">2)。
因此,我在 DataContext 中添加了对表值参数的丑陋支持(只是为了保持一致,并且仍然能够从同一数据上下文中调用我的过程)。
我手动调用 this.Connection.CreateCommand()
,将其强制转换为 SqlCommand
(因为无论如何它都会是 SqlCommand
)列出所有参数,包括 SqlDbType.Structured 参数和 ExecuteNonQuery 参数。
这工作正常。
但是。
在我的辅助方法中,我必须打开上下文连接,因为它可能已关闭。 因为我手动打开连接,所以我必须手动关闭它吗?
我已经经历过这些:
我是否必须关闭如果我使用 Linq,请手动连接 SQL?
我应该何时处置数据上下文
如果我使用 DataContext,何时启动与数据库的连接?< /a>
但他们实际上并没有回答我的问题。我知道枚举结果时连接已关闭,但它比这更复杂。与枚举结果相反,调用方法不是只读操作。
此外,如果存在环境事务,DataContext
肯定不会在第一个方法调用后关闭连接:
using (var ambient_tran = new System.Transactions.TransactionScope())
{
using (var dx = new SomeDataContext())
{
dx.CallMethodOne(foo, bar);
dx.CallMethodTwo(bar, baz); // executed on same connection and same transaction,
// can see uncommitted data entered by CallMethodOne
}
ambient_tran.Complete();
};
正因为如此,我不确定什么是正确的事情关于连接要做的事情。
我只是打开连接并让 DataContext
管理其生命周期,还是有可遵循的模式?
编辑:
实验表明,如果我只是将其保持打开状态,它会表现良好并自动尊重环境事务(如果有)。不过,我更愿意就此提供可靠的建议。
Linq2Sql does not support table-valued parameters for stored procedures (1, 2).
Because of this, I'm adding an ugly support for table-valued parameters in my DataContext (just to keep things consistent and still be able to call my procedures from within the same data context).
I call this.Connection.CreateCommand()
, cast it to SqlCommand
(because it's going to be SqlCommand
anyway to support the TVP), manually list all parameters including the SqlDbType.Structured
ones, and ExecuteNonQuery
it.
This works fine.
But.
Inside my helper method I have to open the context connection because it may be closed. Because I manually open connection, do I have to manually close it?
I've been through these:
Do I have to close the SQL Connection manually if I use Linq?
When should I dispose of a data context
When the connection to database starts if i use DataContext?
but they don't actually answer my question. I understand the connection is closed when result are enumerated, but it's more complex than that. Calling a method is not a read-only operation, as opposed to enumerating results.
Besides, if there's an ambient transaction, DataContext
will definitely not close the connection after the first method call:
using (var ambient_tran = new System.Transactions.TransactionScope())
{
using (var dx = new SomeDataContext())
{
dx.CallMethodOne(foo, bar);
dx.CallMethodTwo(bar, baz); // executed on same connection and same transaction,
// can see uncommitted data entered by CallMethodOne
}
ambient_tran.Complete();
};
Because of all this, I'm not sure what is the right thing to do in regard of the connection.
Am I just to open connection and leave DataContext
manage its lifetime, or is there a pattern to follow?
EDIT:
Experiments show that if I just leave it open, it behaves fine and automatically respects an ambient transaction, if any. However, I would prefer a solid advice on that.
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(1)
如果不需要分布式事务,请避免使用 System.Trasactions。相反,请参阅我的 TextMetal 项目、示例代码,并注意 Plumbing 命名空间。如果您有疑问,请给我发电子邮件。没有 MSDTC 的手脚,只是正确生成的 ADO.NET/LINQ to SQL、工作单元模式和一些精心设计的数据访问代码!
如果有令人信服的理由使用 System.Transactions,那么我建议您仍然尽快关闭连接,因为事务协调器将在连接释放到池后管理提交/回滚。
If you do not need distributed transactions, then avoid using System.Trasactions. Instead, see my TextMetal project, the sample code, and pay attention to the Plumbing namespace. If you have questions, email me. No MSDTC slight of hand, just proper generated ADO.NET/LINQ to SQL, a unit of work pattern, and some well engineered data access code!
If there is a compelling reason to use System.Transactions, then I advise you still close the connection as soon as possible as the transaction coordinator will manage the commit/rollback after the connection is released to the pool.