Home

Exception Handling

 

Introduction to Exceptions

 

An Overview

Imagine you want to write a program that requests a number from the user, multiplies the number by 2, and displays the result. The form of this program can be designed as follows:

private void button1_Click(object sender, System.EventArgs e)
{
	double Number;

	Number = double.Parse(textBox1.Text);

	textBox2.Text = (Number * 2).ToString();
}

This looks like an easy request. When it comes up, the user is asked to simply type a number. The number would then be multiplied by 2 and display the result. Imagine that a user types something that is not a valid number, such as the name of a country or somebody’s telephone number. Since this program was expecting a number and it is not prepared to multiply a string to a number, it would produce an error. Whenever the compiler is handed a task, it would try to perform the assignment. If it can’t perform the assignment, for any reason it is not prepared for, it would throw an error. As a programmer, if you can anticipate the type of error that could occur in your program, you can catch the error yourself and deal with it by telling the compiler what to do when this type of error occurs.

Practical LearningPractical Learning: Introducing Exception Handling

  1. Start a new Windows Application named GCS4 and design the form as follows:
     
    Control Name Text Additional Properties
    Form     Size: 360, 408
    Label   Customer Name:  
    TextBox txtCustomer1    
    Label   mm  
    Label   dd  
    Label   yyyy  
    Label   Order Date:  
    TextBox txtMM 1 TextAlign: Right
    TextBox txtDD 1 TextAlign: Right
    TextBox txtYYYY 1970 TextAlign: Right
    Label   Item Types  
    Label   Qty  
    Label   Unit Price  
    Label   Sub-Total  
    Label   Shirts  
    TextBox txtQtyShirts 0 TextAlign: Right
    TextBox txtUnitPriceShirts 0.95 TextAlign: Right
    TextBox txtSubTotalShirts 0.00 TextAlign: Right
    Label   Pants  
    TextBox txtQtyPants 0 TextAlign: Right
    TextBox txtUnitPricePants 2.75 TextAlign: Right
    TextBox txtSubTotalPants 0.00 TextAlign: Right
    Label   Dresses  
    TextBox txtQtyDresses 0 TextAlign: Right
    TextBox txtUnitPriceDresses 4.50 TextAlign: Right
    TextBox txtSubTotalDresses 0.00 TextAlign: Right
    Button btnProcess Process  
    Label   Customer Name:  
    TextBox txtCustomer2    
    Label   Order date:  
    TextBox txtOrderDate    
    Label   Tax Rate:  
    TextBox txtTaxRate 5.75  
    Label   %  
    Button btnTax Tax  
    Label   Total Order:  
    TextBox txtTotalOrder 0.00 TextAlign: Right
    Label   Tax Amount:  
    TextBox txtTaxAmount 0.00 TextAlign: Right
    Label   Net Price:  
    TextBox txtNetPrice 0.00 TextAlign: Right
    Label   Amount Tended:  
    TextBox txtAmountTended 0.00 TextAlign: Right
    Button btnDifference Diff  
    Label   Difference:  
    TextBox txtDifference 0.00 TextAlign: Right
  2. To arrange the tab sequence, on the main menu, click View -> Tab Order
  3. On the form, click only the following controls whose squares have a white background, in the indicated order:
     
  4. Press Esc
  5. Resize the form to appear as follows:
     
  6. To process an order, double-click the Process button and implement its event as follows:
     
    using System;
    using System.Drawing;
    using System.Collections;
    using System.ComponentModel;
    using System.Windows.Forms;
    using System.Data;
    
    namespace GCS4
    {
    	/// <summary>
    	/// Summary description for Form1.
    	/// </summary>
    	public class Form1 : System.Windows.Forms.Form
    	{
    		. . . No Change
    		
    		/// <summary>
    		/// Required designer variable.
    		/// </summary>
    		private System.ComponentModel.Container components = null;
    
    		// Order Information
    		string CustomerName;
    		string mm, dd, yyyy;
    
    		// Quantities of items
    		uint NumberOfShirts;
    		uint NumberOfPants;
    		uint NumberOfDresses;
    
    		// Price of items
    		decimal PriceOneShirt, PriceAPairOfPants, PriceOneDress;
    
    		// Each of these sub totals will be used for cleaning items
    		decimal SubTotalShirts, SubTotalPants, SubTotalDresses;
    
    		// Values used to process an order
    		decimal TaxRate;
    		decimal TotalOrder, TaxAmount, SalesTotal;
    
    		public Form1()
    		{
    			//
    			// Required for Windows Form Designer support
    			//
    			InitializeComponent();
    
    			//
    			// TODO: Add any constructor code after InitializeComponent call
    			//
    		}
    
    		/// <summary>
    		/// Clean up any resources being used.
    		/// </summary>
    		protected override void Dispose( bool disposing )
    		{
    			if( disposing )
    			{
    				if (components != null) 
    				{
    					components.Dispose();
    				}
    			}
    			base.Dispose( disposing );
    		}
    
    		#region Windows Form Designer generated code
    		/// <summary>
    		/// Required method for Designer support - do not modify
    		/// the contents of this method with the code editor.
    		/// </summary>
    		private void InitializeComponent()
    		{
    			
    			. . . No Change
    			
    
    		}
    		#endregion
    
    		/// <summary>
    		/// The main entry point for the application.
    		/// </summary>
    		[STAThread]
    		static void Main() 
    		{
    			Application.Run(new Form1());
    		}
    
    		private void btnProcess_Click(object sender, System.EventArgs e)
    		{
    			if( this.btnProcess.Text == "Process" )
    			{
    				this.Height = 408;
    				this.btnProcess.Text = "Reset";
    			}
    			else
    			{
    				this.Height = 232;
    				this.txtCustomer1.Text = "";
    				this.txtMM.Text = "1";
    				this.txtDD.Text = "1";
    				this.txtYYYY.Text = "1970";
    				this.txtQtyShirts.Text = "0";
    				this.txtQtyPants.Text  = "0";
    				this.txtQtyDresses.Text = "0";
    				this.txtSubTotalShirts.Text = "$0.00";
    				this.txtSubTotalPants.Text  = "$0.00";
    				this.txtSubTotalDresses.Text = "$0.00";
    
    				this.btnProcess.Text = "Process";
    			}
    			
    			// Request order information from the user
    			CustomerName = this.txtCustomer1.Text;
    			mm   = this.txtMM.Text;
    			dd   = this.txtDD.Text;
    			yyyy = this.txtYYYY.Text;
    
    			// Request the quantity of each category of items
    			// Number of Shirts
    			NumberOfShirts  = uint.Parse(this.txtQtyShirts.Text);
    			// Number of Pants
    			NumberOfPants   = uint.Parse(this.txtQtyPants.Text);
    			// Number of Dresses
    			NumberOfDresses = uint.Parse(this.txtQtyDresses.Text);
    
    			// Unit Prices of items
    			PriceOneShirt     = decimal.Parse(this.txtUnitPriceShirts.Text);
    			PriceAPairOfPants = decimal.Parse(this.txtUnitPricePants.Text);
    			PriceOneDress     = decimal.Parse(this.txtUnitPriceDresses.Text);
    
    			// Perform the necessary calculations
    			SubTotalShirts  = NumberOfShirts  * PriceOneShirt;
    			SubTotalPants   = NumberOfPants   * PriceAPairOfPants;
    			SubTotalDresses = NumberOfDresses * PriceOneDress;
    
    			this.txtSubTotalShirts.Text  = SubTotalShirts.ToString("C");
    			this.txtSubTotalPants.Text   = SubTotalPants.ToString("C");
    			this.txtSubTotalDresses.Text = SubTotalDresses.ToString("C");
    
    			// Calculate the "temporary" total of the order
    			TotalOrder      = SubTotalShirts + SubTotalPants + SubTotalDresses;
    
    			// Display the receipt
    			this.txtCustomer2.Text  = CustomerName;
    			this.txtOrderDate.Text  = mm + "/" + dd + "/" + yyyy;
    			this.txtTotalOrder.Text = TotalOrder.ToString("C");
    		}
    	}
    }
    
  7. Double-click the Tax button and implement its event as follows:
     
    		private void btnTax_Click(object sender, System.EventArgs e)
    		{
    			
    			// Get the tax rate
    			TaxRate =  decimal.Parse(this.txtTaxRate.Text) / 100;
    
    			// Calculate the tax amount using a constant rate
    			TaxAmount       = TotalOrder * TaxRate;
    			// Add the tax amount to the total order
    			SalesTotal      = TotalOrder + TaxAmount;
    
    			this.txtTaxAmount.Text  = TaxAmount.ToString("C");
    			this.txtNetPrice.Text   = SalesTotal.ToString("C");
    		}
  8. Double-click the Diff button and implement its event as follows:
     
    		private void btnDifference_Click(object sender, System.EventArgs e)
    		{
    			decimal AmountTended, Difference;
    
    			// Request money for the order
    			AmountTended    = decimal.Parse(this.txtAmountTended.Text);
    
    			// Calculate the difference owed to the customer
    			// or that the customer still owes to the store
    			Difference      = AmountTended - SalesTotal;
    
    			this.txtDifference.Text = Difference.ToString("C");
    		}
  9. To execute the application, on the Standard toolbar, click the Start Without Debugging button
     
  10. Close the form and return to your programming environment
  11. Execute the application again. This time, type a letter such as d for the quantity of shirts and click Process
     
  12. Click Quit and close the form to return to your programming environment

