1

在我正在处理的一个项目中,我需要从我的数据库中删除一个“用户”。这个“用户”有两个引用它的外键的表。硬删除时,我试图从表 A 和表 B 中删除所有对“用户”具有外键的记录,然后删除该“用户”记录。这一切都是在存储库中并使用对象工厂完成的。

代码如下:

public void RemoveUserByUserId(int userId)
{
var user = m_context.User.GetByKey(userId);

ObjectFactory.Inject(m_context);

UserTokenRepository.RemoveUserTokensByUserId(userId);
UserMappingRepository.RemoveUserMappingsByUserId(userId);

m_context.User.DeleteOnSubmit(user);

m_context.SubmitChanges();

ObjectFactory.ResetDefaults();
}

public static void RemoveUserTokensByUserId(int userId)
{
var dataContext = ObjectFactory.GetInstance<DataContext>();
var userTokens = dataContext.UserToken.ByUserId(userId);
dataContext.UserToken.DeleteAllOnSubmit(userTokens.AsEnumerable());
}

public static void RemoveUserMappingsByUserId(int userId)
{
var dataContext = ObjectFactory.GetInstance<DataContext>();
var userMappings= dataContext.UserMapping.ByUserId(userId);
dataContext.UserMapping.DeleteAllOnSubmit(userMappings.AsEnumerable());
}

如果每个表都有一条记录,则可以正常工作。如果一个表有多个,这只能发生在 UserToken 和 UserMappings 上,我会收到以下错误。

System.ArgumentException was unhandled by user code
Message=An item with the same key has already been added.
  Source=mscorlib
  StackTrace:
       at System.ThrowHelper.ThrowArgumentException(ExceptionResource resource)
       at System.Collections.Generic.Dictionary`2.Insert(TKey key, TValue value, Boolean add)
       at System.Data.Linq.ChangeProcessor.EdgeMap.Add(MetaAssociation assoc, TrackedObject from, TrackedObject to)
       at System.Data.Linq.ChangeProcessor.BuildEdgeMaps()
       at System.Data.Linq.ChangeProcessor.SubmitChanges(ConflictMode failureMode)
       at System.Data.Linq.DataContext.SubmitChanges(ConflictMode failureMode)
       at XXXX.XXXX.XXXX.XXXXDataContext.SubmitChanges(ConflictMode failureMode) in XXXX:line 519
       at System.Data.Linq.DataContext.SubmitChanges()
       at XXXX.UserRepository.RemoveUserByUserId(Int32 userId) in XXXX:line 146
       at XXXX(Int32 profileUserId) in XXXX:line 948
       at XXXX(Int32 profileUserId) in XXXX:line 165
       at SyncInvokeUnregisterUserForActivationFailed(Object , Object[] , Object[] )
       at System.ServiceModel.Dispatcher.SyncMethodInvoker.Invoke(Object instance, Object[] inputs, Object[]& outputs)
       at System.ServiceModel.Dispatcher.DispatchOperationRuntime.InvokeBegin(MessageRpc& rpc)

出于隐私原因,我不得不输入 XXXX。我读到的内容是,但是 LINQ 正在存储为 SQL 准备的更改,它会将其放入字典中,并且不知何故,两件事的关键是相同的,并且在更改发生之前它会出错。

任何帮助将非常感激。

4

2 回答 2

0

我有点在黑暗中射击,并假设您如何阅读它是正确的,以及我在此过程中遇到的一点痛苦。

您要从中删除的表是否有自己的主键,或者您是否使用外键作为 pk?(你知道我的意思......听起来他们没有自己独立的唯一主键,linq 知道)'

linq 使用 GetHash 方法从主键字段构造唯一 id .. 我猜这些字段重复,因此字典问题...问题不是由 2 个具有相同键 id 的不同表引起的。

我猜这些包含多个项目的表是一种在多对多连接的中间。一个快速而肮脏的解决方法是向它们添加一个自动增量列并将其分配给主键(否则尝试使用多字段 pk - 我还没有真正尝试过 linq deliberatley)。

我希望这能给您一些想法...您可以在部分类中覆盖 GetHashCode 以另一种方式解决您的问题...但是您应该只包含在单个项目中不变的数据部分)

于 2010-07-15T18:14:55.600 回答
0

这是一个老问题,但是我们永远无法同时对两个表进行此删除。我们的解决方案是先删除一个,然后再删除另一个,并在两者之间提交更改。

于 2012-02-16T21:30:03.283 回答