Debugging Separate Files

Introduction

As you know already, a program can use many files, some files come from the .NET Framework, some are created by Microsoft Visual Studio when you start a project, and you create the others as you judge them necessary for your project. As a result, when debugging, you can consider files that are linked at one time or another. The process of debugging is primarily the same. You just have to keep in mind that you are dealing with many classes and probably different files. This has some consequences on the results you see.

Practical LearningPractical Learning: Debugging With a Class

  1. Start Microsoft Visual Studio
  2. In the Visual Studio 2022 dialog box, click Create a New Project
  3. In the Create a New Project dialog box, make sure the languages combo box is displaying C#.
    In the list of project templates, make sure Console App is selected.
    Click Next
  4. Change the Project Name to WattsALoan2
  5. Click Next
  6. In the Additional Information combo box, make sure the highest version of the is selected in the Framework combo box.
    Click Create
  7. To create a folder, in the Solution Explorer, right-click WattsALoan2 -> Add -> New Folder
  8. Type Models and press Enter
  9. To create a new class, in the Solution Explorer, right-click Models -> Add -> Class...
  10. Change the Name to Customer
  11. Click Add
  12. Change the class as follows:
    public class Customer
    {
        public string FullName;
        public string PhoneNumber;
    
        public Customer(string name = "John Doe", string phone = "000-000-0000")
        {
            FullName = name;
            PhoneNumber = phone;
        }
    }
  13. To create another class, in the Solution Explorer, right-click Models -> Add -> Class...
  14. Type Employee
  15. Press Enter
  16. Change the class as follows:
    public class Employee
    {
        public long EmployeeNumber;
        public string FirstName;
        public string LastName;
        public string Title;
    
        public Employee(long emplNbr = 0,
                        string fName = "Unknown",
                        string lName = " Not Specified",
                        string position = "Loan Specialist")
        {
            EmployeeNumber = emplNbr;
            FirstName = fName;
            LastName = lName;
            Title = position;
        }
    
        public string GetEmployeeName()
        {
            return LastName + ", " + FirstName;
        }
    }
  17. To add another class to the project, in the Solution Explorer, right-click Models -> Add -> Class...
  18. Change the Name to LoanInformation
  19. Click Add
  20. Change the class as follows:
    public class LoanInformation
    {
        public double Principal;
        public double InterestRate;
        public double Period;
        public double InterestAmount;
        public double FutureValue;
    }
  21. In the Solution Explorer, right-click Programa.cs -> Rename
  22. Type LoanEvaluation (to get LoanEvaluation.cs) and press Enter
  23. Change the document as follows:
    public class LoanEvaluation
    {
        private Employee clerk;
        private Customer client;
        private LoanInformation loan;
    
        public LoanEvaluation()
        {
            clerk = new Employee();
            client = new Customer();
            loan = new LoanInformation();
        }
    
        private void IdentifyEmployee()
        {
            Console.WriteLine("Enter the following pieces of information " +
                              "about the employee who prepared this loan.");
            Console.Write("Employee #: ");
            clerk.EmployeeNumber = long.Parse(Console.ReadLine());
            Console.Write("First Name: ");
            clerk.FirstName = Console.ReadLine();
            Console.Write("Last Name:  ");
            clerk.LastName = Console.ReadLine();
            Console.Write("Title:      ");
            clerk.Title = Console.ReadLine();
        }
    
        private void IdentifyCustomer()
        {
            Console.WriteLine("Enter the following pieces of information " +
                              "about the customer for whom this loan was prepared.");
            Console.Write("Customer Name: ");
            client.FullName = Console.ReadLine();
            Console.Write("Phone Number:  ");
            client.PhoneNumber = Console.ReadLine();
        }
    
        private void GetLoanValues()
        {
            Console.WriteLine("Enter the following pieces of information " +
                              "about the values used for the loan.");
    
            Console.Write("Enter the principal: ");
            loan.Principal = double.Parse(Console.ReadLine());
      
            Console.Write("Enter the interest rate: ");
            loan.InterestRate = double.Parse(Console.ReadLine()) / 100;
    
            Console.Write("Enter the number of months: ");
            loan.Period = double.Parse(Console.ReadLine()) / 12;
        }
    
        public void Show()
        {
            loan.InterestAmount = loan.Principal * loan.InterestRate * loan.Period;
            loan.FutureValue = loan.Principal + loan.InterestAmount;
    
            Console.WriteLine("======================================");
            Console.WriteLine("Loan Summary");
            Console.WriteLine("=------------------------------------=");
            Console.WriteLine("Prepared by:  {0} - {1}\n              {2}",
                              clerk.EmployeeNumber,
                             clerk.GetEmployeeName(), clerk.Title);
            Console.WriteLine("=------------------------------------=");
            Console.WriteLine("Prepared for: {0}\n              {1}",
                              client.FullName, client.PhoneNumber);
            Console.WriteLine("=------------------------------------=");
            Console.WriteLine("Principal:       {0:F}", loan.Principal);
            Console.WriteLine("Interest Rate:   {0:P}", loan.InterestRate);
            Console.WriteLine("Period For:      {0} months", loan.Period * 12);
            Console.WriteLine("Interest Amount: {0:F}", loan.InterestAmount);
            Console.WriteLine("Future Value:    {0:F}", loan.FutureValue);
            Console.WriteLine("======================================");
        }
    }