Exceptional Behaviors

An exception is an unusual situation that could occur in your program. As a programmer, you should anticipate any abnormal behavior that could be caused by the user entering wrong information that could otherwise lead to unpredictable results. The ability to deal with a program’s eventual abnormal behavior is called exception handling. C# provides four keywords to handle an exception. At this time, we will review two of them: try and catch. Later on, one more keyword, throw, will be reviewed. In another lesson, we will introduce the last keyword, finally.

  1. Trying the normal flow: To deal with the expected behavior of a program, use the try keyword as in the following syntax:

    try {Behavior}

    The try keyword is required. It lets the compiler know that you are attempting a normal flow of your program. The actual behavior that needs to be evaluated is included between an opening curly bracket “{“ and a closing curly bracket “}”. Inside of the brackets, implement the normal flow that the program must follow, at least for this section of the code. Here is an example:
     
    private void button1_Click(object sender, System.EventArgs e)
    {
    	double Number;
    
    	try 
    	{
    		Number = double.Parse(textBox1.Text);
    
    		textBox2.Text = (Number * 2).ToString();
    	}
    
    }
  2. Catching Errors: During the flow of the program as part of the try section, if an abnormal behavior occurs, instead of letting the program crash or instead of letting the compiler send the error to the operating system, you can transfer the flow of the program to another section that can deal with it. The syntax used by this section is:

    catch {WhatToDo}

    This section always follows the try section. There must not be any code between the try’s closing bracket and the catch section. The catch keyword is required and follows the try section. Combined with the try block, the syntax of an exception would be:
     
    try
    {
    	// Try the program flow
    }
    catch
    {
    	// Catch the exception
    }

    A program that includes a catch section would appear as follows:
     
    private void button1_Click(object sender, System.EventArgs e)
    {
    	double Number;
    
    	try 
    	{
    		Number = double.Parse(textBox1.Text);
    
    		textBox2.Text = (Number * 2).ToString();
    	}
    	catch
    	{
    
    	}
    }

 

Practical LearningPractical Learning: Introducing Vague Exceptions

  1. To introduce exceptions, access the form's code and change the events of the buttons as follows:
     
    using System;
    using System.Drawing;
    using System.Collections;
    using System.ComponentModel;
    using System.Windows.Forms;
    using System.Data;
    
    namespace GCS4
    {
    	/// <summary>
    	/// Summary description for Form1.
    	/// </summary>
    	public class Form1 : System.Windows.Forms.Form
    	{
    		. . . No Change
    
    		private void btnProcess_Click(object sender, System.EventArgs e)
    		{
    			if( this.btnProcess.Text == "Process" )
    			{
    				this.Height = 408;
    				this.btnProcess.Text = "Reset";
    			}
    			else
    			{
    				this.Height = 232;
    				this.txtCustomer1.Text = "";
    				this.txtMM.Text = "1";
    				this.txtDD.Text = "1";
    				this.txtYYYY.Text = "1970";
    				this.txtQtyShirts.Text = "0";
    				this.txtQtyPants.Text  = "0";
    				this.txtQtyDresses.Text = "0";
    				this.txtSubTotalShirts.Text = "$0.00";
    				this.txtSubTotalPants.Text  = "$0.00";
    				this.txtSubTotalDresses.Text = "$0.00";
    
    				this.btnProcess.Text = "Process";
    			}
    			
    			// Request order information from the user
    			CustomerName = this.txtCustomer1.Text;
    			mm   = this.txtMM.Text;
    			dd   = this.txtDD.Text;
    			yyyy = this.txtYYYY.Text;
    
    			// Request the quantity of each category of items
    			try 
    			{
    				// Number of Shirts
    				NumberOfShirts  = uint.Parse(this.txtQtyShirts.Text);
    			}
    			catch
    			{
    			}
    
    			try 
    			{
    				// Number of Pants
    				NumberOfPants   = uint.Parse(this.txtQtyPants.Text);
    			}
    			catch {}
    			try 
    			{
    				// Number of Dresses
    				NumberOfDresses = uint.Parse(this.txtQtyDresses.Text);
    			}
    			catch {}
    
    			// Unit Prices of items
    			try 
    			{
    				PriceOneShirt     = decimal.Parse(this.txtUnitPriceShirts.Text);
    			}
    			catch{}
    			try 
    			{
    				PriceAPairOfPants = decimal.Parse(this.txtUnitPricePants.Text);
    			}
    			catch{}
    			try 
    			{
    				PriceOneDress     = decimal.Parse(this.txtUnitPriceDresses.Text);
    			}
    			catch{}
    
    			// Perform the necessary calculations
    			SubTotalShirts  = NumberOfShirts  * PriceOneShirt;
    			SubTotalPants   = NumberOfPants   * PriceAPairOfPants;
    			SubTotalDresses = NumberOfDresses * PriceOneDress;
    
    			this.txtSubTotalShirts.Text  = SubTotalShirts.ToString("C");
    			this.txtSubTotalPants.Text   = SubTotalPants.ToString("C");
    			this.txtSubTotalDresses.Text = SubTotalDresses.ToString("C");
    
    			// Calculate the "temporary" total of the order
    			TotalOrder      = SubTotalShirts + SubTotalPants + SubTotalDresses;
    
    			// Display the receipt
    			this.txtCustomer2.Text  = CustomerName;
    			this.txtOrderDate.Text  = mm + "/" + dd + "/" + yyyy;
    			this.txtTotalOrder.Text = TotalOrder.ToString("C");
    		}
    
    		private void btnTax_Click(object sender, System.EventArgs e)
    		{
    			try 
    			{
    				// Get the tax rate
    				TaxRate =  decimal.Parse(this.txtTaxRate.Text) / 100;
    			}
    			catch{}
    
    			// Calculate the tax amount using a constant rate
    			TaxAmount       = TotalOrder * TaxRate;
    			// Add the tax amount to the total order
    			SalesTotal      = TotalOrder + TaxAmount;
    
    			this.txtTaxAmount.Text  = TaxAmount.ToString("C");
    			this.txtNetPrice.Text   = SalesTotal.ToString("C");
    		}
    
    		private void btnDifference_Click(object sender, System.EventArgs e)
    		{
    			decimal AmountTended = 0.00M, Difference;
    
    			try 
    			{
    				// Request money for the order
    				AmountTended    = decimal.Parse(this.txtAmountTended.Text);
    			}
    			catch{}
    
    			// Calculate the difference owed to the customer
    			// or that the customer still owes to the store
    			Difference      = AmountTended - SalesTotal;
    
    			this.txtDifference.Text = Difference.ToString("C");
    		}
    	}
    }
  2. Execute the application. This time, type invalid values in the quantity text boxes and other text boxes where the user is supposed to enter some values
  3. Click Process
     
  4. Return to your programming environment

