Attach() if you have something detached
It is clear from the forums that this whole business of attaching detached objects in LINQ to SQL (DLinq) is confusing. Some of it is intrinsic, some is perhaps our design and perhaps a bit attributable to the misnomer "detached object support". More about the misnomer later but first here is the skinny on Attach() set of APIs.
Think of a DataContext instance as a happy universe of objects where the entities (or objects with unique keys) are known to the DataContext instance since they were retrieved using that instance and the unknown ones are simply new entities that need to be inserted. Unfortunately, this simple picture is complicated by the fact that in a multi-tier system, entities may be sent over the wire in an ASP.NET app or web service client and then they may need to be brought back to the mid-tier for update. Add to that the fact that the mid-tier is often stateless so the DataContext instance used to retrieve the entity from the database may be long gone. A new DataContext instance can be spun up for update but the new one does not know about the deserialized entity that it did not retrieve. Attach() solves this problem by telling the new DataContext instance that the attached entity is meant to be updated and should not be inserted into the database.
Attach() needs to preserve the optimistic concurrency capability since that is how concurrent change conflicts can be detected and handled. So Attach() needs to deal with current and original values. This is what gives rise to the different flavors.
- Original values used for conflict detection: db.Customers.Attach(originalCust) should be used to attach the original values. The instance can then be modified (playback) before calling SubmitChanges().
- Original and current copies available: db.Customers.Attach(currentCust, originalCust) does it in one shot. Of course, this requires two instances with original and current values respectively.
- Timestamp or no optimistic concurrency members: db.Products.Attach(currentProd) In this case, original values are not required so current entity instance is enough. There is no playback needed before calling SubmitChanges().
When it comes to "detached object support", a question I often get is - what do I call to detach an entity? The answer is nothing! If an entity is serialized and deserialized back, it is already detached from the DataContext instance used for retrieval. If not, it may have deferred loaders that are tethered to the original DataContext. More significantly, Attach() is not intended to enable movement of entities across DataContext instances in the same app domain. This is not the intent (more about why that's the case some other day). In fact as I have mentioned on the LINQ forum, this is likely to cause an exception sooner or later so we have modified Attach() to throw if it is used on an object that is still attached to some DataContext instance. This change was done after beta2 and should be visible in RTM.
Detachedly yours,
Dinesh
Comments
Anonymous
October 08, 2007
PingBack from http://www.artofbam.com/wordpress/?p=6318Anonymous
October 08, 2007
Hi Dinesh, I agree that the whole notion of attaching/detaching entity instances is confusing - especially so in a multi-tier scenario. Maybe this calls for a rethink of the method names? What if the "Attach" method overloads that take 2 parameters were renamed to a "Merge" method? Since this is closer to the semantics to what is actually taking place. The "Attach" overload that takes a single parameter still makes sense since you want associate an instance with a DataContext (for playback). Regards, -krishAnonymous
October 08, 2007
It is clear from the forums that this whole business of attaching detached objects in LINQ to SQL (DLinq)Anonymous
October 08, 2007
Dinesh Kulkarni, program manager on the LINQ to SQL project, presents the logic behind LINQ to SQL'sAnonymous
October 08, 2007
Dinesh Kulkarni, program manager on the LINQ to SQL project, presents the logic behind LINQ to SQL'sAnonymous
October 08, 2007
Dinesh Kulkarni, program manager on the LINQ to SQL project, presents the logic behind LINQ to SQL'sAnonymous
October 09, 2007
Dinesh, So how does Attach 'know' that an entity was attached to a different DataContext if the entity itself has no state (it's derived from Object after all)? The issue that comes up is if you pass object across TIERS (not necessarily physical tiers) where there maybe multiple DataContexts on different business objects working with entities. If you can deal with deserialized copies, then why make a distinction between that and a 'formerly attached' object from a different context? Surely it can't be difficult to dump original object context whatever that may be. It's just very inconsistent of the API. You're basically forcing us as developers to go through girations to make this happen (say manually serialize/deserialize). Maybe you could do another post and demonstrate how you envision how to do the .Attach() with the current and original values including describing some scenarios where the original values come from. One more thing: Currently you can get .Attach to work without a second original values instance if the entity/table has a timestamp. Is that behavior still there?Anonymous
October 09, 2007
Rick, Two key points about working with detached objects
- There are actually deferred loaders in EntityRef/EntitySets if deferred loading is not turned off. If it is, then the objects are effectively detached.
- It is possible for us to forcibly remove the deferred loaders. I am not convinced, we want to particularly support this pattern and until then at least, we would rather point out the lack of support than let you incorrectly assume that this is the intent. This is worth a post. Your comment about showing the common pattern with Attach() is on the mark. I will add it to my list (or maybe Matt's :-) ) Attach() w/o second original instance did and will continue to work for timestamp-based opt concurrency. Dinesh
Anonymous
October 11, 2007
Krish, Merge() has a certain legacy semantics with DataSet. That is really collection level semantics. So that really wasn't a candidate. Also, the goal is to bring an object not known to the DataContext into the context. In ORM parlance (e.g. hibernate, EJB and in our world ObjectSpaces), this has long been called detached object support so ORMers often ask about it. That is where the name Attach() comes from. We didn't see much value in breaking from this precedent.Anonymous
October 13, 2007
In a scenario where an application wants to allow different data providers, is it possible to use the entities generated by the designer and let each provider attach them. In the case of the SqlServer provider of course the DataContext would be the one managing the entites and we are back to the case without providers. The scenario I am thinking of is: A generic business object: BO<TEntity> with among other things basic CRUD methods, that delegates to each provider to implement these methods. The providers return IEnumerables so that the UI does not query the DB directly.Anonymous
October 14, 2007
Welcome to the thirty-third edition of Community Convergence. This week we have a new video called ProgrammingAnonymous
October 14, 2007
The comment has been removedAnonymous
October 21, 2007
I am curious about the third flavor of Attach: db.Products.Attach(currentProd) What happens if there is an old version of the product instance sitting in the DataContext? Will playback happen, i.e will all the changes that are present in the currentProd be applied to the DataContext? I am thiking of a scenario where the DataContext state is saved in the session (I know this is frowned upon, but I am thinking of using someting like this is in a serious enterprise app where application richness is important and there are not that many users) and the object is serialized and then deserialzed, and I want to attach the current version of the instance back to the DataContext. Thanks, JayAnonymous
October 22, 2007
Jay, Attach() will fail if an object with matching object identity (same unique key values) exists in the DataContext. In your example above, DataContext state could be saved but it will have to be restored to a new DataContext instance. At that point, flavor #2 would be appropriate to use instead of #3. DineshAnonymous
October 22, 2007
Thanks Dinesh for your prompt response. Followup questions:
- I am not sure if I follow your recommendation to restore the old DataContext state to a new DataContext. Is that a datacontext copy of some sort? How does that cicumvent the matching identity issue. Wouldn't the 'restored' datacontext have an object with the same identity as the object we are trying to attach?
- The larger problem I am trying to address, and you might be able to point me in the right direction, is how do you work with a datacontext across multiple request/response cycles without hitting the database. This would be useful if you are doing a series of actions with multiple postbacks (and heavy invocatoin of methods in the model during every postback), but want to treat the entire workflow as a unit and want to commit/rollback the entire unit of work.
- I guess, I could loop through all the properties of the old instance in the DataContext and modify it to the current values im my modified instance. Wouldn't it be useful to have a flavor of Attach (or some other DataContext method) which would do this? Thanks, Jay
Anonymous
November 21, 2007
LINQ to SQL: Table<T>.Detach Method Does Not Exist Moving from Visual Studio 2008 Beta 2 to VisualAnonymous
May 26, 2008
The common guidance for updates is to keep the original values in view state so that you can recreateAnonymous
June 01, 2008
The common guidance for updates is to keep the original values in view state so that you can recreateAnonymous
July 01, 2008
Back to the "tips" series after a little break ... One common question I get is about caching of data