我一直在努力解决这个问题一段时间,但我无法解决这个问题。已阅读多篇文章并尝试了几种不同的解决方案但没有任何效果。现在我觉得我已经停下来,真的需要帮助。
我正在使用带有DB first和edmx文件的EF 5+。我的数据库中有3个不同的表: 1.结算 2.成本 3.转移
结算在我的edmx文件中有一个由Association和Shift连接的集合(带有链接表)。
我需要在我的数据库中插入一个新的结算,并引用已有的Cost和Shift集合。 我试图插入的我的结算实体中包含的班次和成本包含所有相关数据,并且没有任何方式被修改(与我从数据库中检索的相同)。 这是我将实体插入我的数据库的方法。
public bool CreateSettlement(Settlement settlement)
{
bool _success;
var _context = new EtaxiEnteties();// ObjectFactory.Get<IETaxiEntitiesContext>();
try
{
var _newSettlement = new Settlement
{
CreateDate = settlement.CreateDate,
Driver = settlement.Driver,
DriverID = settlement.DriverID,
Car = settlement.Car,
CarID = settlement.CarID,
DocPath = settlement.DocPath
};
foreach (var _shift in settlement.Shifts)
{
//var _sh = _context.Shifts.Find(_shift.ShiftID);
//_context.Entry(_sh).CurrentValues.SetValues(_shift);
_newSettlement.Shifts.Add(_shift);
}
foreach (var _cost in settlement.Costs)
{
////var _sh = _context.Costs.Find(_cost.CostID);
////_context.Entry(_sh).CurrentValues.SetValues(_cost);
_newSettlement.Costs.Add(_cost);
}
_context.Settlements.Add(_newSettlement);
_success = _context.SaveChanges() > 0;
}
catch (Exception ex)
{
throw new Exception(ex.Message);
}
return _success;
}
对此问题的任何帮助都将非常感谢。
以下是我如何将费用和转移添加到我的收藏中:
我在页面中创建了一个结算:
_settlement = new Settlement
{
CreateDate = DateTime.Now,
Driver = _driver,
DriverID = _driver.DriverID,
Car = _car,
CarID = _car.CarID,
DocPath = _path
};
然后当我创建一个包含2个独立网格视图中选定行的pdf文件时:
foreach (GridDataItem _selectedRow in gwShifts.MasterTableView.Items)
{
if (_selectedRow.Selected)
{
var _shift =
_diaryRepository.GetShiftByID((int) _selectedRow.GetDataKeyValue("ShiftID")).FirstOrDefault();
if (_shift != null)
{
_settlement.Shifts.Add(_shift);
_settlementData.Shifts.Add(_shift);
_settlementData.SplitPercentace = GetTemplateValue(_selectedRow, "lblSplit");
_settlementData.SettlementAmount = GetTemplateValue(_selectedRow, "lblSettlementAmount");
if (_settlementData.Shifts != null)
{
_tableShifts.AddCell(
new PdfPCell(
new Phrase(
_settlementData.Shifts.FirstOrDefault().ShiftDate.ToShortDateString(),
_bodyFont)) {Border = 0});
_tableShifts.AddCell(
new PdfPCell(
new Phrase(
string.Format("{0:c}", _settlementData.Shifts.FirstOrDefault().GrossAmount),
_bodyFont)) {Border = 0});
_tableShifts.AddCell(
new PdfPCell(
new Phrase(
string.Format("{0:c}", _settlementData.Shifts.FirstOrDefault().MoneyAmount),
_bodyFont)) {Border = 0});
_tableShifts.AddCell(new PdfPCell(new Phrase(_settlementData.SplitPercentace, _bodyFont))
{Border = 0});
_tableShifts.AddCell(
new PdfPCell(new Phrase(_settlementData.SettlementAmount, _boldTableFont))
{Border = 0});
_totalAmount.AddRange(new[]
{
Convert.ToInt32(
_settlementData.SettlementAmount.Replace(".", "").
Replace(",", "").Replace("kr", ""))
});
_settlementData.Shifts.Remove(_shift);
}
}
}
}
var _summaryCell =
new PdfPCell(new Phrase("Upphæð: " + string.Format("{0:c}", _totalAmount.Sum()), _boldTableFont))
{
Border = 0,
Colspan = 5,
HorizontalAlignment = Element.ALIGN_RIGHT,
Padding = 5,
BorderWidthTop = 1
};
_tableShifts.AddCell(_summaryCell);
if (_totalAmount.Count != 0)
_totalAmount.Clear();
}
您可以看到我如何将Shift添加到此结算实体:
var _shift =
_diaryRepository.GetShiftByID((int) _selectedRow.GetDataKeyValue("ShiftID")).FirstOrDefault();
if (_shift != null)
{
_settlement.Shifts.Add(_shift);
然后我把它发送到reporistory(参见上面的方法)
if(_driverRepository.CreateSettlement(_settlement))
{
SetMessage("Uppgjör hefur verið skapað og sent bílstjóra ef e-póstur er skráður á viðkomandi bílstjóra.", "Uppgjör skapað");
pnlSettlement.Visible = false;
pnlDocCreation.Visible = false;
pnlResult.Visible = false;
}
我还尝试简单地将param结算直接添加到上下文中但得到了类似的错误。
答案 0 :(得分:0)
我认为这与您填充Shifts
和Costs
集合的方式有关。您正在尝试添加已创建的记录(即已设置其主键值)以与新的Settlement
实体一起保存,但我相信实体框架不会尝试创建链接的新记录到您的新Settlement
实体,而不是按原样将它们保存到表中。在这种情况下,您确实会遇到多个实体具有相同主键的情况。
我会尝试以下操作(我只会使用Shifts
循环向您展示,但您也应该能够将其应用到Costs
循环中:
foreach (var _shift in settlement.Shifts)
{
var newShift = new Shift { /*Copy all of the values from _shift here*/ };
_newSettlement.Shifts.Add(_shift);
context.Shifts.Add(newShift);
}
如果这不起作用,我建议调试Costs
和Shifts
,以确保您在这些集合中没有任何重复项。
答案 1 :(得分:0)
如果您不想要新的Shifts
&amp; Costs
那么我只能假设你需要将现有的那些重新命名为新的Settlement
foreach (var shift in settlement.Shifts)
{
//either
shift.Settlement = newSettlement;
//or
shift.SettlementId = newSettlement.SettlementId;
//depending on your object model
}
我刚刚意识到我误解了这个问题。图中未显示2个附加表(成本和转移)。问题是尝试创建将结算与SettlementCost
连接起来的SettlementShift
和Costs\Shifts
个实体。
答案 2 :(得分:0)
好的,想出了一个“丑陋”的解决方案......但它已经解决了。 将模型更改为一个(结算) - &gt;许多(转变或成本)关系。
我创建了一个新的结算并将其保存到数据库中。 从每个数据库更新SettlementID中检索每个Shift和Cost,并将这些保存到DB。
try
{
var _newSettlement = new Settlement
{
CreateDate = settlement.CreateDate,
DriverID = settlement.DriverID,
CarID = settlement.CarID,
DocPath = settlement.DocPath
};
Add(_newSettlement);
_success = SaveWithSuccess() > 0;
var _settlement = GetAll().FirstOrDefault(x => x.SettlementID == _newSettlement.SettlementID);
if (_success)
{
foreach (var _shift in settlement.Shifts)
{
var _sh = _diaryRepository.GetShiftByID(_shift.ShiftID).FirstOrDefault();
_sh.SettlementID = _settlement.SettlementID;
_diaryRepository.UpdateShift(_sh);
}
foreach (var _cost in settlement.Costs)
{
var _ch = _costRepository.GetCostByID(_cost.CostID);
_ch.SettlementID = _settlement.SettlementID;
_costRepository.UpdateCost(_ch);
}
}
}
不是一个漂亮的,但它解决了这个问题。 我不担心DB请求加载..在这种情况下不会那么高。
我认为有一个更好的解决方案,但我此时无法找到它。
感谢您为帮助所做的一切努力:)