Exceptions and Custom Messages

As mentioned already, if an error occurs when processing the program in the try section, the compiler transfers the processing to the next catch section. You can then use the catch section to deal with the error. At a minimum, you can display a message to inform the user. Here is an example:

private void button1_Click(object sender, System.EventArgs e)
{
	double Number;

	try 
	{
		Number = double.Parse(textBox1.Text);

		textBox2.Text = (Number * 2).ToString();
	}
	catch
	{
		label3.Text = "Invalid Number";
	}
}
Custom Message

Of course, your message may not be particularly clear but this time, the program will not crash. In the next sections, we will learn better ways of dealing with the errors and the messages.

Practical LearningPractical Learning: Displaying Custom Messages

  1. To display custom messages to the user, change the form's file as follows:
     
    using System;
    using System.Drawing;
    using System.Collections;
    using System.ComponentModel;
    using System.Windows.Forms;
    using System.Data;
    
    namespace GCS4
    {
    	/// <summary>
    	/// Summary description for Form1.
    	/// </summary>
    	public class Form1 : System.Windows.Forms.Form
    	{
    		. . . No Change
    
    		static void Main() 
    		{
    			Application.Run(new Form1());
    		}
    
    		private void btnProcess_Click(object sender, System.EventArgs e)
    		{
    			if( this.btnProcess.Text == "Process" )
    			{
    				this.Height = 408;
    				this.btnProcess.Text = "Reset";
    			}
    			else
    			{
    				this.Height = 232;
    				this.txtCustomer1.Text = "";
    				this.txtMM.Text = "1";
    				this.txtDD.Text = "1";
    				this.txtYYYY.Text = "1970";
    				this.txtQtyShirts.Text = "0";
    				this.txtQtyPants.Text  = "0";
    				this.txtQtyDresses.Text = "0";
    				this.txtSubTotalShirts.Text = "$0.00";
    				this.txtSubTotalPants.Text  = "$0.00";
    				this.txtSubTotalDresses.Text = "$0.00";
    
    				this.btnProcess.Text = "Process";
    			}
    			
    			// Request order information from the user
    			CustomerName = this.txtCustomer1.Text;
    			mm   = this.txtMM.Text;
    			dd   = this.txtDD.Text;
    			yyyy = this.txtYYYY.Text;
    
    			// Request the quantity of each category of items
    			try 
    			{
    				// Number of Shirts
    				NumberOfShirts  = uint.Parse(this.txtQtyShirts.Text);
    			}
    			catch
    			{
    				MessageBox.Show("The value you typed for the number of " +
    					            "shirts is not a valid number." +
    					"\nPlease enter a natural number such as 2 or 24 or even 248");
    			}
    
    			try 
    			{
    				// Number of Pants
    				NumberOfPants   = uint.Parse(this.txtQtyPants.Text);
    			}
    			catch 
    			{
    				MessageBox.Show("The value you typed for the number of " +
    					"pair or pants is not a valid number." +
    					"\nPlease enter a natural number such as 2 or 24 or even 248");
    			}
    			try 
    			{
    				// Number of Dresses
    				NumberOfDresses = uint.Parse(this.txtQtyDresses.Text);
    			}
    			catch 
    			{
    				MessageBox.Show("The value you typed for the number of " +
    					"dresses is not a valid number." +
    					"\nPlease enter a natural number such as 2 or 24 or even 248");
    			}
    
    			// Unit Prices of items
    			try 
    			{
    				PriceOneShirt     = decimal.Parse(this.txtUnitPriceShirts.Text);
    			}
    			catch
    			{
    				MessageBox.Show("The value you entered for the unit price of " +
    					"a shirt is not a recognizable currency amount." +
    					"\nOnly natural or decimal numbers are allowed. " + 
    					"Please consult the management to know the valid prices.");
    			}
    			try 
    			{
    				PriceAPairOfPants = decimal.Parse(this.txtUnitPricePants.Text);
    			}
    			catch
    			{
    				MessageBox.Show("The value you entered for the unit price of " +
    					"a pair of pants is not a recognizable currency amount." +
    					"\nOnly natural or decimal numbers are allowed. " + 
    					"You can consult the management to find out about " +
    					"the allowable prices.");
    			}
    			try 
    			{
    				PriceOneDress     = decimal.Parse(this.txtUnitPriceDresses.Text);
    			}
    			catch
    			{
    				MessageBox.Show("The value you entered for the unit price of " +
    					"a dress is not a valid amount." +
    					"\nYou must enter only a natural or a decimal number. " + 
    					"For more information, please consult the management to get " +
    					"the right prices.");
    			}
    
    			// Perform the necessary calculations
    			SubTotalShirts  = NumberOfShirts  * PriceOneShirt;
    			SubTotalPants   = NumberOfPants   * PriceAPairOfPants;
    			SubTotalDresses = NumberOfDresses * PriceOneDress;
    
    			this.txtSubTotalShirts.Text  = SubTotalShirts.ToString("C");
    			this.txtSubTotalPants.Text   = SubTotalPants.ToString("C");
    			this.txtSubTotalDresses.Text = SubTotalDresses.ToString("C");
    
    			// Calculate the "temporary" total of the order
    			TotalOrder      = SubTotalShirts + SubTotalPants + SubTotalDresses;
    
    			// Display the receipt
    			this.txtCustomer2.Text  = CustomerName;
    			this.txtOrderDate.Text  = mm + "/" + dd + "/" + yyyy;
    			this.txtTotalOrder.Text = TotalOrder.ToString("C");
    		}
    
    		private void btnTax_Click(object sender, System.EventArgs e)
    		{
    			try 
    			{
    				// Get the tax rate
    				TaxRate =  decimal.Parse(this.txtTaxRate.Text) / 100;
    			}
    			catch
    			{
    				MessageBox.Show("The value you entered is not recognized as a valid tax rate." +
    					"\nA valid tax rate is a value between 0 and 100.00" +
    					"\nPlease try again.");
    			}
    
    			// Calculate the tax amount using a constant rate
    			TaxAmount       = TotalOrder * TaxRate;
    			// Add the tax amount to the total order
    			SalesTotal      = TotalOrder + TaxAmount;
    
    			this.txtTaxAmount.Text  = TaxAmount.ToString("C");
    			this.txtNetPrice.Text   = SalesTotal.ToString("C");
    		}
    
    		private void btnDifference_Click(object sender, System.EventArgs e)
    		{
    			decimal AmountTended = 0.00M, Difference;
    
    			try 
    			{
    				// Request money for the order
    				AmountTended    = decimal.Parse(this.txtAmountTended.Text);
    			}
    			catch
    			{
    				MessageBox.Show("The value you entered for the amount tended is not valid. " +
    					"Only natural or decimal numbers are allowed." + 
    					"Please try again.");
    			}
    
    			// Calculate the difference owed to the customer
    			// or that the customer still owes to the store
    			Difference      = AmountTended - SalesTotal;
    
    			this.txtDifference.Text = Difference.ToString("C");
    		}
    	}
    }
  2. Test the application with valid and invalid values. Here is an example:
     

  3. Return to Notepad