Using Separate Files

Instead of just one class or one file, we have learned, and will continue learning, that, to organize your project, you can create the classes of your project in different files. When you debug a project that uses different classes or files that contain classes, the debugger is aware of the objects and where they are created. As we will learn later, there are various windows that assist you with identifying the objects of your project. As a result, the tools such as the Locals window display their contents accordingly.

As you know already, the starting point of a C# application is the main document (in previous C# console application, that would be a class that contains a method named Main). If you start debugging an application that uses many code files (files that contain classes), the debugger must first identify the primary document of the project (if you are debugging a normal console application but the debugger cannot find a file that contains the Main() function, you would receive an error, indicating that your project does not contain an entry point).

Practical LearningPractical Learning: Debugging Classes

  1. Change the LoanEvaluation.cs file as follows:
    LoanEvaluation evaluation = new LoanEvaluation();
    
    Console.Title = "Watts A Loan?";
    
    Console.WriteLine("This application allows you to evaluate a loan");
    evaluation.IdentifyEmployee();
    evaluation.IdentifyCustomer();
    
    Console.Clear();
    
    evaluation.GetLoanValues();
    
    Console.Clear();
    
    evaluation.Show();
    
    public class LoanEvaluation
    {
        private Employee clerk;
        private Customer client;
        private LoanInformation loan;
    
        public LoanEvaluation()
        {
            clerk = new Employee();
            client = new Customer();
            loan = new LoanInformation();
        }
    
        private void IdentifyEmployee()
        {
            Console.WriteLine("Enter the following pieces of information " +
                              "about the employee who prepared this loan.");
            Console.Write("Employee #: ");
            clerk.EmployeeNumber = long.Parse(Console.ReadLine());
            Console.Write("First Name: ");
            clerk.FirstName = Console.ReadLine();
            Console.Write("Last Name:  ");
            clerk.LastName = Console.ReadLine();
            Console.Write("Title:      ");
            clerk.Title = Console.ReadLine();
        }
    
        private void IdentifyCustomer()
        {
            Console.WriteLine("Enter the following pieces of information " +
                              "about the customer for whom this loan was prepared.");
            Console.Write("Customer Name: ");
            client.FullName = Console.ReadLine();
            Console.Write("Phone Number:  ");
            client.PhoneNumber = Console.ReadLine();
        }
    
        private void GetLoanValues()
        {
            Console.WriteLine("Enter the following pieces of information " +
                              "about the values used for the loan.");
    
            Console.Write("Enter the principal: ");
            loan.Principal = double.Parse(Console.ReadLine());
      
            Console.Write("Enter the interest rate: ");
            loan.InterestRate = double.Parse(Console.ReadLine()) / 100;
    
            Console.Write("Enter the number of months: ");
            loan.Period = double.Parse(Console.ReadLine()) / 12;
        }
    
        public void Show()
        {
            loan.InterestAmount = loan.Principal * loan.InterestRate * loan.Period;
            loan.FutureValue = loan.Principal + loan.InterestAmount;
    
            Console.WriteLine("======================================");
            Console.WriteLine("Loan Summary");
            Console.WriteLine("=------------------------------------=");
            Console.WriteLine("Prepared by:  {0} - {1}\n              {2}",
                              clerk.EmployeeNumber,
                             clerk.GetEmployeeName(), clerk.Title);
            Console.WriteLine("=------------------------------------=");
            Console.WriteLine("Prepared for: {0}\n              {1}",
                              client.FullName, client.PhoneNumber);
            Console.WriteLine("=------------------------------------=");
            Console.WriteLine("Principal:       {0:F}", loan.Principal);
            Console.WriteLine("Interest Rate:   {0:P}", loan.InterestRate);
            Console.WriteLine("Period For:      {0} months", loan.Period * 12);
            Console.WriteLine("Interest Amount: {0:F}", loan.InterestAmount);
            Console.WriteLine("Future Value:    {0:F}", loan.FutureValue);
            Console.WriteLine("======================================");
        }
    }
  2. To restart debugging, press F11.
    If you don't see the Locals window, to display it, on the main menu, click Debug -> Window -> Locals
  3. Move the windows to make sure you can see the Code Editor, the Locals window, and the DOS window
  4. Notice that the debugging yellow arrow indicator is positioned on the left of the opening curly bracket of the Main() function.
    Notice that at this time, the Locals window contains only the evaluation variable whose value is null.
    At this time, the DOS window shows a blinking caret
  5. To continue debugging, on the main menu, click Debug -> Step Into
  6. Again, on the main menu, click Debug -> Step Into.
    Notice that the debugger is in the LoanEvaluation.cs file and to its constructor
  7. In the Locals window, click the + button of this to expand it. Notice that it show the global variables of the class
  8. To continue debugging, on the Standard toolbar, click the Step Into button Step Into three times.
    Notice that the excution switches to the constructor of the Employee class in the Employee.cs file
  9. In the Locals window, click the + button of this to expand
  10. To continue debugging, press F11 seven times.
    The debugging returns to the LoanEvaluation.cs file and focuses on the clerk variable that is of type Employee
  11. On the Standard toolbar, click the Step Into button Step Into.
    The debugging moves to the constructor of the Customer class in the Customer.cs file
  12. press F11 six times.
    The debugging returns to the client variable in the LoanEvaluation.cs file
  13. press F11 three times.
    The debugging returns to the evaluation variable in the Main() function
  14. press F11 twice
    Notice that the title bar of the DOS window has changed
  15. In the Locals window, click all the + buttons to show the current values of the variables for the Employee, the Customer, and the LoanInformation variables
  16. Keep pressing F11 until the title bar of the DOS window indicates that it has focus and you are asked to enter an employee number
  17. When asked for an employee number, type 20584 and press Enter
  18. The focus moves back to the Code Editor.
    Notice that the value of the employee number has changed in the Locals window
    press F11 twice.
    The focus moves back to the DOS window
  19. For the First Name, type Joshua and press Enter
  20. The focus moves back to the Code Editor.
    press F11 twice.
    The focus moves back to the DOS window
  21. For the Last Name, type Mahmouda and press Enter
  22. The focus moves back to the Code Editor.
    press F11 twice.
    The focus moves back to the DOS window
  23. For the Title, type Account Representative and press Enter
  24. The focus moves back to the Code Editor.
    press F11 seven times.
    The focus moves back to the DOS window
  25. When asked to provide a customer name, type Lauren Sachs and press Enter
  26. The focus moves back to the Code Editor.
    press F11 twice.
    The focus moves back to the DOS window
  27. For the Phone Number, type (301) 438-6243 and press Enter
  28. The focus moves back to the Code Editor.
    press F11 many times until the DOS window receives focus
  29. value for the principal, type 24750 and press Enter
  30. The focus moves to the Code Editor.
    Notice that the value of the principal has changed in the Locals grid.
    Press F11
  31. press F11 again
  32. When the focus moves to the DOS window, when you are asked to provide the interest rate, type 11.35 and press Enter
  33. press F11
  34. press F11 again
  35. The focus moves to the DOS window.
    For the number of months, type 60 and press Enter
  36. Keep pressing F11.
    Observe the values in the Locals window and see the new lines in the DOS window
  37. Continue pressing F8 until the DOS window closes
  38. Make sure the LoanEvaluation.cs file is displaying.
    Click the margin on the left side of loan.FutureValue = loan.Principal + loan.InterestAmount;
  39. On the main menu, click WINDOW -> Employee.cs
  40. Click the margin on the left site of return LastName + ", " + FirstName;
  41. On the main menu, click Debug -> Start Debugging
  42. In the DOS window, enter the values as follows and press Enter after each
     
    Employee #: 41920
    First Name: Donna
    Last Name: Jones
    Title: Account Manager
    Customer Name: Hermine Simms
    Customer Phone: 410-573-2031
    Principal: 3250
    Number of Months: 36

    The focus moves to the LoanEvaluation.cs file in the Code Editor
  43. Press F5 to continue.
    Debugging switches to the Employee.cs file
  44. Press F5 to see the result in the DOS window
  45. On the main menu, click FILE -> Close Solution
  46. When asked whether you want to save, click Discard

