views:

169

answers:

3

Is there any point in keeping track of the classic bool disposed field on an otherwise threadsafe type for the purposes of conditionally throwing an ObjectDisposedException at the beginning of all primary exposed methods?

I've seen this pattern recommended in a few places online but I'm not sure if the authors are using it correctly, so this question assumes that they are.

In such a scenario, it seems that the only way to ensure that the disposed condition is true beyond the condition's evaluation is to use a synchronization mechanism such as lock() over the entire body of each exposed member including the Dispose(bool) method. Wouldn't this make the type effectively single-threaded again?

And if this is true, then there'd be no point in using it, and therefore you can't rely on the ObjectDisposedException mechanism in some IDisposable implementations - so then why would we EVER employ this mechanism if it isn't necessary?

====

I guess IDisposable and ObjectDisposedException just don't go together for thread-safe types.

+3  A: 

Perhaps a more efficient way of making a threadsafe object not become disposed while a method is running is to use a ReaderWriterLockSlim. Have all the public methods obtain a read lock while executing and release it when they're done. Have Dispose obtain a writer lock. It will wait till all the other methods are done before it obtains it's write lock. It then sets isDisposed inside the write lock which it exclusively holds. Any calls to public methods after Dispose is done can see isDisposed and throw the ObjectDisposedException.

ReaderWriterLockSlim

http://msdn.microsoft.com/en-us/library/system.threading.readerwriterlockslim.aspx

Sam
Cool idea. Got a sample? Not finding documentation on ReaderWriterSlimLock. Is this something you've seen before or something you've just brainstormed?
uosɐſ
Just brainstorming, but I think it should work well for your request. I misspelled it in my original answer, it's `ReaderWriterLockSlim`. I added a link to MSDN docs.
Sam
Actually, that almost works, except that FXcop rightly points out that ReaderWriterLockSlim implements IDisposable and should be disposed along with the rest of the object's resources. So once dispose is called, I would no longer be able to get a read lock in order to test if the object has been disposed or not.
uosɐſ
I don't consider this question closed, but this was a good attempt and I don't want to leave the question open.
uosɐſ
+2  A: 

If the behavior of your object will be different if it is already disposed, and if it's likely that it will be used after it's been disposed, then you need to keep track of this. It's better to throw ObjectDisposedException than to throw whatever random exception that will occur if the object is already disposed and you don't check first.

John Saunders
Yes, but this question is about how to throw an ObjectDisposedException with certainty in a multi-threaded scenario.
uosɐſ
@Jason: in that case, edit your question to say so.
John Saunders
A: 

Given that the "Disposed" boolean is only updated in one place, and it is a bug in your caller to use the object after they had called Disposed.

I think it is good enough to throw ObjectDisposedException "most of the time" after Dispose has been called. I see ObjectDisposedException as being a debug helper not thingthing that a caller should be catching.

Ian Ringrose