Exceptions in the .NET Framework

 

The Exception Class

In traditionally-oriented error dealing languages such as C/C++ or Object Pascal, you could create any exception of your choice, including numeric or strings. To customize exception handling, you could also create your own class(es). Most libraries such as Borland's VCL and Microsoft's MFC also shipped with their own classes to handle exceptions. Event the Win32 library provides its type of mechanism to face errors. To support exception handling, the .NET Framework provides a special class called Exception. Once the compiler encounters an error, the Exception class allows you to identify the type of error and take an appropriate action.

Normally, Exception mostly serves as the general class of exceptions. Anticipating various types of problems that can occur in a program, Microsoft derived various classes from Exception to make this issue friendlier. As a result, almost any type of exception you may encounter already has a class created to deal with it. Therefore, when your program faces an exception, you can easily identify the type of error. There are so many exception classes that we cannot study or review them all. The solution we will use is to introduce or review a class when we meet its type of error.

The Exception's Message

In exception handling, errors are dealt with in the catch section. To do this, use catch as if it were a method. This means that, on the right side of catch, opening a parenthesis, declare a variable of the type of exception you want to deal with. By default, an exception is first of type Exception. Based on this, a typical formula to implement exception handling is:

try
{
	// Process the normal flow of the program here
}
catch(Exception e)
{
	// Deal with the exception here
}

When an exception occurs in the try section, code compilation is transferred to the catch section. If you declare the exception as an Exception type, this class will identify the error. One of the properties of the Exception class is called Message. This property contains a string that describes the type of error that occurred. You can then use this Exception.Message property to display an error message if you want. Here is an example:
private void button1_Click(object sender, System.EventArgs e)
{
	double Number;

	try 
	{
		Number = double.Parse(textBox1.Text);

		textBox2.Text = (Number * 2).ToString();
	}
	catch(Exception Ex)
	{
		MessageBox.Show(Ex.Message);
	}
}
An exception using the Exception.Message message
 

Custom Error Messages

As you can see, one of the strengths of the Exception.Message property is that it gives you a good indication of the type of problem that occurred. Sometimes, the message provided by the Exception class may not appear explicit enough. In fact, you may not want to show it to the user since, as in this case, the user may not understand what the expression "correct format" in this context means and why it is being used. As an alternative, you can create your own message and display it to the user. Here is an example:
private void button1_Click(object sender, System.EventArgs e)
{
	double Number;

	try 
	{
		Number = double.Parse(textBox1.Text);

		textBox2.Text = (Number * 2).ToString();
	}
	catch(Exception Ex)
	{
		MessageBox.Show("The operation could not be carried because " +
				"the number you typed is not valid");
	}
}
An exception with a custom message

You can also combine the Exception.Message message and your own message:

private void button1_Click(object sender, System.EventArgs e)
{
	double Number;

	try 
	{
		Number = double.Parse(textBox1.Text);

		textBox2.Text = (Number * 2).ToString();
	}
	catch(Exception Ex)
	{
		MessageBox.Show(Ex.Message +
                                "\nThe operation could not be carried because " +
				"the number you typed is not valid");
	}
}
 

A Review of .NET Exception Classes

 

Introduction

The .NET Framework provides various classes to handle almost any type of exception you can think of. There are so many of these classes that we can only mention the few that we will regularly use in our application.

There are two main ways you can use one of the classes of the .NET Framework. If you know for sure that a particular exception will be produced, pass its name to a catch() clause. You don't have to name the argument. Then, in the catch() section, display a custom message. The second option you have consists of using the throw keyword. We will study it later.

From now on, we will try to always indicate the type of exception that could be thrown if something goes wrong in a program.

FormatException

Everything the user types into an application using the keyboard is primarily a string and you must convert it to the appropriate type before using it. When you request a specific type of value from the user, after the user has typed it and you decide to convert it to the appropriate type, if your conversion fails, the program produces (we will use he word "throw") an error. The error is of from the FormatException class.

Here is a program that deals with a FormatException exception:
private void button1_Click(object sender, System.EventArgs e)
{
	double Number;

	try 
	{
		Number = double.Parse(textBox1.Text);

		textBox2.Text = (Number * 2).ToString();
	}
	catch(FormatException)
	{
		MessageBox.Show("You typed an invalid number");
	}
}
The FormatException exception
 