The Error List

You are probably familiar with the Error List window because if you have ever made any mistake in your code, it would come up. The Error List is a window that displays the list of the current errors in your code. Most of the times, the Error List is present, usually below the Code Editor. At any time, to display it, on the main menu, you can click VIEW -> Error List or press Ctrl + W. While you are working on your code, the Error List may be minimized. To permanently keep it showing, you can click its AutoHide button.

The Error List uses two sequences of how it decides to show the errors. While writing your code, if the live parser, which continuously runs while you are writing your code, finds a problem, it makes a list of all types of violations, even if there is only one mistake, or there appears to be only one mistake. It shows the list in a table:

Error List

Another sequence, or a different list, gets created if you build your code. This time, it is the debugger's list that would show:

Error List

It is important to know that one mistake could be violating more than one rule of the C# language.

As seen in our introduction to syntax errors, if you are using Microsoft Visual Studio, while you are writing your code, if the parser senses a problem, it underlines the section in the Code Editor and the Error List shows its analysis of that problem. The Error list uses a table made of 5 columns that are Description, File, Line, Column, and Project. You don't have to use all those columns. To specify what columns to show or hide, right-click anywhere in the body of the Error List and position the mouse on Show Columns:

Error List

To show a column, put a check mark on its menu item. To hide a column, remove its check mark. The list of errors displays in an incremental order specified by the parser. Otherwise, you can arrange the order based on a column of your choice. To do this, right-click any entry in the Error List, position the mouse on Sort By, and click the column of your choice.

