Introduction
This article is an introduction to the using statement in c# and also provides some insight into the actual implementation of the statement.
The Code
When you are using an object that encapsulates any resource, you have to make sure that when you are done with the object, the object's Dispose method is called. This can be done more easily using the using statement in C#. The using statement simplifies the code that you have to write to create and then finally clean up the object. The using statement obtains the resource specified, executes the statements and finally calls the Dispose method of the object to clean up the object. The following piece of code illustrates its use.
using (TextWriter w = File.CreateText("log.txt"))
{
w.WriteLine("This is line one");
}
Now that's cool. But before we can start using this code, let us try to understand what happens behind the screen. Lets have a look at the IL code for the above code section.
.locals init ([0] class [mscorlib]System.IO.TextWriter w)
IL_0000: ldstr "log.txt"
IL_0005: call class [mscorlib]System.IO.StreamWriter
[mscorlib]System.IO.File::CreateText(string)
IL_000a: stloc.0
.try
{
IL_000b: ldloc.0
IL_000c: ldstr "This is line one"
IL_0011: callvirt instance void [mscorlib]
System.IO.TextWriter::WriteLine(string)
IL_0016: leave.s IL_0022
} finally
{
IL_0018: ldloc.0
IL_0019: brfalse.s IL_0021
IL_001b: ldloc.0
IL_001c: callvirt instance void [mscorlib]
System.IDisposable::Dispose()
IL_0021: endfinally
}
Hmmmm.... Well doesn't look like this is my code. That's because I see a try and a finally in the IL code (something that I haven't implemented). Wait a minute. IT IS MY CODE....
Waaaaaah... Somebody changed my code...
Well the truth is, somebody did change your code. The CLR. The CLR converts your code into MSIL. And the using statement gets translated into a try and finally block. This is how the using statement is represented in IL. A using statement is translated into three parts: acquisition, usage, and disposal. The resource is first acquired, then the usage is enclosed in a try statement with a finally clause. The object then gets disposed in the finally clause. For example the following lines of code using the using statement,
using (MyResource myRes = new MyResource())
{
myRes.DoSomething();
}
gets translated to,
MyResource myRes= new MyResource();
try
{
myRes.DoSomething();
}
finally
{
if (myRes!= null)
((IDisposable)myRes).Dispose();
}
Hmmm... That explains it.
The above code that uses the using statement corresponds to one of the two possible expansions. When MyResource is a value type, the expansion in the finally block will be
finally
{
((IDisposable)myRes).Dispose();
}
If MyResource is of reference type, the expansion becomes
finally
{
if(myRes != null)
((IDisposable)myRes).Dispose();
}
This way, if a null resource is acquired, then no call will be made to Dispose, thus avoiding any exception that occurs.
Well, that explains everything.
Using 'using'
A typical scenario where we could use the using statement is :
string connString = "Data Source=localhost;Integrated " +
"Security=SSPI;Initial Catalog=Northwind;";
using (SqlConnection conn = new SqlConnection(connString))
{
SqlCommand cmd = conn.CreateCommand();
cmd.CommandText = "SELECT ID, Name FROM Customers";
conn.Open();
using (SqlDataReader dr = cmd.ExecuteReader())
{
while (dr.Read())
Console.WriteLine("{0}\t{1}", dr.GetString(0), dr.GetString(1));
}
}
Note
The using statement is only useful for objects with a lifetime that does not extend beyond the method in which the objects are constructed. Remember that the objects you instantiate must implement the System.IDisposable interface.
There is no equivalent for the using statement in vb.net. You have to use the try finally block.