Practical LearningPractical Learning: Using the FormatException Class

  1. Change the form's file as follows:
     
    using System;
    using System.Drawing;
    using System.Collections;
    using System.ComponentModel;
    using System.Windows.Forms;
    using System.Data;
    
    namespace GCS4
    {
    	/// <summary>
    	/// Summary description for Form1.
    	/// </summary>
    	public class Form1 : System.Windows.Forms.Form
    	{
    		. . . No Change
    
    
    		private void btnProcess_Click(object sender, System.EventArgs e)
    		{
    			. . . No Change
    
    			
    			// Request order information from the user
    			CustomerName = this.txtCustomer1.Text;
    			mm   = this.txtMM.Text;
    			dd   = this.txtDD.Text;
    			yyyy = this.txtYYYY.Text;
    
    			// Request the quantity of each category of items
    			try 
    			{
    				// Number of Shirts
    				NumberOfShirts  = uint.Parse(this.txtQtyShirts.Text);
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you typed for the number of " +
    					            "shirts is not a valid number." +
    					"\nPlease enter a natural number such as 2 or 24 or even 248");
    			}
    
    			try 
    			{
    				// Number of Pants
    				NumberOfPants   = uint.Parse(this.txtQtyPants.Text);
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you typed for the number of " +
    					"pair or pants is not a valid number." +
    					"\nPlease enter a natural number such as 2 or 24 or even 248");
    			}
    			try 
    			{
    				// Number of Dresses
    				NumberOfDresses = uint.Parse(this.txtQtyDresses.Text);
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you typed for the number of " +
    					"dresses is not a valid number." +
    					"\nPlease enter a natural number such as 2 or 24 or even 248");
    			}
    
    			// Unit Prices of items
    			try 
    			{
    				PriceOneShirt     = decimal.Parse(this.txtUnitPriceShirts.Text);
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you entered for the unit price of " +
    					"a shirt is not a recognizable currency amount." +
    					"\nOnly natural or decimal numbers are allowed. " + 
    					"Please consult the management to know the valid prices.");
    			}
    			try 
    			{
    				PriceAPairOfPants = decimal.Parse(this.txtUnitPricePants.Text);
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you entered for the unit price of " +
    					"a pair of pants is not a recognizable currency amount." +
    					"\nOnly natural or decimal numbers are allowed. " + 
    					"You can consult the management to find out about " +
    					"the allowable prices.");
    			}
    			try 
    			{
    				PriceOneDress     = decimal.Parse(this.txtUnitPriceDresses.Text);
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you entered for the unit price of " +
    					"a dress is not a valid amount." +
    					"\nYou must enter only a natural or a decimal number. " + 
    					"For more information, please consult the management to get " +
    					"the right prices.");
    			}
    
    			. . . No Change
    
    		}
    
    		private void btnTax_Click(object sender, System.EventArgs e)
    		{
    			try 
    			{
    				// Get the tax rate
    				TaxRate =  decimal.Parse(this.txtTaxRate.Text) / 100;
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you entered is not recognized as a valid tax rate." +
    					"\nA valid tax rate is a value between 0 and 100.00" +
    					"\nPlease try again.");
    			}
    
    			. . . No Change
    
    		}
    
    		private void btnDifference_Click(object sender, System.EventArgs e)
    		{
    			decimal AmountTended = 0.00M, Difference;
    
    			try 
    			{
    				// Request money for the order
    				AmountTended    = decimal.Parse(this.txtAmountTended.Text);
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you entered for the amount tended is not valid. " +
    					"Only natural or decimal numbers are allowed." + 
    					"Please try again.");
    			}
    
    			. . . No Change
    
    		}
    	}
    }
  2. Execute the application and return to your programming environment

OverflowException

A computer application receives, processes, and produces values on a regular basis as the program is running. To better manage these values, as we saw when studying variables and data types in Lesson 2, the compiler uses appropriate amounts of space to store its values. It is not unusual that either you the programmer or a user of your application provides an value that is beyond the allowed range based on the data type. For example, we saw that a byte uses 8 bits to store a value and a combination of 8 bits can store a number no more than 255. If you provide a value higher than 255 to be stored in a byte, you get an error. Consider the following program:

private void button1_Click(object sender, System.EventArgs e)
{
	byte NumberOfPages;

	NumberOfPages = byte.Parse(this.textBox1.Text);

	this.textBox2.Text = NumberOfPages.ToString();
}

When a value beyond the allowable range is asked to be stored in memory, the compiler produces (the verb is "throws" as we will learn soon) an error of the OverflowException class. Here is an example of running the program with a bad number:

As with the other errors, when this exception is thrown, you should take appropriate action.

ArgumentOutOfRangeException

Once again, we know that a value is passed to the Parse() method of its data type for analysis. For a primitive data type, the Parse() method scans the string and if the string cannot be converted into a valid character or number, the compiler usually throws a FormatException exception as we saw above. Other classes such as DateTime also use a Parse() method to scan the value submitted to it. For example, if you request a date value from the user, the DateTime.Parse() method scans the string to validate it. In US English, Parse() expects the user to type a string in the form m/d/yy or mm/dd/yy or mm/dd/yyyy. Consider the following program:

private void button1_Click(object sender, System.EventArgs e)
{
	DateTime DateHired;

	DateHired = DateTime.Parse(this.textBox1.Text);

	this.textBox2.Text = DateHired.ToString();
}

If the user types a value that cannot be converted into a valid date, the compiler throws an ArgumentOutOfRangeException exception. Here is an example of running the above program with an invalid date:

One way you can avoid this is to guide the user but still take appropriate actions, just in case this error is thrown.

Practical LearningPractical Learning: Using an ArgumentOutOfRangeException Exception

  1. Change the Click event of the Process button as follows:
     
    using System;
    using System.Drawing;
    using System.Collections;
    using System.ComponentModel;
    using System.Windows.Forms;
    using System.Data;
    
    namespace GCS4
    {
    	/// <summary>
    	/// Summary description for Form1.
    	/// </summary>
    	public class Form1 : System.Windows.Forms.Form
    	{
    		
    		. . . No Change
    		
    		private System.ComponentModel.Container components = null;
    
    		// Order Information
    		string CustomerName;
    		DateTime OrderDate;
    
    		// Quantities of items
    		uint NumberOfShirts;
    		uint NumberOfPants;
    		uint NumberOfDresses;
    
    		// Price of items
    		decimal PriceOneShirt, PriceAPairOfPants, PriceOneDress;
    
    		// Each of these sub totals will be used for cleaning items
    		decimal SubTotalShirts, SubTotalPants, SubTotalDresses;
    
    		// Values used to process an order
    		decimal TaxRate;
    		decimal TotalOrder, TaxAmount, SalesTotal;
    		private System.Windows.Forms.Button btnProcess;
    
    		public Form1()
    		{
    			//
    			// Required for Windows Form Designer support
    			//
    			InitializeComponent();
    
    			//
    			// TODO: Add any constructor code after InitializeComponent call
    			//
    		}
    
    		. . . No Change
    		
    
    		private void btnProcess_Click(object sender, System.EventArgs e)
    		{
    			if( this.btnProcess.Text == "Process" )
    			{
    				this.Height = 408;
    				this.btnProcess.Text = "Reset";
    			}
    			else
    			{
    				this.Height = 232;
    				this.txtCustomer1.Text = "";
    				this.txtMM.Text = "1";
    				this.txtDD.Text = "1";
    				this.txtYYYY.Text = "1970";
    				this.txtQtyShirts.Text = "0";
    				this.txtQtyPants.Text  = "0";
    				this.txtQtyDresses.Text = "0";
    				this.txtSubTotalShirts.Text = "$0.00";
    				this.txtSubTotalPants.Text  = "$0.00";
    				this.txtSubTotalDresses.Text = "$0.00";
    
    				this.btnProcess.Text = "Process";
    			}
    			
    			// Request order information from the user
    			CustomerName = this.txtCustomer1.Text;
    			
    			try 
    			{
    				int mm, dd, yyyy;
    				mm   = int.Parse(this.txtMM.Text);
    				dd   = int.Parse(this.txtDD.Text);
    				yyyy = int.Parse(this.txtYYYY.Text);
    				OrderDate = new DateTime(yyyy, mm, dd);
    			}
    			catch(ArgumentOutOfRangeException)
    			{
    				MessageBox.Show("The date you entered is not valid - Please try again!");
    			}
    
    			// Request the quantity of each category of items
    			try 
    			{
    				// Number of Shirts
    				NumberOfShirts  = uint.Parse(this.txtQtyShirts.Text);
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you typed for the number of " +
    					            "shirts is not a valid number." +
    						"\nPlease enter a natural number such as 2 or 24 or even 248");
    			}
    
    			. . . No Change 
    			
    
    			// Display the receipt
    			this.txtCustomer2.Text  = CustomerName;
    			this.txtOrderDate.Text  = OrderDate.ToString("D");
    			this.txtTotalOrder.Text = TotalOrder.ToString("C");
    		}
    	}
    }
  2. Execute the application.
  3. To test it enter valid and invalid values for the controls. Here is an example:
     

  4. Close the form and return to your programming environment