As mentioned already, the Error List uses various columns to show its findings:

To jump to an error from the Error List, locate its entry in the Error List window and double-click it. The caret would be positioned to that Line number, that character Column or word, and in that File of that Project. If you correct the problem and if the parser concludes that your correction is fine, the error would be removed automatically from the Error List. You can continue this to correct all problems and, eventually, the Error List would be emptied.

Objects Assisting With Debugging

The Immediate Window

The Immediate window is a special text editor that can be used to test values, operations (calculations), variables, and methods. To display the Immediate window, on the main menu, you can click Debug -> Windows -> Immediate. The Immediate window appears as a blank object:

Immediate Window

To use it, you must write something. What you write depends on what you want to test. An expression you write should start with a question mark but in some cases you can omit that symbol. Because the Immediate window is a text editor, you can copy code from somewhere else and paste it in it. If the immediate window starts being crowded, to empty it, you can right-click inside the window and click Clear All.

After typing or pasting the expression, press Enter. The next line would show the result. For example, imagine you want to test an arithmetic operation such as the addition of 248.49 and 57.26, you would type ?248.49 + 57.26 (the empty spaces are optional and you can include as many as you want) and press Enter. Here is an example:

Immediate Window

If you want to test a variable or a method, you must first write code that has the variable. That is, before testing a variable, create a method or use the Main() function and declare the variable in it. If you try testing a variable that is not declared, you would receive an error. One way you can test a variable consists of assigning a certain value, probably a wrong value, to it and observe the result. You can start the assignment expression with a question mark but that mark is not necessary. Here is an example:

Immediate Window

The Immediate window allows you to test the value that a variable is currently holding. To get this information, in the Immediate window, type the name of the variable preceded by a question mark (you can also juste type the name of the variable) and press Enter:

Immediate Window

If the variable had previously received a value, when you enquire of it in the Immediate window, its current value would show:

Immediate Window

Another test you can perform on a variable consists of adding a value to it to increase it, or subtracting a value from it.

To test a method in the Immediate window, the method should return a value. To get the value that a mehod is currently returning, type its name in the Immediate window and press Enter. You can also precede the name of the method with a question mark. Here is an example:

Immediate Window

In the same way, you can create more elaborate methods and test them in the Immediate window.

Practical LearningPractical Learning: Using the Immediate Window

  1. To start a new project, on the main menu, click File -> New -> Project
  2. In the middle list, click Empty Project
  3. Change the Name to PayrollEvaluation1
  4. Click OK
  5. To create a file, in the Solution Explorer, right-click PayrollEvaluation1 -> Add -> New Item...
  6. In the middle list, click Code File
  7. Change the Name to Evaluation and click Add
  8. In the empty document, type the following:
    double grossPay = 0D;
    double timeWorked = 0D;
    double hourlySalary = 0.00D;
    // double federalIncomeTax = 0.00D;
    // const double FICATaxRate = 0.0765; // 7.65%
    // double FICATaxAmount = 0.00D;
    double totalDeductions = 0.0D;
    double netPay = 0.00D;
    
    Console.Title = "Payroll Evaluation";
    /*      
    	Console.WriteLine("To test a payroll, enter the following pieces of information");
            Console.Write("Time Wored:    ");
            timeWorked = double.Parse(Console.ReadLine());
            Console.Write("Hourly Salary: ");
            hourlySalary = double.Parse(Console.ReadLine());
            Console.Write("Enter the amount of the deductions: ");
            totalDeductions = double.Parse(Console.ReadLine());
            */
            
    grossPay = hourlySalary * timeWorked;
    netPay = grossPay - totalDeductions;
    Console.Clear();
    
    Console.WriteLine("=======================");
    Console.WriteLine("Payroll Evaluation");
    Console.WriteLine("-----------------------");
    Console.WriteLine("Time Worked:   {0, 7}", timeWorked.ToString("F"));
    Console.WriteLine("Hourly Salary: {0, 7}", hourlySalary.ToString("F"));
    Console.WriteLine("Gross Pay:     {0, 7}", grossPay.ToString("F"));
    Console.WriteLine("Deductions:    {0, 7}", totalDeductions.ToString("F"));
    Console.WriteLine("Net Pay:       {0, 7}", netPay.ToString("F"));
    Console.WriteLine("=======================\n");
  9. To execute the application, on the main menu, click Debug -> Start Debugging.
    Notice that all values are 0:

    Payroll Evaluation

  10. Press Enter to close the DOS window and return to your programming environment
  11. To step into code, on the main menu, click Debug -> Step Into
  12. If the Imediate window is not displaying, on the main menu, click Debug -> Windows -> Immediate.
    Just in case, right-click inside the Immediate window and click Clear All.
    Press F11 to step into code
  13. Continue pressing the key (3 times) to step into code until you get to the line double netPay = 0.00D;
  14. Click inside the Immediate window
  15. Type timeWorked = 42.50 and press Enter
  16. Type hourlySalary = 24.65 and press Enter
  17. Type totalDeductions = 286.50 and press Enter

    Immediate Window

  18. Press F11 three or four times to step into code until you get to the line that has Console.Clear();

    Debugging

  19. Click the next empty line in the Immediate window
  20. Type ?grossPay and press Enter
  21. Type ?netPay and press Enter

    Immediate Window

  22. Press the F11 key to step into code a few times until the DOS window gets focus

    Immediate Window

  23. Press Enter to return to Microsoft Visual C#
  24. Press F11 continuously until you get to the end of code and the DOS window closes

