Dispose pattern: Difference between revisions
→Motivation: elab, narrowly about exceptions |
Fgnievinski (talk | contribs) |
||
(32 intermediate revisions by 20 users not shown) | |||
Line 1: | Line 1: | ||
{{short description|Software design pattern in which resources held by objects can be explicitly released}} |
|||
{{redirect|Dispose||Disposal (disambiguation)}} |
|||
{{redirect|Dispose|the music album|Dispose (album)|other uses|Disposal (disambiguation)}} |
|||
{{Refimprove|date=February 2013}} |
{{Refimprove|date=February 2013}} |
||
In [[computer programming]], the '''dispose pattern''' is a [[design pattern (computer science)|design pattern]] which is used to handle resource cleanup and prevent [[resource leak]]s in [[runtime environment]]s that use [[automatic garbage collection]]. The fundamental problem that the dispose pattern aims to solve is that, because objects in a garbage-collected environment have [[finalizer]]s rather than [[destructor (computer science)|destructors]], there is no guarantee that an object will be destroyed at any deterministic point in time. The dispose pattern works around this by giving an object a [[method (computer science)|method]] (usually called <code>Dispose</code> or similar) which frees any resources the object is holding onto. |
|||
In [[object-oriented programming]], the '''dispose pattern''' is a [[design pattern (computer science)|design pattern]] for [[resource management (computing)|resource management]]. In this pattern, a [[system resource|resource]] is held by an [[object (computing)|object]], and released by calling a conventional [[method (computer science)|method]] – usually called <code>close</code>, <code>dispose</code>, <code>free</code>, <code>release</code> depending on the language – which releases any resources the object is holding onto. Many [[programming language]]s offer [[language construct]]s to avoid having to call the dispose method explicitly in common situations. |
|||
Many garbage-collected languages offer language constructs to avoid having to call the dispose method explicitly in many situations. These language constructs leads to results similar to what is obtained with the [[Resource Acquisition Is Initialization]] (RAII) idiom in languages with deterministic memory management (e.g. [[C++]]). |
|||
The dispose pattern is primarily used in languages whose [[runtime environment]] have [[automatic garbage collection]] (see motivation below). |
|||
== Exceptions == |
|||
== Motivation == |
|||
It is very common to write code similar to the listing below when using resources that might throw exceptions in garbage-collected languages: |
|||
=== Wrapping resources in objects === |
|||
Wrapping resources in objects is the object-oriented form of [[Encapsulation (computer programming)|encapsulation]], and underlies the dispose pattern. |
|||
Resources are typically represented by [[Handle (computing)|handles]] (abstract references), concretely usually integers, which are used to communicate with an external system that provides the resource. For example, files are provided by the [[operating system]] (specifically the [[file system]]), which in many systems represents open files with a [[file descriptor]] (an integer representing the file). |
|||
<source lang="csharp"> |
|||
// Attempt to acquire the resource. |
|||
These handles can be used directly, by storing the value in a variable and passing it as an argument to functions that use the resource. However, it is frequently useful to abstract from the handle itself (for example, if different operating systems represent files differently), and to store additional auxiliary data with the handle, so handles can be stored as a field in a [[Record (computer science)|record]], along with other data; if this in an [[opaque data type]], then this provides [[information hiding]] and the user is abstracted from the actual representation. |
|||
// May fail, in which case throws exception and the try...finally statement is not executed. |
|||
For example, in [[C file input/output]], files are represented by objects of the <code>FILE</code> type (confusingly called "[[file handle]]s": these are a language-level abstraction), which stores an (operating system) handle to the file (such as a [[file descriptor]]), together with auxiliary information like I/O mode (reading, writing) and position in the stream. These objects are created by calling <code>[[C file input/output#fopen|fopen]]</code> (in object-oriented terms, a [[Constructor_(object-oriented_programming)|constructor]]), which acquires the resource and returns a pointer to it; the resource is released by calling <code>[[C file input/output#fclose|fclose]]</code> on a pointer to the <code>FILE</code> object.<ref>{{man|bd|stdio.h|SUS}}</ref> In code: |
|||
<syntaxhighlight lang="c"> |
|||
FILE *f = fopen(filename, mode); |
|||
// Do something with f. |
|||
fclose(f); |
|||
</syntaxhighlight> |
|||
Note that <code>fclose</code> is a function with a <code>FILE *</code> parameter. In object-oriented programming, this is instead an [[instance method]] on a file object, as in Python: |
|||
<syntaxhighlight lang="python"> |
|||
f = open(filename) |
|||
# Do something with f. |
|||
f.close() |
|||
</syntaxhighlight> |
|||
This is precisely the dispose pattern, and only differs in syntax and code structure{{efn|In [[class-based programming]], methods are defined in a class, using an implicit <code>this</code> or <code>self</code> parameter, rather than as functions taking an explicit parameter.}} from traditional file opening and closing. Other resources can be managed in exactly the same way: being acquired in a constructor or factory, and released by an explicit <code>close</code> or <code>dispose</code> method. |
|||
=== Prompt release === |
|||
The fundamental problem that freeing resources aims to solve is that resources are expensive (for example, there may be a limit on the number of open files), and thus should be released promptly. Further, some finalization work is sometimes needed, particularly for I/O, such as flushing buffers to ensure that all data is actually written. |
|||
If a resource is unlimited or effectively unlimited, and no explicit finalization is necessary, it is not important to release it, and in fact short-lived programs often do not explicitly release resources: due to short run time, they are unlikely to exhaust resources, and they rely on the [[runtime system]] or [[operating system]] to do any finalization. |
|||
However, in general resources must be managed (particularly for long-lived programs, programs that use many resources, or for safety, to ensure that data is written out). Explicit disposal means that resource finalization and release is deterministic and prompt: the <code>dispose</code> method does not complete until these are done. |
|||
An alternative to requiring explicit disposal is to tie resource management to [[object lifetime]]: resources are acquired during [[object creation]], and released during [[object destruction]]. This approach is known as the [[Resource Acquisition Is Initialization]] (RAII) idiom, and is used in languages with deterministic memory management (e.g. [[C++]]). In this case, in the example above, the resource is acquired when the file object is created, and when the scope of the variable <code>f</code> is exited, the file object that <code>f</code> refers to is destroyed, and as part of this, the resource is released. |
|||
RAII relies on object lifetime being deterministic; however, with automatic memory management, [[object lifetime]] is not a concern of the programmer: objects are destroyed at some point after they are no longer used, but ''when'' is abstracted. Indeed, lifetime is often not deterministic, though it may be, notably if [[reference counting]] is used. Indeed, in some cases there is no guarantee that objects will ''ever'' be finalized: when the program terminates, it may not finalize the objects, and instead just let the operating system reclaim memory; if finalization is required (e.g., to flush buffers), data loss can occur. |
|||
Thus by not coupling resource management to object lifetime, the dispose pattern allows ''resources'' to be released promptly, while giving implementation flexibility for memory management. The cost of this is that resources must be managed manually, which can be tedious and error-prone. |
|||
== Early exit == |
|||
A key problem with the dispose pattern is that if the <code>dispose</code> method is not called, the resource is leaked. A common cause of this is early exit from a function, due to an early return or exception. |
|||
For example: |
|||
<syntaxhighlight lang="python"> |
|||
def func(filename): |
|||
f = open(filename) |
|||
if a: |
|||
return x |
|||
f.close() |
|||
return y |
|||
</syntaxhighlight> |
|||
If the function returns at the first return, the file is never closed and the resource is leaked. |
|||
<syntaxhighlight lang="python"> |
|||
def func(filename): |
|||
f = open(filename) |
|||
g(f) # Do something with f that may raise an exception. |
|||
f.close() |
|||
</syntaxhighlight> |
|||
If the intervening code raises an exception, the function exits early and the file is never closed, so the resource is leaked. |
|||
Both of these can be handled by a <code>try...finally</code> construct, which ensures that the finally clause is always executed on exit: |
|||
<syntaxhighlight lang="python"> |
|||
def func(filename): |
|||
try: |
|||
f = open(filename) |
|||
# Do something. |
|||
finally: |
|||
f.close() |
|||
</syntaxhighlight> |
|||
More generically: |
|||
<syntaxhighlight lang="csharp"> |
|||
Resource resource = getResource(); |
Resource resource = getResource(); |
||
try { |
try { |
||
// Resource has been acquired. |
// Resource has been acquired; perform actions with the resource. |
||
// Perform actions with the resource. |
|||
... |
... |
||
} finally { |
} finally { |
||
// Release resource, even if an exception |
// Release resource, even if an exception was thrown. |
||
resource.dispose(); |
resource.dispose(); |
||
} |
} |
||
</syntaxhighlight> |
|||
</source> |
|||
The <code>try...finally</code> construct is necessary for proper [[exception safety]], since the <code>finally</code> block enables execution of cleanup logic regardless of if an exception is thrown or not in the <code>try</code> block. |
The <code>try...finally</code> construct is necessary for proper [[exception safety]], since the <code>finally</code> block enables execution of cleanup logic regardless of if an exception is thrown or not in the <code>try</code> block. |
||
One disadvantage of this approach is that it requires the programmer to explicitly add cleanup code in a <code>finally</code> block. This leads to code size bloat, and failure to do so will lead to resource leakage in the program. |
One disadvantage of this approach is that it requires the programmer to explicitly add cleanup code in a <code>finally</code> block. This leads to code size bloat, and failure to do so will lead to resource leakage in the program. |
||
In more complex situations, one or several resource may be acquired, in some combination, and correctly releasing resources in all cases can be very verbose. For example, because the dispose method may itself throw an exception, the <code>try...finally</code> statements must in principle be nested. |
|||
<source lang="csharp"> |
|||
Resource resource = null; |
|||
Resource resource2 = null; |
|||
try { |
|||
if (condition) { |
|||
// Attempt to acquire the resource. |
|||
resource = getResource(); |
|||
} |
|||
try { |
|||
resource2 = getResource(); |
|||
// Perform actions with the resources. |
|||
... |
|||
} finally { |
|||
// Resource might not have been acquired, or already freed |
|||
if (resource2 != null) |
|||
resource2.dispose(); |
|||
} |
|||
} finally { |
|||
// Resource might not have been acquired, or already freed |
|||
if (resource != null) |
|||
resource.dispose(); |
|||
} |
|||
</source> |
|||
== Language constructs == |
== Language constructs == |
||
To make the dispose pattern less verbose, several languages have some kind of built-in support for resources held and released in the same [[Block (programming)|block of code]]. |
To make the safe use of the dispose pattern less verbose, several languages have some kind of built-in support for resources held and released in the same [[Block (programming)|block of code]]. |
||
The [[C Sharp (programming language)|C#]] language features the <code>using</code> statement |
The [[C Sharp (programming language)|C#]] language features the <code>using</code> statement<ref>Microsoft MSDN: [http://msdn.microsoft.com/en-us/library/yh598w02.aspx using Statement (C# Reference)]</ref> that automatically calls the <code>Dispose</code> method on an object that implements the <code>IDisposable</code> [[interface (computer science)|interface]]: |
||
< |
<syntaxhighlight lang="csharp"> |
||
using (Resource resource = GetResource()) |
using (Resource resource = GetResource()) |
||
{ |
{ |
||
Line 64: | Line 105: | ||
... |
... |
||
} |
} |
||
</syntaxhighlight> |
|||
</source> |
|||
which is equal to: |
which is equal to: |
||
< |
<syntaxhighlight lang="csharp"> |
||
Resource resource = GetResource() |
Resource resource = GetResource() |
||
try |
try |
||
Line 79: | Line 120: | ||
((IDisposable)resource).Dispose(); |
((IDisposable)resource).Dispose(); |
||
} |
} |
||
</syntaxhighlight> |
|||
</source> |
|||
Similarly, the [[Python (programming language)|Python]] language has a <code>with</code> statement that can be used to similar effect with a ''context manager'' object. The ''context manager protocol'' requires implementing <code>__enter__</code> and <code>__exit__</code> methods which get automatically called by the <code>with</code> statement construct, to prevent duplication of code that would otherwise occur with the <code>try</code>/<code>finally</code> pattern.<ref>{{cite web |author=[[Guido van Rossum]], Nick Coghlan |date= |
Similarly, the [[Python (programming language)|Python]] language has a <code>with</code> statement that can be used to similar effect with a ''context manager'' object. The ''context manager protocol'' requires implementing <code>__enter__</code> and <code>__exit__</code> methods which get automatically called by the <code>with</code> statement construct, to prevent duplication of code that would otherwise occur with the <code>try</code>/<code>finally</code> pattern.<ref>{{cite web |author=[[Guido van Rossum]], Nick Coghlan |date=13 June 2011 |title=PEP 343: The "with" Statement |publisher=Python Software Foundation |url=http://legacy.python.org/dev/peps/pep-0343/ }}</ref> |
||
< |
<syntaxhighlight lang="python"> |
||
with resource_context_manager() as resource: |
with resource_context_manager() as resource: |
||
# Perform actions with the resource. |
# Perform actions with the resource. |
||
Line 88: | Line 129: | ||
# Perform other actions where the resource is guaranteed to be deallocated. |
# Perform other actions where the resource is guaranteed to be deallocated. |
||
... |
... |
||
</syntaxhighlight> |
|||
</source> |
|||
The [[Java (programming language)|Java]] language introduced a new syntax called <code>try</code>-with-resources in Java version 7.<ref>Oracle Java tutorial: [http://docs.oracle.com/javase/tutorial/essential/exceptions/tryResourceClose.html The try-with-resources Statement]</ref> It can be used on objects that implement the AutoCloseable interface (that defines method close()): |
The [[Java (programming language)|Java]] language introduced a new syntax called <code>try</code>-with-resources in Java version 7.<ref>Oracle Java tutorial: [http://docs.oracle.com/javase/tutorial/essential/exceptions/tryResourceClose.html The try-with-resources Statement]</ref> It can be used on objects that implement the AutoCloseable interface (that defines method close()): |
||
< |
<syntaxhighlight lang="java"> |
||
try ( |
try (OutputStream x = new OutputStream(...)) { |
||
// |
// Do something with x |
||
} catch(IOException ex){ |
} catch (IOException ex) { |
||
// |
// Handle exception |
||
// The resource x is automatically closed |
// The resource x is automatically closed |
||
} // try |
} // try |
||
</syntaxhighlight> |
|||
</source> |
|||
== Problems == |
|||
Beyond the key problem of correct resource management in the presence of returns and exceptions, and heap-based resource management (disposing objects in a different scope from where they are created), there are many further complexities associated with the dispose pattern. These problems are largely avoided by [[RAII]]. However, in common simple use these complexities do not arise: acquire a single resource, do something with it, automatically release it. |
|||
A fundamental problem is that having a resource is no longer a [[class invariant]] (the resource is held from object creation until it is disposed, but the object is still live at this point), so the resource may not be available when the object tries to use it, for example trying to read from a closed file. This means that all methods on the object that use the resource potentially fail, concretely usually by returning an error or raising an exception. In practice this is minor, as use of resources can usually fail for other reasons as well (for example, trying to read past the end of a file), so these methods already might fail, and not having a resource just adds another possible failure. A standard way to implement this is to add a boolean field to the object, called <code>disposed</code>, which is set to true by <code>dispose</code>, and checked by a [[guard clause]] to all methods (that use the resource), raising an exception (such as <code>ObjectDisposedException</code> in .NET) if the object has been disposed.<ref name="msdn_dispose">{{cite web |url=https://msdn.microsoft.com/en-us/library/b1yfkh5e(v=vs.110).aspx |title=Dispose Pattern}}</ref> |
|||
Further, it is possible to call <code>dispose</code> on an object more than once. While this may indicate a programming error (each object holding a resource must be disposed ''exactly'' once), it is simpler, more robust, and thus usually preferable for <code>dispose</code> to be [[idempotent]] (meaning "calling multiple times is the same as calling once").<ref name="msdn_dispose"/> This is easily implemented by using the same boolean <code>disposed</code> field and checking it in a guard clause at the start of <code>dispose</code>, in that case returning immediately, rather than raising an exception.<ref name="msdn_dispose"/> Java distinguishes disposable types (those that implement [https://docs.oracle.com/javase/8/docs/enwiki/api/java/lang/AutoCloseable.html AutoCloseable]) from disposable types where dispose is idempotent (the subtype [https://docs.oracle.com/javase/8/docs/enwiki/api/java/io/Closeable.html Closeable]). |
|||
Disposal in the presence of inheritance and composition of objects that hold resources have analogous problems to destruction/finalization (via destructors or finalizers). Further, since the dispose pattern usually does not have language support for this, [[boilerplate code]] is necessary. Firstly, if a derived class overrides a <code>dispose</code> method in the base class, the overriding method in the derived class generally needs to call the <code>dispose</code> method in the base class, in order to properly release resources held in the base. Secondly, if an object has a "has a" relationship with another object that holds a resource (i.e., if an object indirectly uses a resource through another object that directly uses a resource), should the indirectly using object be disposable? This corresponds to whether the relationship is ''owning'' ([[object composition]]) or ''viewing'' ([[object aggregation]]), or even just ''communicating'' ([[association (object-oriented programming)|association]]), and both conventions are found (indirect user is responsible for the resource or is not responsible). If the indirect use is responsible for the resource, it must be disposable, and dispose the owned objects when it is disposed (analogous to destroying or finalizing owned objects). |
|||
Composition (owning) provides [[Encapsulation (computer programming)|encapsulation]] (only the object that is used needs to be tracked), but at the cost of considerable complexity when there are further relationships between objects, while aggregation (viewing) is considerably simpler, at the cost of lacking encapsulation. In [[.NET Framework|.NET]], convention is to only have direct user of resources be responsible: "You should implement IDisposable only if your type uses unmanaged resources directly."<ref name="idisposable">{{cite web |url=https://learn.microsoft.com/en-us/dotnet/fundamentals/runtime-libraries/system-idisposable#implement-idisposable |title=IDisposable Interface |accessdate=2024-12-09}}</ref> See [[Resource management (computing)|resource management]] for details, and further examples. |
|||
== See also == |
== See also == |
||
* [[Object lifetime]] |
* [[Object lifetime]] |
||
* [[Resource Acquisition Is Initialization]] (RAII) |
* [[Resource Acquisition Is Initialization]] (RAII) |
||
==Notes== |
|||
{{notelist}} |
|||
==References== |
==References== |
||
{{reflist}} |
{{reflist}} |
||
==Further reading== |
|||
* Microsoft Developer Network: [https://msdn.microsoft.com/en-us/library/b1yfkh5e(v=vs.110).aspx Dispose Pattern] |
|||
[[Category:Object-oriented programming]] |
[[Category:Object-oriented programming]] |
Latest revision as of 17:41, 5 January 2025
This article needs additional citations for verification. (February 2013) |
In object-oriented programming, the dispose pattern is a design pattern for resource management. In this pattern, a resource is held by an object, and released by calling a conventional method – usually called close
, dispose
, free
, release
depending on the language – which releases any resources the object is holding onto. Many programming languages offer language constructs to avoid having to call the dispose method explicitly in common situations.
The dispose pattern is primarily used in languages whose runtime environment have automatic garbage collection (see motivation below).
Motivation
[edit]Wrapping resources in objects
[edit]Wrapping resources in objects is the object-oriented form of encapsulation, and underlies the dispose pattern.
Resources are typically represented by handles (abstract references), concretely usually integers, which are used to communicate with an external system that provides the resource. For example, files are provided by the operating system (specifically the file system), which in many systems represents open files with a file descriptor (an integer representing the file).
These handles can be used directly, by storing the value in a variable and passing it as an argument to functions that use the resource. However, it is frequently useful to abstract from the handle itself (for example, if different operating systems represent files differently), and to store additional auxiliary data with the handle, so handles can be stored as a field in a record, along with other data; if this in an opaque data type, then this provides information hiding and the user is abstracted from the actual representation.
For example, in C file input/output, files are represented by objects of the FILE
type (confusingly called "file handles": these are a language-level abstraction), which stores an (operating system) handle to the file (such as a file descriptor), together with auxiliary information like I/O mode (reading, writing) and position in the stream. These objects are created by calling fopen
(in object-oriented terms, a constructor), which acquires the resource and returns a pointer to it; the resource is released by calling fclose
on a pointer to the FILE
object.[1] In code:
FILE *f = fopen(filename, mode);
// Do something with f.
fclose(f);
Note that fclose
is a function with a FILE *
parameter. In object-oriented programming, this is instead an instance method on a file object, as in Python:
f = open(filename)
# Do something with f.
f.close()
This is precisely the dispose pattern, and only differs in syntax and code structure[a] from traditional file opening and closing. Other resources can be managed in exactly the same way: being acquired in a constructor or factory, and released by an explicit close
or dispose
method.
Prompt release
[edit]The fundamental problem that freeing resources aims to solve is that resources are expensive (for example, there may be a limit on the number of open files), and thus should be released promptly. Further, some finalization work is sometimes needed, particularly for I/O, such as flushing buffers to ensure that all data is actually written.
If a resource is unlimited or effectively unlimited, and no explicit finalization is necessary, it is not important to release it, and in fact short-lived programs often do not explicitly release resources: due to short run time, they are unlikely to exhaust resources, and they rely on the runtime system or operating system to do any finalization.
However, in general resources must be managed (particularly for long-lived programs, programs that use many resources, or for safety, to ensure that data is written out). Explicit disposal means that resource finalization and release is deterministic and prompt: the dispose
method does not complete until these are done.
An alternative to requiring explicit disposal is to tie resource management to object lifetime: resources are acquired during object creation, and released during object destruction. This approach is known as the Resource Acquisition Is Initialization (RAII) idiom, and is used in languages with deterministic memory management (e.g. C++). In this case, in the example above, the resource is acquired when the file object is created, and when the scope of the variable f
is exited, the file object that f
refers to is destroyed, and as part of this, the resource is released.
RAII relies on object lifetime being deterministic; however, with automatic memory management, object lifetime is not a concern of the programmer: objects are destroyed at some point after they are no longer used, but when is abstracted. Indeed, lifetime is often not deterministic, though it may be, notably if reference counting is used. Indeed, in some cases there is no guarantee that objects will ever be finalized: when the program terminates, it may not finalize the objects, and instead just let the operating system reclaim memory; if finalization is required (e.g., to flush buffers), data loss can occur.
Thus by not coupling resource management to object lifetime, the dispose pattern allows resources to be released promptly, while giving implementation flexibility for memory management. The cost of this is that resources must be managed manually, which can be tedious and error-prone.
Early exit
[edit]A key problem with the dispose pattern is that if the dispose
method is not called, the resource is leaked. A common cause of this is early exit from a function, due to an early return or exception.
For example:
def func(filename):
f = open(filename)
if a:
return x
f.close()
return y
If the function returns at the first return, the file is never closed and the resource is leaked.
def func(filename):
f = open(filename)
g(f) # Do something with f that may raise an exception.
f.close()
If the intervening code raises an exception, the function exits early and the file is never closed, so the resource is leaked.
Both of these can be handled by a try...finally
construct, which ensures that the finally clause is always executed on exit:
def func(filename):
try:
f = open(filename)
# Do something.
finally:
f.close()
More generically:
Resource resource = getResource();
try {
// Resource has been acquired; perform actions with the resource.
...
} finally {
// Release resource, even if an exception was thrown.
resource.dispose();
}
The try...finally
construct is necessary for proper exception safety, since the finally
block enables execution of cleanup logic regardless of if an exception is thrown or not in the try
block.
One disadvantage of this approach is that it requires the programmer to explicitly add cleanup code in a finally
block. This leads to code size bloat, and failure to do so will lead to resource leakage in the program.
Language constructs
[edit]To make the safe use of the dispose pattern less verbose, several languages have some kind of built-in support for resources held and released in the same block of code.
The C# language features the using
statement[2] that automatically calls the Dispose
method on an object that implements the IDisposable
interface:
using (Resource resource = GetResource())
{
// Perform actions with the resource.
...
}
which is equal to:
Resource resource = GetResource()
try
{
// Perform actions with the resource.
...
}
finally
{
// Resource might not been acquired, or already freed
if (resource != null)
((IDisposable)resource).Dispose();
}
Similarly, the Python language has a with
statement that can be used to similar effect with a context manager object. The context manager protocol requires implementing __enter__
and __exit__
methods which get automatically called by the with
statement construct, to prevent duplication of code that would otherwise occur with the try
/finally
pattern.[3]
with resource_context_manager() as resource:
# Perform actions with the resource.
...
# Perform other actions where the resource is guaranteed to be deallocated.
...
The Java language introduced a new syntax called try
-with-resources in Java version 7.[4] It can be used on objects that implement the AutoCloseable interface (that defines method close()):
try (OutputStream x = new OutputStream(...)) {
// Do something with x
} catch (IOException ex) {
// Handle exception
// The resource x is automatically closed
} // try
Problems
[edit]Beyond the key problem of correct resource management in the presence of returns and exceptions, and heap-based resource management (disposing objects in a different scope from where they are created), there are many further complexities associated with the dispose pattern. These problems are largely avoided by RAII. However, in common simple use these complexities do not arise: acquire a single resource, do something with it, automatically release it.
A fundamental problem is that having a resource is no longer a class invariant (the resource is held from object creation until it is disposed, but the object is still live at this point), so the resource may not be available when the object tries to use it, for example trying to read from a closed file. This means that all methods on the object that use the resource potentially fail, concretely usually by returning an error or raising an exception. In practice this is minor, as use of resources can usually fail for other reasons as well (for example, trying to read past the end of a file), so these methods already might fail, and not having a resource just adds another possible failure. A standard way to implement this is to add a boolean field to the object, called disposed
, which is set to true by dispose
, and checked by a guard clause to all methods (that use the resource), raising an exception (such as ObjectDisposedException
in .NET) if the object has been disposed.[5]
Further, it is possible to call dispose
on an object more than once. While this may indicate a programming error (each object holding a resource must be disposed exactly once), it is simpler, more robust, and thus usually preferable for dispose
to be idempotent (meaning "calling multiple times is the same as calling once").[5] This is easily implemented by using the same boolean disposed
field and checking it in a guard clause at the start of dispose
, in that case returning immediately, rather than raising an exception.[5] Java distinguishes disposable types (those that implement AutoCloseable) from disposable types where dispose is idempotent (the subtype Closeable).
Disposal in the presence of inheritance and composition of objects that hold resources have analogous problems to destruction/finalization (via destructors or finalizers). Further, since the dispose pattern usually does not have language support for this, boilerplate code is necessary. Firstly, if a derived class overrides a dispose
method in the base class, the overriding method in the derived class generally needs to call the dispose
method in the base class, in order to properly release resources held in the base. Secondly, if an object has a "has a" relationship with another object that holds a resource (i.e., if an object indirectly uses a resource through another object that directly uses a resource), should the indirectly using object be disposable? This corresponds to whether the relationship is owning (object composition) or viewing (object aggregation), or even just communicating (association), and both conventions are found (indirect user is responsible for the resource or is not responsible). If the indirect use is responsible for the resource, it must be disposable, and dispose the owned objects when it is disposed (analogous to destroying or finalizing owned objects).
Composition (owning) provides encapsulation (only the object that is used needs to be tracked), but at the cost of considerable complexity when there are further relationships between objects, while aggregation (viewing) is considerably simpler, at the cost of lacking encapsulation. In .NET, convention is to only have direct user of resources be responsible: "You should implement IDisposable only if your type uses unmanaged resources directly."[6] See resource management for details, and further examples.
See also
[edit]Notes
[edit]- ^ In class-based programming, methods are defined in a class, using an implicit
this
orself
parameter, rather than as functions taking an explicit parameter.
References
[edit]- ^ The Single UNIX Specification, Version 4 from The Open Group – Base Definitions Reference,
- ^ Microsoft MSDN: using Statement (C# Reference)
- ^ Guido van Rossum, Nick Coghlan (13 June 2011). "PEP 343: The "with" Statement". Python Software Foundation.
- ^ Oracle Java tutorial: The try-with-resources Statement
- ^ a b c "Dispose Pattern".
- ^ "IDisposable Interface". Retrieved 2024-12-09.
Further reading
[edit]- Microsoft Developer Network: Dispose Pattern