DivideByZeroException

Division by zero is an operation to always avoid. It is so important that it is one of the most fundamental exceptions of the computer. It is addressed at the core level even by the Intel and AMD processors. It is also addressed by the operating systems at their level. It is also addressed by most, if not all, compilers. It is also addressed by most, if not, all libraries. This means that this exception is never welcomed anywhere. The .NET Framework also provides it own class to face this operation.

If an attempt to divide a value by 0, the compiler throws a DivideByZeroException exception. We will see an example later.

Techniques of Using Exceptions

 

Throwing an Exception

As mentioned above, the Exception class is equipped with a Message property that carried a message for the error that occurred. We also mentioned that the message of this property may not be particularly useful to a user. Fortunately, you can create your own message and pass it to the Exception. To be able to receive custom messages, the Exception class provides the following constructor:

public Exception(string message);

To use it, in the section that you are anticipating the error, type the throw keyword followed by a new instance of the Exception class using the constructor that takes a string. Here is an example:

private void button1_Click(object sender, System.EventArgs e)
		{
			double Operand1, Operand2;
			double Result = 0.00;
			string Operator;

			try 
			{
				Operand1 = double.Parse(this.txtNumber1.Text);
				Operator = this.txtOperator.Text;
				Operand2 = double.Parse(this.txtNumber2.Text);

				if( Operator != "+" && Operator != "-" && Operator != "*" && Operator != "/" )
					throw new Exception(Operator);

				switch(Operator)
				{
					case "+":
						Result = Operand1 + Operand2;
						break;
					case "-":
						Result = Operand1 - Operand2;
						break;
					case "*":
						Result = Operand1 * Operand2;
						break;
					case "/":
						Result = Operand1 / Operand2;
						break;
					default:
						MessageBox.Show("Bad Operation");
						break;
				}

				this.txtResult.Text = Result.ToString();
			}
			catch(Exception ex)
			{
				MessageBox.Show("Operation Error: " + ex.Message + " is not a valid operator");
			}
		}
Calculator 1: Exception Handling
 

Catching Various Exceptions

In the above examples, when we anticipated some type of problem, we instructed the compiler to use our default catch section. We left it up to the compiler to find out when there was a problem and we provided a catch section to deal with it. A method with numerous or complex operations and requests can also produce different types of errors. With such a type of program, you should be able to face different problems and deal with them individually, each by its own kind. To do this, you can create different catch sections, each made for a particular error. The formula used would be:

try {
	// Code to Try
}
catch(Arg1)
{
	// One Exception
}
catch(Arg2)
{
	// Another Exception
}

The compiler would proceed in a top-down:

  1. Following the normal flow of the program, the compiler enters the try block
  2. If no exception occurs in the try block, the rest of the try block is executed
    If an exception occurs in the try block, the compiler registers the type of error that occurred. If there is a throw line, the compiler registers it also:
    1. The compiler gets out of the try section
    2. The compiler examines the first catch. If the first catch matches the thrown error, that catch executes and the exception handling routine may seize. If the first catch doesn’t match the thrown error, the compiler proceeds with the next catch
    3. The compiler checks the next match, if any and proceeds as in the first match. This continues until the compiler finds a catch that matches the thrown error
    4. If one of the catches matches the thrown error, its body executes. If no catch matches the thrown error, the compiler calls the Exception class and uses the default message

Multiple catches are written if or when a try block is expected to throw different types of errors. For example, in our calculator, we want to consider only the addition, the subtraction, the multiplication, and the division. It is also likely that the user may type one or two invalid numbers. This leads us to know that our program can produce at least two types of errors at this time. Based on this, we can address them using two catch clauses as follows:

private void button1_Click(object sender, System.EventArgs e)
		{
			double Operand1, Operand2;
			double Result = 0.00;
			string Operator;

			try 
			{
				Operand1 = double.Parse(this.txtNumber1.Text);
				Operator = this.txtOperator.Text;
				Operand2 = double.Parse(this.txtNumber2.Text);

				if( Operator != "+" && Operator != "-" && Operator != "*" && Operator != "/" )
					throw new Exception(Operator);

				switch(Operator)
				{
					case "+":
						Result = Operand1 + Operand2;
						break;
					case "-":
						Result = Operand1 - Operand2;
						break;
					case "*":
						Result = Operand1 * Operand2;
						break;
					case "/":
						Result = Operand1 / Operand2;
						break;
					default:
						MessageBox.Show("Bad Operation");
						break;
				}

				this.txtResult.Text = Result.ToString();
			}
			catch(FormatException)
			{
				MessageBox.Show("You type an invalid number. Please correct it");
			}
			catch(Exception ex)
			{
				MessageBox.Show("Operation Error: " + ex.Message + " is not a valid operator");
			}
		}
Catching multiple exceptions

This program works fine as long as the user types two valid numbers and a valid arithmetic operator. Anything else, such an invalid number or an unexpected operator would cause an error to be thrown:

Obviously various bad things could happen when this program is running. Imagine that the user wants to perform a division. You need to tell the compiler what to do if the user enters the denominator as 0 (or 0.00). If this happens, one of the options you should consider is to display a message and get out. Fortunately, the .NET Framework provides the DivideByZeroException class to deal with an exception caused by division by zero. As done with the message passed to the Exception class, you can compose your own message and pass it to the DivideByZeroException(string message) constructor.

Exception is the parent of all exception classes. This corresponds to the three periods of a C++' catch(...) block. Therefore, if you write various catch blocks, the one that takes the Exception as argument should be the last.

Here is an example that catches two types of exceptions:

private void button1_Click(object sender, System.EventArgs e)
		{
			double Operand1, Operand2;
			double Result = 0.00;
			string Operator;

			try 
			{
				Operand1 = double.Parse(this.txtNumber1.Text);
				Operator = this.txtOperator.Text;
				Operand2 = double.Parse(this.txtNumber2.Text);

				if( Operator != "+" && Operator != "-" && Operator != "*" && Operator != "/" )
					throw new Exception(Operator);

				switch(Operator)
				{
					case "+":
						Result = Operand1 + Operand2;
						break;
					case "-":
						Result = Operand1 - Operand2;
						break;
					case "*":
						Result = Operand1 * Operand2;
						break;
					case "/":
						if( Operand2 == 0 )
							throw new DivideByZeroException("Division by zero is not allowed");
						Result = Operand1 / Operand2;
						break;
					default:
						MessageBox.Show("Bad Operation");
						break;
				}

				this.txtResult.Text = Result.ToString();
			}
			catch(FormatException)
			{
				MessageBox.Show("You type an invalid number. Please correct it");
			}
			catch(DivideByZeroException ex)
			{
				Console.WriteLine(ex.Message);
			}
			catch(Exception ex)
			{
				MessageBox.Show("Operation Error: " + ex.Message + " is not a valid operator");
			}
		}
 

