我有一个简单的雪花模式,我从中生成了我的 Entity Framework 模型。
问题是我试图将一个子实体映射到一个现有的父和/或祖 parent 实体,但它仍然插入它。
我关注了这个:
Insert new object with existing object
Prevent Entity Framework to Insert Values for Navigational Properties
有趣的是,即使父实体的 EntityState 是“未更改”的, Entity Framework 仍然会尝试插入它。
public void Save(Car car)
{
using (DBContext context = new DBContext())
{
// No need to save if it already exists
if ( context.Cars.FirstOrDefault(x => x.RegistrationNumber == car.RegistrationNumber) != null)
{
return;
}
else
{
// Check if the parent POCOs exist in the DB.
Model existingModel = context.Models.FirstOrDefault(x => x.Name == car.Model.Name);
Manufacturer existingManufacturer = context.Manufacturers.FirstOrDefault(x=> x.Name == car.Model.Manufacturer.Name)
Trader existingTrader = context.Traders.FirstOrDefault(x=> x.Name == car.Trader.Name)
TraderCompany existingTraderCompany = context.TraderCompanys.FirstOrDefault(x=> x.Name == car.Trader.TraderCompany.Name)
context.ContextOptions.LazyLoadingEnabled = false;
//Attach to the context if existing in the DB, i.e mark the existing POCOs not to be added the DB
if (existingModel != null)
{
car.Model = existingModel;
Assert.IsTrue(context.ObjectStateManager.GetObjectStateEntry(car.Model).State == EntityState.Unchanged);
}
if (existingManufacturer != null)
{
car.Model.Manufacturer = existingManufacturer;
Assert.IsTrue(context.ObjectStateManager.GetObjectStateEntry(car.Model.Manufacturer).State == EntityState.Unchanged);
}
if (existingTrader != null)
{
car.Trader = existingTrader;
Assert.IsTrue(context.ObjectStateManager.GetObjectStateEntry(car.Trader).State == EntityState.Unchanged);
}
if (existingTraderCompany != null)
{
car.Trader.TraderCompany = existingTraderCompany;
Assert.IsTrue(context.ObjectStateManager.GetObjectStateEntry(car.Trader.TraderCompany).State == EntityState.Unchanged);
}
//Mark the Car for Addition to the DB
context.Cars.AddObject(car);
context.ObjectStateManager.ChangeObjectState(car, EntityState.Added);
//If the POCOs do not exist in the DB mark them for addition
if (existingModel == null)
{
context.ObjectStateManager.ChangeObjectState(car.Model,EntityState.Added);
}
if (existingManufacturer == null)
{
context.ObjectStateManager.ChangeObjectState(car.Model.Manufacturer,EntityState.Added);
}
if (existingTrader == null)
{
context.ObjectStateManager.ChangeObjectState(car.Trader,EntityState.Added);
}
if (existingTraderCompany == null)
{
context.ObjectStateManager.ChangeObjectState(car.Trader.TraderCompany,EntityState.Added);
}
context.SaveChanges();
}
}
}
经过几天的修修补补,我设法想出了一个对我有用的解决方法。
传递给 CarRepository.Save() 的 Car 似乎有某种内部上下文,它无法检测... 既然如此,不可能将其从上下文中分离并将其添加到 CarRepository.Save() 中的上下文中.为了将其实际添加到此上下文中,我深度/延迟复制了 Car 对象及其导航属性(如果存在)。
public void Save(Car car)
{
using (DBContext context = new DBContext())
{
// No need to save if it already exists
if ( context.Cars
.Any(x => x.RegistrationNumber == car.RegistrationNumber))
{
return;
}
else
{
//Assign scalar properties to the deep copy
Car carToBeSaved = new Car
{
carToBeSaved.RegistrationNumber = car.RegistrationNumber,
carToBeSaved.Price = car.Price
}
//Car -> Trader -> ...
if(car.Trader != null)
{
Trader existingTrader =
context.Traders
.FirstOrDefault(x => x.Name == car.Trader.Name);
//If exists in DB assign, if not deep copy
carToBeSaved.Trader = existingTrader ??
new Trader
{
Name = car.Trader.Name,
JobTitle = car.Trader.JobTitle
}
//Car -> Trader -> TraderCompany
if(car.Trader.TraderCompany != null)
{
TraderCompany existingTraderCompany =
context.TradersCompanys
.FirstOrDefault(x => x.Name == car.Trader
.TraderCompany
.Name);
//If exists in DB assign, if not deep copy
carToBeSaved.Trader.TraderCompany = existingTraderCompany ??
new TraderCompany
{
Name = car.Trader.TraderCompany.Name,
Address = car.Trader.TraderCompany.Address,
PhoneNumber = car.Trader.TraderCompany.PhoneNumber
}
}
}
//Car -> Model -> ...
if(car.Model != null)
{
Model existingModel =
context.Models
.FirstOrDefault(x => x.Name == car.Model.Name);
//If exists in DB assign, if not deep copy
carToBeSaved.Model = existingModel ??
new Model
{
Name = car.Model.Name
}
//Car -> Model -> Manufacturer
if(car.Model.Manufacturer != null)
{
Manufacturer existingManufacturer =
context.Manufacturers
.FirstOrDefault(x => x.Name == car.Model
.Manufacturer
.Name);
//If exists in DB assign, if not deep copy
carToBeSaved.Model.Manufacturer = existingManufacturer ??
new Manufacturer
{
Name = car.Model.Manufacturer.Name
}
}
}
//Mark the Car for Addition to the DB
context.Cars.AddObject(car);
context.SaveChanges();
}
}
}
如果有人对此有任何想法,请分享。
谢谢。
最佳答案
我发现解决此类问题的最可靠方法是使用现有对象的 Id 而不是关联实体实例。因此,在找到现有交易员后,您可以像这样为交易员设置“外键”:
car.TraderId = existingTrader.Id;
有一段时间,这对我来说似乎是一种 hack,但在 2013 年 4 月的 MSDN 杂志中,我读到 Julie Lerman 也推荐这种方法。
关于c# - 父对象在 EntityState.Unchanged 中,但它仍然插入到数据库中,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13086922/
总的来说,我对ruby还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用
在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
我有一个表单,其中有很多字段取自数组(而不是模型或对象)。我如何验证这些字段的存在?solve_problem_pathdo|f|%>... 最佳答案 创建一个简单的类来包装请求参数并使用ActiveModel::Validations。#definedsomewhere,atthesimplest:require'ostruct'classSolvetrue#youcouldevencheckthesolutionwithavalidatorvalidatedoerrors.add(:base,"WRONG!!!")unlesss
好的,所以我的目标是轻松地将一些数据保存到磁盘以备后用。您如何简单地写入然后读取一个对象?所以如果我有一个简单的类classCattr_accessor:a,:bdefinitialize(a,b)@a,@b=a,bendend所以如果我从中非常快地制作一个objobj=C.new("foo","bar")#justgaveitsomerandomvalues然后我可以把它变成一个kindaidstring=obj.to_s#whichreturns""我终于可以将此字符串打印到文件或其他内容中。我的问题是,我该如何再次将这个id变回一个对象?我知道我可以自己挑选信息并制作一个接受该信
如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象
我在Rails工作并有以下类(class):classPlayer当我运行时bundleexecrailsconsole然后尝试:a=Player.new("me",5.0,"UCLA")我回来了:=>#我不知道为什么Player对象不会在这里初始化。关于可能导致此问题的操作/解释的任何建议?谢谢,马里奥格 最佳答案 havenoideawhythePlayerobjectwouldn'tbeinitializedhere它没有初始化很简单,因为你还没有初始化它!您已经覆盖了ActiveRecord::Base初始化方法,但您没有调
我有一个服务模型/表及其注册表。在表单中,我几乎拥有服务的所有字段,但我想在验证服务对象之前自动设置其中一些值。示例:--服务Controller#创建Action:defcreate@service=Service.new@service_form=ServiceFormObject.new(@service)@service_form.validate(params[:service_form_object])and@service_form.saverespond_with(@service_form,location:admin_services_path)end在验证@ser
有时我需要处理键/值数据。我不喜欢使用数组,因为它们在大小上没有限制(很容易不小心添加超过2个项目,而且您最终需要稍后验证大小)。此外,0和1的索引变成了魔数(MagicNumber),并且在传达含义方面做得很差(“当我说0时,我的意思是head...”)。散列也不合适,因为可能会不小心添加额外的条目。我写了下面的类来解决这个问题:classPairattr_accessor:head,:taildefinitialize(h,t)@head,@tail=h,tendend它工作得很好并且解决了问题,但我很想知道:Ruby标准库是否已经带有这样一个类? 最佳
如何在ruby中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL