- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
根据 CQRS 模式,我有一个简单的命令如下:
public sealed class EditPersonalInfoCommandHandler : ICommandHandler<EditPersonalInfoCommand> {
private readonly AppDbContext _context;
public EditPersonalInfoCommandHandler(AppDbContext context) {
_context = context;
}
public Result Handle(EditPersonalInfoCommand command) {
var studentRepo = new StudentRepository(_context);
Student student = studentRepo.GetById(command.Id);
if (student == null) {
return Result.Failure($"No Student found for Id {command.Id}");
}
student.Name = command.Name;
student.Email = command.Email;
_context.SaveChanges();
return Result.Success();
}
}
现在我需要尝试 _context.SaveChanges()
最多 5 次,如果失败并出现异常。为此,我可以简单地在方法中使用一个 for 循环:
for(int i = 0; i < 5; i++) {
try {
//required logic
} catch(SomeDatabaseException e) {
if(i == 4) {
throw;
}
}
}
要求是将方法作为一个单元来执行。问题是,一旦 _context.SaveChanges()
抛出异常,就不能使用相同的 _context
重新尝试逻辑。文档说:
Discard the current DbContext.Create a new DbContext and restore the state of your application from the database.Inform the user that the last operation might not have been completed successfully.
但是,在 Startup.cs
中,我将 AppDbContext
作为作用域依赖项。为了重新尝试方法逻辑,我需要一个 AppDbContext
的新实例,但注册为作用域将不允许这样做。
我想到的一个解决方案是使 AppDbContext
成为 transient 的。但我有一种感觉,这样做会为我自己打开一整套新问题。谁能帮我解决这个问题?
最佳答案
保存上下文时至少有太多错误。第一个发生在命令执行期间。第二次发生在提交期间(这种情况很少发生)。即使数据已成功更新,也可能会发生第二个错误。所以您的代码只处理第一种错误,而没有考虑第二种错误。
对于第一种错误,您可以在命令处理程序中注入(inject) DbContext 工厂或使用 IServiceProvider
直接地。这是一种反模式,但在这种情况下我们别无选择,如下所示:
readonly IServiceProvider _serviceProvider;
public EditPersonalInfoCommandHandler(IServiceProvider serviceProvider) {
_serviceProvider = serviceProvider;
}
for(int i = 0; i < 5; i++) {
try {
using var dbContext = _serviceProvider.GetRequiredService<AppDbContext>();
//consume the dbContext
//required logic
} catch(SomeDatabaseException e) {
if(i == 4) {
throw;
}
}
}
但是正如我所说,要处理这两种错误,我们应该使用所谓的 IExecutionStrategy
在 EFCore 中。介绍了几个选项here .但我认为以下内容最适合您的情况:
public Result Handle(EditPersonalInfoCommand command) {
var strategy = _context.Database.CreateExecutionStrategy();
var studentRepo = new StudentRepository(_context);
Student student = studentRepo.GetById(command.Id);
if (student == null) {
return Result.Failure($"No Student found for Id {command.Id}");
}
student.Name = command.Name;
student.Email = command.Email;
const int maxRetries = 5;
int retries = 0;
strategy.ExecuteInTransaction(_context,
context => {
if(++retries > maxRetries) {
//you need to define your custom exception to be used here
throw new CustomException(...);
}
context.SaveChanges(acceptAllChangesOnSuccess: false);
},
context => context.Students.AsNoTracking()
.Any(e => e.Id == command.Id &&
e.Name == command.Name &&
e.Email == command.Email));
_context.ChangeTracker.AcceptAllChanges();
return Result.Success();
}
请注意,我想您的上下文公开了一个 DbSet<Student>
通过属性 Students
.如果您有任何其他与连接无关的错误,IExecutionStrategy
将不会处理。这很有意义。因为那是您需要修复逻辑的时候,所以重试数千次无济于事,而且总是以该错误告终。这就是为什么我们不需要关心详细的最初抛出的异常(当我们使用 IExecutionStrategy
时不会暴露)。相反,我们使用自定义异常(如我上面的代码中所述)来通知由于某些与连接相关的问题而导致保存更改失败。
关于c# - 如何在 EF Core 中具有作用域 Dbcontext 的同时实现数据库弹性?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/66573522/
这是我的本地域名 http://10.10.1.101/uxsurvey/profile/dashboard 在 Controller 中,我为用户列表设置了一个操作 redirect(control
要处理 Canonical URL,最佳做法是执行 301 重定向还是更好地为 www 和非 www 域使用相同的 IP 地址? 例如: 想要的规范 URL/域是 http://example.com
1 内网基础 内网/局域网(Local Area Network,LAN),是指在某一区域内有多台计算机互联而成的计算机组,组网范围通常在数千米以内。在局域网中,可以实现文件管理、应用软件共享、打印机
1 内网基础 内网/局域网(Local Area Network,LAN),是指在某一区域内有多台计算机互联而成的计算机组,组网范围通常在数千米以内。在局域网中,可以实现文件管理、应用软件共享、打印机
我想创建一个 weblogic 集群,其中有两个托管服务器,每个服务器在物理上独立的远程计算机上运行 根据weblogic文档 All Managed Servers in a cluster mus
我正在运行 grails 3.1.4,但在创建允许我将多个域对象绑定(bind)到其他几个域对象的模式时遇到了问题。作为我正在尝试做的一个例子: 我有三个类(class)。书籍、作者和阅读列表。 作者
我试图使用@count函数来根据它获取数据,但是在没有崩溃报告的情况下它以某种方式崩溃了。 这是代码 class PSMedia: Object { @objc dynamic var id
有谁知道是否有办法只输入字母字符而不输入数字?我想过这样的事情 CREATE DOMAIN countryDomain AS VARCHAR(100) CHECK( VALUE ??? );
我的代码: const checkoutUrl = 'https://example.com/checkout/*' window.onload = startup() function st
一些不是我编写的应用程序,也不是用 PHP 编写的,它为域 www.example.com 创建了一个 cookie。 我正在尝试替换该 cookie。所以在 PHP 中我做到了: setcookie
什么是 oauth 域?是否有任何免费的 oauth 服务?我可以将它用于 StackApps registration 吗? ?我在谷歌上搜索了很多,但找不到答案。 最佳答案 这是redirect_
自从 In October 2009, the Internet Corporation for Assigned Names and Numbers (ICANN) approved the cre
我使用 apache 作为我的应用程序 Web 服务器的代理,并希望即时更改与 sessionid cookie 关联的域名。 该cookie有一个与之关联的.company.com域,我想使用apa
我只想托管一个子域到cloudflare。我不想将主域名的域名服务器更改为他们的域名服务器。真的有可能吗? 最佳答案 是的,这是可能的,但是需要通过CloudFlare合作伙伴进行设置,或者您需要采用
When using socket in the UNIX domain, it is advisable to use path name for the directory directory m
想象两个共享一个域类的 Grails 应用程序。也许是 Book 域类。 一个应用程序被标识为数据的所有者,一个应用程序必须访问域数据。类似于亚马逊和亚马逊网络服务。 我想拥有的应用程序将使用普通的域
我有一个包含字段“URL”的表单。第一部分需要用户在文本框中填写。第二部分是预定义的,显示在文本框的右侧。 例如,用户在文本框中输入“test”。第二部分预定义为“.example.com”。因此,总
如果我要关闭并取消分配 azure 中的域 Controller ,从而生成新的 vm Generationid,我需要采取哪些步骤来恢复它? 最佳答案 what steps do I need to
我想尝试使用 Azure 作为托管提供商(我有一个域)。我读过那篇文章https://learn.microsoft.com/en-us/azure/app-service-web/web-sites
所以.... 我想知道是否有人可以在这方面协助我? 基本上,我已经创建了一个自托管的Docker容器,用作构建代理(Azure DevOps) 现在,我已经开始测试代理,并且由于我们的放置文件夹位于W
我是一名优秀的程序员,十分优秀!