The Autos Window

As debugging progresses, sometimes you may want to know the values that the variables are holding on the current and the preceding lines. To give you this information, Microsoft Visual Studio provides the Autos window (Microsoft Visual Studio Express 2013 for Windows Desktop doesn't have the Autos window). Normally, when you start debugging, the Autos window comes up. If it is hidden while you are debugging, on the main menu, click Debug -> Windows -> Autos.

The Watch Window

Imagine you have a variable that is accessed in various parts of your code. One way you can test the behavior of that variable is to test its value as it circumstancially changes time after time. The Watch window is an object that allows you to monitor the values that a variable (or many variables) holds (or have) in various parts of a method.

To get the Watch window, start debugging your application (Debug -> Start Debugging or DEBUG -> Step Into). The Watch window would appear, usually next to the Locals window. The Watch window appears as a table with three columns. Their roles will be obvious to you once the window contains something.

To actually use the services of a Watch window, you must create one or more entries, which are referred to as watches. Before creating a watch, you must start stepping into your code, which is done by clicking DEBUG -> Step Into or by pressing F11.

To create a watch, remember that you must first Step Into your code, then click Debug -> QuickWatch... In the Expression combo box, type the name of the variable you want to watch. Here is an example:

Quick Watch

You can also type an expression such as a value added to a variable. Here is an example:

Quick Watch

If you want to submit the entry, click the Add Watch button. As an alternatice, in both Microsoft Visual C# versions, in the Watch window, double-click under the Name header, type either the name of the variable only or an expression that has the variable and an operation, then press Enter.

After creating the desired entries in the Watch window, continue debugging. You will see the values being automatically updated in the Value column of the Watch window.

If you don't need a certain entry in the Watch window, you can remove it, or you can delete all entries in the window. To remove an entry, right-click it and click Delete Watch. To remove all entries, right-click anywhere in the Watch window and click Clear All.

Practical LearningPractical Learning: Using the Watch Window

  1. To start debugging, on the main menu, click Debug -> Step Into
  2. If the Watch window is not displaying and if you are using Microsoft Visual Studio Express 2013 for Windows Desktop, on the main menu, click Debug -> Windows -> Watch -> Watch 1.
    Just in case, right-click inside the Watch window and click Clear All
  3. In the Watch window, double-click under Name, type timeWorked * 1.50 and press Enter (this will be used to evaluate overtime)
  4. Still in the Watch window, click under timeWorked * 1.50, type hourlySalary + 0.55 and press Enter
  5. Click under hourlySalary + 0.55, type timeWorked * hourlySalary and press the down arrow key
  6. Type (timeWorked * hourlySalary) - totalDeductions and press Enter:

    Watch

  7. Make sure you have access to the Immediate window.
    Click inside the Immediate window
  8. Type timeWorked = 7.50 and press Enter
  9. Observe the update in the Watch window

    Immediate Window

  10. In the Immediate window, click the empty line, type hourlySalary = 18.24 and press Enter
  11. Check the change in the Watch window

    Immediate Window

  12. In the Immediate window, click the empty line, type totalDeductions = 38.75 and press Enter
  13. See the result in the Watch window

    Immediate Window

  14. Click the last line (the first empty line) in the Immediate window
  15. Press the up arrow key to access the previously entered lines (the text lines you had typed will be displaying one after the other) until you get timeWorked
  16. Change it to timeWorked = 12.75 and press Enter

    Immediate Window

  17. Press the up arrow key until hourlySalary = 18.24 is selected
  18. Edit it to show hourlySalary = 26.16 and press Enter

    Immediate Window

  19. press F11 continuously until you get to the end of code and the DOS window closes

The IntelliTrace Window

While debugging, you are usually curious to know what line, section, or file is currently being examined:

If you are using Microsoft Visual Studio 2013 Ultimate, to get all of this information in one group, you can use a window named IntelliTrace. To get the IntelliTrace window (in Microsoft Visual Studio 2013 Ultimate only)


Previous Copyright © 2010-2025, FunctionX Friday 15 October 2021 Next