- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我有一个 Tools_NawContext
类扩展 DbContext
和一个 DbResult
类来调整 SaveChanges
方法的结果发生异常时的一点。抛出异常时,我会创建一条特定的错误消息,我知道它属于我尝试添加、删除或编辑的一个实体。用户可以根据错误信息采取适当的措施并重试。
public partial class Tools_NawContext : DbContext
{
public Tools_NawContext(DbContextOptions<Tools_NawContext> options) : base(options) { }
public DbResult TrySaveChanges()
{
try {
int numberOfRowsSaved = SaveChanges();
return new DbResult(numberOfRowsSaved);
} catch(Exception ex) {
return new DbResult(ex);
}
}
}
public class DbResult
{
public DbResult(int numberOfRowsSaved) {
this.Succeeded = true;
this.NumberOfRowsSaved = numberOfRowsSaved;
}
public DbResult(Exception exception)
{
this.Exception = exception;
if(exception.GetType() == typeof(DbUpdateException) && exception.InnerException != null) {
if (exception.InnerException.Message.StartsWith("The DELETE statement conflicted with the REFERENCE constraint")) {
this.DuplicateKeyError = true;
this.DuplicateKeyErrorMessage = "There are other objects related to this object. First delete all the related objects.";
} else if (exception.InnerException.Message.StartsWith("Violation of PRIMARY KEY constraint")) {
this.DuplicateKeyError = true;
this.DuplicateKeyErrorMessage = "There is already a row with this key in the database.";
} else if (exception.InnerException.Message.StartsWith("Violation of UNIQUE KEY constraint")) {
this.DuplicateKeyError = true;
this.DuplicateKeyErrorMessage = "There is already a row with this key in the database.";
}
} else if(exception.GetType() == typeof(System.InvalidOperationException) && exception.Message.StartsWith("The association between entity types")) {
this.DuplicateKeyError = true;
this.DuplicateKeyErrorMessage = "There are other objects related to this object. First delete all the related objects.";
}
}
public bool Succeeded { get; private set; }
public int NumberOfRowsSaved { get; private set; }
public bool DuplicateKeyError { get; private set; }
public string DuplicateKeyErrorMessage { get; private set; }
public Exception Exception { get; private set; }
public List<string> ErrorMessages { get; set; }
public string DefaultErrorMessage { get { if (Succeeded == false) return "Er is een fout in de database opgetreden."; else return ""; } private set { } }
}
但是我现在正在尝试导入一些 JSON 并想再次使用 TrySaveChanges
方法。然而这次经过一些检查后,我首先将多个实体添加到上下文中,而不仅仅是 1 个。添加完所有实体后,我将调用 TrySaveChanges
方法。它仍然有效,但如果出现问题,我无法确定哪些实体未能保存。如果我添加 1000 个实体,只有 1 个会失败,我无法确定哪里出错了。 我如何确定哪些添加的实体抛出错误?下面是我如何使用它的示例。
我有 2 个 EF 生成的类。 Testresultaten
和 Keuring
public partial class Testresultaten
{
public int KeuringId { get; set; }
public int TestId { get; set; }
public string Resultaat { get; set; }
public string Status { get; set; }
public int TestinstrumentId { get; set; }
public virtual Keuring Keuring { get; set; }
public virtual Test Test { get; set; }
public virtual Testinstrument Testinstrument { get; set; }
}
public partial class Keuring
{
public Keuring()
{
Keuring2Werkcode = new HashSet<Keuring2Werkcode>();
Testresultaten = new HashSet<Testresultaten>();
}
public int Id { get; set; }//NOTE: Auto-incremented by DB!
public int GereedschapId { get; set; }
public DateTime GekeurdOp { get; set; }
public int KeuringstatusId { get; set; }
public int TestmethodeId { get; set; }
public DateTime GekeurdTot { get; set; }
public string GekeurdDoor { get; set; }
public string Notitie { get; set; }
public virtual ICollection<Keuring2Werkcode> Keuring2Werkcode { get; set; }
public virtual ICollection<Testresultaten> Testresultaten { get; set; }
public virtual Gereedschap Gereedschap { get; set; }
public virtual Keuringstatus Keuringstatus { get; set; }
public virtual Testmethode Testmethode { get; set; }
}
我有一个 _KeuringImporter
类,它有一个方法可以将 newKeuring
和 testresultatenList
添加到 dbContext(_Tools_NawContext
).
private Result<KeuringRegel, Keuring> SetupKeuringToDB2(KeuringRegel row, int rownr, Keuring newKeuring)
{
_Tools_NawContext.Keuring.Add(newKeuring);
List<string> errorMessages = new List<string>();
List<Testresultaten> testresultatenList = new List<Testresultaten>();
foreach (string testName in row.testNames.Keys.ToList())
{
string testValue = row.testNames[testName].ToString();
Test test = _Tools_NawContext.Test.Include(item => item.Test2Testmethode).SingleOrDefault(item => item.Naam.Equals(testName, StringComparison.OrdinalIgnoreCase));
//-----!!NOTE!!-----: Here KeuringId = newKeuring.Id is a random negative nr and is not beeing roundtriped to the db yet!
Testresultaten newTestresultaten = new Testresultaten() { KeuringId = newKeuring.Id, TestId = test.Id, Resultaat = testValue, Status = row.Status, TestinstrumentId = 1 };
testresultatenList.Add(newTestresultaten);
}
_Tools_NawContext.Testresultaten.AddRange(testresultatenList);
return new Result<KeuringRegel, Keuring>(row, newKeuring, errorMessages);
}
就像我说的。我用它来导入 JSON。如果一个 JSON 文件包含 68 行,则该方法被调用 68 次。或者说:68 个新的 Keuring
项目被附加到 DbContext 并且每次 Testresultaten
列表被添加到 DbContext。
设置好所有内容后,我终于从我的 Controller 中调用了 SaveSetupImportToDB
。 (此方法也是我的 _KeuringImporter
类的一部分。)
public DbResult SaveSetupImportToDB()
{
DbResult dbResult = _Tools_NawContext.TrySaveChanges();
return dbResult;
}
我如何实现我想要的?在上面的例子中,在我的 MS SQL 数据库中,Keuring
表有一个主键 Id
,它由数据库自动递增。该表还具有 GereedschapId
和 GekeurdOp
的组合唯一键。
我可以在将 newKeuring
添加到上下文之前编写一些检查,如下所示:
private Result<KeuringRegel, Keuring> SetupKeuringToDB2(KeuringRegel row, int rownr, Keuring newKeuring)
{
List<string> errorMessages = new List<string>();
var existingKeuring = _Tools_NawContext.Keuring.SingleOrDefault(x => x.Id == newKeuring.Id);
if(existingKeuring == null) { errorMessages.Add("There is already a keuring with id " + newKeuring.Id + " in the db."); }
existingKeuring = _Tools_NawContext.Keuring.SingleOrDefault(x => x.GereedschapId == newKeuring.GereedschapId && x.GekeurdOp == newKeuring.GekeurdOp);
if (existingKeuring == null) { errorMessages.Add("There is already a keuring with GereedschapId " + newKeuring.GereedschapId + " and GekeurdOp " + newKeuring.GekeurdOp + " in the db."); }
//Some more checks to cerrect values of properties:
//-DateTimes are not in future
//-Integers beeing greater then zero
//-String lengths not beeing larger then 500 characters
//-And so on, etc...
_Tools_NawContext.Keuring.Add(newKeuring);
List<Testresultaten> testresultatenList = new List<Testresultaten>();
foreach (string testName in row.testNames.Keys.ToList())
{
string testValue = row.testNames[testName].ToString();
Test test = _Tools_NawContext.Test.Include(item => item.Test2Testmethode).SingleOrDefault(item => item.Naam.Equals(testName, StringComparison.OrdinalIgnoreCase));
//-----!!NOTE!!-----: Here KeuringId = newKeuring.Id is a random negative nr and is not beeing roundtriped to the db yet!
Testresultaten newTestresultaten = new Testresultaten() { KeuringId = newKeuring.Id, TestId = test.Id, Resultaat = testValue, Status = row.Status, TestinstrumentId = 1 };
testresultatenList.Add(newTestresultaten);
}
_Tools_NawContext.Testresultaten.AddRange(testresultatenList);
return new Result<KeuringRegel, Keuring>(row, newKeuring, errorMessages);
}
添加的第一个检查是简单的检查,以查看数据库中是否已经存在一个项目。我将不得不对添加到数据库中的每个实体进行这些检查。我更喜欢在不检查的情况下添加它们,在调用 SaveChanges
时捕获异常并告诉用户出了什么问题。通过我的应用程序为我节省了很多检查时间。我知道我无法检查所有情况,这就是为什么 DbResult
类还具有 DefaultErrorMessage
属性的原因。如果我当时“crud”1 个实体,这一切都很好。一次添加多个实体时就会出现问题。关于如何改进我的代码以便找出问题所在的任何建议?理想情况下,在调用 SaveChanges()
之后。但欢迎任何其他想法!可能会更改 DbContext
上的一个属性,如果实体被添加到上下文中,它会检查该实体是否已经存在。
最佳答案
如果您调用 SaveChanges
并且失败,则批处理中的所有操作都将回滚。最重要的是,您将获得一个 DbUpdateException
,其属性 Entries
将包含导致错误的条目。上下文本身仍会保存跟踪对象的状态(包括失败),您可以使用 ChangeTracker.Entries()
(可能您不需要它)
try
{
model.SaveChanges();
}
catch (DbUpdateException e)
{
//model.ChangeTracker.Entries();
//e.Entries - Resolve errors and try again
}
我是你的情况,你可以做一个循环,继续尝试,直到所有的东西都被保存起来,比如
while (true)
{
try
{
model.SaveChanges();
break;
}
catch (DbUpdateException e)
{
foreach (var entry in e.Entries)
{
// Do some logic or fix
// or just detach
entry.State = System.Data.Entity.EntityState.Detached;
}
}
}
关于c# - 添加一批实体。如何确定调用 SaveChanges() 时哪些实体失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46113398/
为了让我的代码几乎完全用 Jquery 编写,我想用 Jquery 重写 AJAX 调用。 这是从网页到 Tomcat servlet 的调用。 我目前情况的类似代码: var http = new
我想使用 JNI 从 Java 调用 C 函数。在 C 函数中,我想创建一个 JVM 并调用一些 Java 对象。当我尝试创建 JVM 时,JNI_CreateJavaVM 返回 -1。 所以,我想知
环顾四周,我发现从 HTML 调用 Javascript 函数的最佳方法是将函数本身放在 HTML 中,而不是外部 Javascript 文件。所以我一直在网上四处寻找,找到了一些简短的教程,我可以根
我有这个组件: import {Component} from 'angular2/core'; import {UserServices} from '../services/UserService
我正在尝试用 C 实现一个简单的 OpenSSL 客户端/服务器模型,并且对 BIO_* 调用的使用感到好奇,与原始 SSL_* 调用相比,它允许一些不错的功能。 我对此比较陌生,所以我可能会完全错误
我正在处理有关异步调用的难题: 一个 JQuery 函数在用户点击时执行,然后调用一个 php 文件来检查用户输入是否与数据库中已有的信息重叠。如果是这样,则应提示用户确认是否要继续或取消,如果他单击
我有以下类(class)。 public Task { public static Task getInstance(String taskName) { return new
嘿,我正在构建一个小游戏,我正在通过制作一个数字 vector 来创建关卡,该数字 vector 通过枚举与 1-4 种颜色相关联。问题是循环(在 Simon::loadChallenge 中)我将颜
我有一个java spring boot api(数据接收器),客户端调用它来保存一些数据。一旦我完成了数据的持久化,我想进行另一个 api 调用(应该处理持久化的数据 - 数据聚合器),它应该自行异
首先,这涉及桌面应用程序而不是 ASP .Net 应用程序。 我已经为我的项目添加了一个 Web 引用,并构建了各种数据对象,例如 PayerInfo、Address 和 CreditCard。但问题
我如何告诉 FAKE 编译 .fs文件使用 fsc ? 解释如何传递参数的奖励积分,如 -a和 -target:dll . 编辑:我应该澄清一下,我正在尝试在没有 MSBuild/xbuild/.sl
我使用下划线模板配置了一个简单的主干模型和 View 。两个单独的 API 使用完全相同的配置。 API 1 按预期工作。 要重现该问题,请注释掉 API 1 的 URL,并取消注释 API 2 的
我不确定什么是更好的做法或更现实的做法。我希望从头开始创建目录系统,但不确定最佳方法是什么。 我想我在需要显示信息时使用对象,例如 info.php?id=100。有这样的代码用于显示 Game.cl
from datetime import timedelta class A: def __abs__(self): return -self class B1(A):
我在操作此生命游戏示例代码中的数组时遇到问题。 情况: “生命游戏”是约翰·康威发明的一种细胞自动化技术。它由一个细胞网格组成,这些细胞可以根据数学规则生存/死亡/繁殖。该网格中的活细胞和死细胞通过
如果我像这样调用 read() 来读取文件: unsigned char buf[512]; memset(buf, 0, sizeof(unsigned char) * 512); int fd;
我用 C 编写了一个简单的服务器,并希望调用它的功能与调用其他 C 守护程序的功能相同(例如使用 ./ftpd start 调用它并使用 ./ftpd stop 关闭该实例)。显然我遇到的问题是我不知
在 dos 中,当我粘贴此命令时它会起作用: "C:\Program Files (x86)\Google\Chrome\Application\chrome.exe" https://google.
在 dos 中,当我粘贴此命令时它会起作用: "C:\Program Files (x86)\Google\Chrome\Application\chrome.exe" https://google.
我希望能够从 cmd 在我的 Windows 10 计算机上调用 python3。 我已重新安装 Python3.7 以确保选择“添加到路径”选项,但仍无法调用 python3 并使 CMD 启动 P
我是一名优秀的程序员,十分优秀!