Practical LearningPractical Learning: Identifying the Thrown Exception

  1. To catch various exceptions, change the form's file as follows:
     
    using System;
    using System.Drawing;
    using System.Collections;
    using System.ComponentModel;
    using System.Windows.Forms;
    using System.Data;
    
    namespace GCS4
    {
    	/// <summary>
    	/// Summary description for Form1.
    	/// </summary>
    	public class Form1 : System.Windows.Forms.Form
    	{
    		. . . No Change
    
    		private void btnProcess_Click(object sender, System.EventArgs e)
    		{
    			. . . No Change
    			
    			// Request order information from the user
    			CustomerName = this.txtCustomer1.Text;
    			
    			try 
    			{
    				int mm, dd, yyyy;
    				mm   = int.Parse(this.txtMM.Text);
    				dd   = int.Parse(this.txtDD.Text);
    				yyyy = int.Parse(this.txtYYYY.Text);
    				OrderDate = new DateTime(yyyy, mm, dd);
    			}
    			// This exception is thrown if the user types a value that cannot
    			// be converted into a valid number
    			catch(FormatException ex)
    			{
    				MessageBox.Show("Error: " + ex.Message +
    					"\nThe value you entered " +
    					"is not a valid number");
    			}
    			// This exception is thrown if the values that user had typed cannot
    			// produce a valid date value
    			catch(ArgumentOutOfRangeException)
    			{
    				MessageBox.Show("The date you entered is not valid - Please try again!");
    			}
    
    			// Request the quantity of each category of items
    			try 
    			{
    				// Number of Shirts
    				NumberOfShirts  = uint.Parse(this.txtQtyShirts.Text);
    			}
    			// This exception is thrown if the user typed a value that cannot
    			// be recognized as a valid number
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you typed for the number of " +
    					            "shirts is not a valid number." +
    								"\nPlease enter a natural number such as 2 or 24 or even 248");
    			}
    			// This exception is thrown if the user types a negative value
    			catch(OverflowException)
    			{
    				MessageBox.Show("The number you typed is negative but " +
    					"we cannot accept a negative number of shirts");
    			}
    
    			try 
    			{
    				// Number of Pants
    				NumberOfPants   = uint.Parse(this.txtQtyPants.Text);
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you typed for the number of " +
    					"pair or pants is not a valid number." +
    					"\nPlease enter a natural number such as 2 or 24 or even 248");
    			}
    			catch(OverflowException)
    			{
    				MessageBox.Show("The number you typed is negative but " +
    					"we cannot accept a negative number of shirts");
    			}
    			try 
    			{
    				// Number of Dresses
    				NumberOfDresses = uint.Parse(this.txtQtyDresses.Text);
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you typed for the number of " +
    					"dresses is not a valid number." +
    					"\nPlease enter a natural number such as 2 or 24 or even 248");
    			}
    			catch(OverflowException)
    			{
    				MessageBox.Show("The number you typed is negative but " +
    					"we cannot accept a negative number of shirts");
    			}
    
    			// Unit Prices of items
    			try 
    			{
    				PriceOneShirt     = decimal.Parse(this.txtUnitPriceShirts.Text);
    				if(PriceOneShirt < 0 )
    					throw new Exception("Negative numbers are not allowed for the price of a shirt");
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you entered for the unit price of " +
    					"a shirt is not a recognizable currency amount." +
    					"\nOnly natural or decimal numbers are allowed. " + 
    					"Please consult the management to know the valid prices.");
    			}
    			catch(Exception ex)
    			{
    				MessageBox.Show(ex.Message);
    			}
    			try 
    			{
    				PriceAPairOfPants = decimal.Parse(this.txtUnitPricePants.Text);
    				if( PriceAPairOfPants < 0 )
    					throw new Exception("Negative numbers are not allowed for the price of a pair of pants");
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you entered for the unit price of " +
    					"a pair of pants is not a recognizable currency amount." +
    					"\nOnly natural or decimal numbers are allowed. " + 
    					"You can consult the management to find out about " +
    					"the allowable prices.");
    			}
    			try 
    			{
    				PriceOneDress     = decimal.Parse(this.txtUnitPriceDresses.Text);
    				if( PriceOneDress < 0 )
    					throw new Exception("Negative numbers are not allowed for the price of a dress");
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you entered for the unit price of " +
    					"a dress is not a valid amount." +
    					"\nYou must enter only a natural or a decimal number. " + 
    					"For more information, please consult the management to get " +
    					"the right prices.");
    			}
    			catch(Exception ex)
    			{
    				MessageBox.Show(ex.Message);
    			}
    
    			. . . No Change
    		}
    
    		private void btnTax_Click(object sender, System.EventArgs e)
    		{
    			try 
    			{
    				// Get the tax rate
    				TaxRate =  decimal.Parse(this.txtTaxRate.Text) / 100;
    				if(PriceOneShirt < 0 )
    					throw new Exception("Negative numbers are not allowed for a tax rate");
    			}
    			catch(FormatException)
    			{
    				MessageBox.Show("The value you entered is not recognized as a valid tax rate." +
    					"\nA valid tax rate is a value between 0 and 100.00" +
    					"\nPlease try again.");
    			}
    			catch(Exception ex)
    			{
    				MessageBox.Show(ex.Message);
    			}
    
    			// Calculate the tax amount using a constant rate
    			TaxAmount       = TotalOrder * TaxRate;
    			// Add the tax amount to the total order
    			SalesTotal      = TotalOrder + TaxAmount;
    
    			this.txtTaxAmount.Text  = TaxAmount.ToString("C");
    			this.txtNetPrice.Text   = SalesTotal.ToString("C");
    		}
    
    		private void btnDifference_Click(object sender, System.EventArgs e)
    		{
    			. . . No Change
    		}
    	}
    }
  2. Test the application and return to your programming environment

Exceptions Nesting

The calculator simulator we have studied so far performs a division as one of its assignments. We learned that, in order to perform any operation, the compiler must first make sure that the user has entered a valid operator. Provided the operator is one of those we are expecting, we also must make sure that the user typed valid numbers. Even if these two criteria are met, it was possible that the user enter 0 for the denominator. The block that is used to check for a non-zero denominator depends on the exception that validates the operators. The exception that could result from a zero denominator depends on the user first entering a valid number for the denominator.

You can create an exception inside of another. This is referred to as nesting an exception. This is done by applying the same techniques we used to nest conditional statements. This means that you can write an exception that depends on, and is subject to, another exception. To nest an exception, write a try block in the body of the parent exception. The nested try block must be followed by its own catch(es) clause. To effectively handle the exception, make sure you include an appropriate throw in the try block. Here is an example:

