C# 无法确定etaxiDataModel关系的主体端。多个添加的实体可能具有相同的主键

C# 无法确定etaxiDataModel关系的主体端。多个添加的实体可能具有相同的主键,c#,entity-framework,entity-framework-5,C#,Entity Framework,Entity Framework 5,我已经与这个问题斗争了一段时间,现在无法解决这个问题。我读过多篇文章,尝试过几种不同的解决方案,但都没有效果。现在我觉得我已经停下来了,真的需要帮助 我正在使用EF 5+和DB first和edmx文件。我的数据库中有3个不同的表: 1.结算 2.成本 3.移位 结算在我的edmx文件中具有通过关联连接的成本和班次集合(带有链接表) 我需要在我的db中插入一个新的结算,并引用已经存在的成本和班次收款。 我试图插入的结算实体中包含的班次和成本包含所有相关数据,且这些数据均未以任何方式修改(与我从

我已经与这个问题斗争了一段时间,现在无法解决这个问题。我读过多篇文章,尝试过几种不同的解决方案,但都没有效果。现在我觉得我已经停下来了,真的需要帮助

我正在使用EF 5+和DB first和edmx文件。我的数据库中有3个不同的表: 1.结算 2.成本 3.移位

结算在我的edmx文件中具有通过关联连接的成本和班次集合(带有链接表)

我需要在我的db中插入一个新的结算,并引用已经存在的成本和班次收款。 我试图插入的结算实体中包含的班次和成本包含所有相关数据,且这些数据均未以任何方式修改(与我从db检索的数据相同)。 这里是我将实体插入数据库的方法

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;
    }
然后,当我使用两个独立的网格视图中的选定行创建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();
        }
在这里,您可以看到我如何将班次添加到此结算实体:

     var _shift =
                        _diaryRepository.GetShiftByID((int) _selectedRow.GetDataKeyValue("ShiftID")).FirstOrDefault();

                    if (_shift != null)
                    {
                        _settlement.Shifts.Add(_shift);
然后我将其发送到报告室(参见上面的方法)


我还试图简单地将param结算直接添加到上下文中,但得到了类似的错误

我认为这与您如何填充
班次
成本
集合有关。您正在尝试添加已创建的记录(即,它们已经设置了主键值)以与新的
结算
实体一起保存,但我相信实体框架不会尝试创建链接到新的
结算
实体的新记录,而是按原样将它们保存到表中。在这种情况下,您确实会遇到多个实体具有相同主键的情况

我将尝试以下方法(我将向您展示仅使用
移位
循环,但您应该能够将其应用于
成本
循环):


如果这不起作用,我建议调试
成本
班次
,以确保这些集合中没有任何重复项。

如果您不想要新的
班次
成本
,那么我只能假设您需要将现有的班次重新发布到新的
结算

foreach (var shift in settlement.Shifts)
{  
    //either
    shift.Settlement = newSettlement;
    //or
    shift.SettlementId = newSettlement.SettlementId;
    //depending on your object model
}

我刚刚意识到我误解了这个问题。图中没有显示另外两个表(成本和班次)。问题是试图创建
结算成本
结算转移
实体,将结算连接到
成本转移

好的,提出了一个“丑陋”的解决方案..但解决了。 将模型更改为一(结算)->多(班次或成本)关系

我创建了一个新的定居点,并将其保存到DB中。 从DB update SettlementID中检索每个班次和成本,并将其保存到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请求负载。在这种情况下,它不会那么高

我认为有一个更好的解决办法,但我当时没能找到


感谢您的帮助:)

对不起,这与问题无关(我认为这本身是好的:+1),但我看到了您在这里捕捉异常的做法,我想善意地建议,这样做可能并不理想。您正在捕获常规异常,然后使用原始异常的消息创建新的常规异常。这意味着您不仅会丢失异常类型,还会丢失原始异常的堆栈跟踪,这会使调试整个堆变得更加困难。在这种特殊情况下,您最好不要捕获任何内容。您好,kmp,我知道:)仍然//TODO异常处理。我只是把它放在那里,这样我在设置时就有了一些东西。在填充
Shift
Cost
集合之前,尝试将
newsetment
实例添加到上下文中。如果仍无法确定实例化的基础
成本
&
转移
集合的方式/时间/地点?很抱歉,无法确定“etaxidatmodel.FK_Car_Account”关系的主体端。多个添加的实体可能具有相同的主键。能否将其转过来
\u shift.Settlement=\u newSettlement
请确保将
newShift.SettlementId
设置为
newSettlement.SettlementId
这不会在我的数据库中创建新的班次吗?不起作用..我使用此方法时,班次(和成本)增加了一倍您是否遇到了与以前相同的错误?如果没有,则调试
班次
成本
集合,确保没有重复的班次和成本。我也从不改变那些实体。只需使用他们提供的数据。当我删除_newsetment.shift.Add(_shift)时;我的收藏中没有重复的。但是,我收到一条错误消息:无法确定“etaxidatmodel.FK_Shift_Account”关系的主体端。多个添加的实体可能具有相同的主键。所有班次都与帐户(ID)有关系,并且所选的所有班次都具有相同的帐户ID(这有关系吗?
newSettlement.SettlementId
不是选项:此时它是一个新对象。确定,当前状态:尝试此操作:_shift.Settlements=new[]{u newSettlement};然后I:_context.resolutions.Add(_newSettlement)_success=\u context.SaveChanges()>0;它将结算保存到DB中,但与班次或时间无关Cost@GertArnold您可能是对的,但我没有假设数据库生成的id's@qujck是的,可能是。Ingimar,我不认识你代码中qujck的建议。@Gert Arnold:_shift.resolutions=new[]{u newSettlement};因为我两边都有收藏品
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);
}
foreach (var shift in settlement.Shifts)
{  
    //either
    shift.Settlement = newSettlement;
    //or
    shift.SettlementId = newSettlement.SettlementId;
    //depending on your object model
}
 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);

                }
            }
        }