private void button1_Click(object sender, System.EventArgs e)
		{
			double Operand1, Operand2;
			double Result = 0.00;
			string Operator;

			try 
			{
				Operand1 = double.Parse(this.txtNumber1.Text);
				Operator = this.txtOperator.Text;
				Operand2 = double.Parse(this.txtNumber2.Text);

		if( Operator != "+" && Operator != "-" && Operator != "*" && Operator != "/" )
					throw new Exception(Operator);

				switch(Operator)
				{
					case "+":
						Result = Operand1 + Operand2;
						this.txtResult.Text = Result.ToString();
						break;
					case "-":
						Result = Operand1 - Operand2;
						this.txtResult.Text = Result.ToString();
						break;
					case "*":
						Result = Operand1 * Operand2;
						this.txtResult.Text = Result.ToString();
						break;
					case "/":
						try 
						{
							if( Operand2 == 0 )
			throw new DivideByZeroException("Division by zero is not allowed");
						Result = Operand1 / Operand2;
						this.txtResult.Text = Result.ToString();
						}
						catch(DivideByZeroException ex)
						{
							Console.WriteLine(ex.Message);
						}
						break;
					default:
						MessageBox.Show("Bad Operation");
						break;
				}

			}
			catch(FormatException)
			{
			MessageBox.Show("You type an invalid number. Please correct it");
			}
			catch(Exception ex)
			{
		MessageBox.Show("Operation Error: " + ex.Message + " is not a valid operator");
			}
		}

Exceptions and Methods

One of the most effective techniques used to deal with code is to isolate assignments. We have learned this when studying methods of classes. For example, the switch statement that was performing the operations in the “normal” version of our program can be written as follows:

private double Calculation(double Value1, double Value2, char Symbol)
	{
		double Result = 0.00;

		switch(Symbol)
		{
			case '+':
				Result = Value1 + Value2;
				break;
			case '-':
				Result = Value1 - Value2;
				break;
			case '*':
				Result = Value1 * Value2;
				break;
			case '/':
				Result = Value1 / Value2;
				break;
		}

		return Result;
	}

	private void button1_Click(object sender, System.EventArgs e)
	{
		double Operand1, Operand2;
		double Result = 0.00;
		char Operator;

		try 
		{
			Operand1 = double.Parse(this.txtNumber1.Text);
			Operator = char.Parse(this.txtOperator.Text);
			Operand2 = double.Parse(this.txtNumber2.Text);

			if( Operator != '+' && Operator != '-' && 
				Operator != '*' && Operator != '/' )
				throw new Exception(Operator.ToString());

		if( Operator == '/' )
			if( Operand2 == 0 )
			throw new DivideByZeroException("Division by zero is not allowed");
			Result = Operand1 / Operand2;
			this.txtResult.Text = Result.ToString();
				
			Result = Calculation(Operand1, Operand2, Operator);
			this.txtResult.Text = Result.ToString();
		}
	catch(FormatException)
	{
		MessageBox.Show("You type an invalid number. Please correct it");
	}
	catch(DivideByZeroException ex)
	{
		MessageBox.Show(ex.Message);
	}
	catch(Exception ex)
	{
		MessageBox.Show("Operation Error: " + ex.Message + " is not a valid operator");
	}
}

This is an example of running the program:

You can still use regular methods along with methods that handle exceptions. Here is an example:

private double Addition(double Value1, double Value2)
{
	double Result;
		Result = Value1 + Value2;
	return Result;
}

private double Subtraction(double Value1, double Value2)
{
	double Result;

	Result = Value1 - Value2;
	return Result;
}

private double Multiplication(double Value1, double Value2)
{
	double Result;

	Result = Value1 * Value2;
	return Result;
}

private double Division(double Value1, double Value2)
{
	double Result;

	Result = Value1 / Value2;
	return Result;
}

private void button1_Click(object sender, System.EventArgs e)
{
	double Operand1, Operand2;
	double Result = 0.00;
	char Operator;

	try 
	{
		Operand1 = double.Parse(this.txtNumber1.Text);
		Operator = char.Parse(this.txtOperator.Text);
		Operand2 = double.Parse(this.txtNumber2.Text);

		if( Operator != '+' && Operator != '-' && 
			Operator != '*' && Operator != '/' )
			throw new Exception(Operator.ToString());

		if( Operator == '/' )
			if( Operand2 == 0 )
			throw new DivideByZeroException("Division by zero is not allowed");
				
		switch(Operator)
		{
			case '+':
				Result = Addition(Operand1, Operand2);
				break;
			case '-':
				Result = Subtraction(Operand1, Operand2);
				break;
			case '*':
				Result = Multiplication(Operand1, Operand2);
				break;
			case '/':
				Result = Division(Operand1, Operand2);
				break;
		}

		this.txtResult.Text = Result.ToString();
	}
	catch(FormatException)
	{
		MessageBox.Show("You type an invalid number. Please correct it");
	}
	catch(DivideByZeroException ex)
	{
		MessageBox.Show(ex.Message);
	}
	catch(Exception ex)
	{
		MessageBox.Show("Operation Error: " + ex.Message + " is not a valid operator");
	}
}

As done in Main(), any method of a program can take care of its own exceptions that would occur in its body. Isolating assignments and handing them to methods is an important matter in the area of application programming. Consider the following Division method:

private double Addition(double Value1, double Value2)
{
	double Result;

	Result = Value1 + Value2;
	return Result;
}

private double Subtraction(double Value1, double Value2)
{
	double Result;

	Result = Value1 - Value2;
	return Result;
}

private double Multiplication(double Value1, double Value2)
{
	double Result;

	Result = Value1 * Value2;
	return Result;
}

private double Division(double Value1, double Value2)
{
	double Result;

	try 
	{
		if( Value2 == 0 )
			throw new DivideByZeroException("Division by zero is not allowed");
				
		Result = Value1 / Value2;
		return Result;
	}
	catch(DivideByZeroException ex)
	{
		MessageBox.Show(ex.Message);
	}

	return 0.00;
}

private void button1_Click(object sender, System.EventArgs e)
{
	double Operand1, Operand2;
	double Result = 0.00;
	char Operator;

	try 
	{
		Operand1 = double.Parse(this.txtNumber1.Text);
		Operator = char.Parse(this.txtOperator.Text);
		Operand2 = double.Parse(this.txtNumber2.Text);

		if( Operator != '+' && Operator != '-' && 
			Operator != '*' && Operator != '/' )
			throw new Exception(Operator.ToString());

		switch(Operator)
		{
			case '+':
				Result = Addition(Operand1, Operand2);
				break;
			case '-':
				Result = Subtraction(Operand1, Operand2);
				break;
			case '*':
				Result = Multiplication(Operand1, Operand2);
				break;
			case '/':
				Result = Division(Operand1, Operand2);
				break;
		}

		this.txtResult.Text = Result.ToString();
	}
	catch(FormatException)
	{
		MessageBox.Show("You type an invalid number. Please correct it");
	}
	catch(Exception ex)
	{
		MessageBox.Show("Operation Error: " + ex.Message + " is not a valid operator");
	}
}

One of the ways you can use methods in exception routines is to have a central method that receives variables, and sends them to an external method. The external method tests the value of a variable. If an exception occurs, the external method displays or sends a throw. This throw can be picked up by the method that sent the error.

 

Home Copyright © 2004-2010 FunctionX, Inc.