Errors in a Program

Introduction

Apparently no matter how careful and meticulous you are, there will be errors in your program. When, not if, they occur, you should be able to address the issue. In most cases, both C# and Microsoft Visual Studio can assist you.

The errors your program will encounter can be classified in three categories: runtime, syntax, and logic errors. We will study runtime errors in the next two lessons about exception handling.

Practical LearningPractical Learning: Introducing Errors

  1. Start Microsoft Visual Studio
  2. Create a new Console App named WattsALoan
  3. In the Solution Explorer, right-click Program.cs -> Rename
  4. Type LoanEvaluation (to get LoanEvaluation.cs) and press Enter

Syntax Errors

A syntax error is due to a misuse of the C# language in your code. For example, in Lesson 2, we saw that, on one hand C# is case-sensitive and, on the other C# has a set of keywords that you should (must) not use to name your variables. The first rule can be difficult to observe if you come from a case-insensitive language like Pascal or Visual Basic. It can also happen through distraction. The second rule is easy to observe if you have used another C-based language such as C++ or Java. Both rules are easy to violate if you write your code using a normal text editor like Notepad.

Fortunately, the built-in Code Editor of Microsoft Visual Studio makes it extremely easy to be aware of syntax errors as soon as they occur:

As you can see, if you create your application in Microsoft Visual Studio, the Code Editor is fully equipped with tools to assist you to detect and correct syntax errors. If you still violate a syntax rule, when you build your project, the compiler would detect the error and point out the line, the section, and the file name where the error occurred. Here is an example:

Operator Misuse

Practical LearningPractical Learning: Introducing Syntax Errors

Logic Errors

A logic error occurs when the program (the code) is written fine but the result it produces is not reliable. With a logic error, the Code Editor does not see anything wrong in the document and therefore cannot point out a problem. One of the worse types of logic errors is one that makes a computer crash sometimes, regularly, or unpredictably, while there is nothing obviously wrong in the code.

Logic errors are, or can be, difficult to spot because you will have to know for sure that the result is wrong and why (and sometimes worse, you will have to agree or accept that it is your program that is causing a problem in the computer: a bitter pill to swallow; imagine a user reports that her computer crashes every time she starts the application you created). Because you or the user of your program would know with certainty that the result is questionable, you would have to use some means of correcting it. One of the techniques you can use is referred to as debugging.

Debugging Fundamentals

Introduction

A logic error is called a bug. Debugging is the process of examining code to look for bugs or to identify problems. Debugging is the ability to monitor the behavior of a variable, a class, or its members throughout a program. Microsoft Visual Studio provides many features to perform debugging operations.

The debugger is the program you use to debug your code. The code or application that you are debugging is called the debuggee.

Probably the most fundamental way of examining code is to read every word and every line, with your eyes, using your experience as a programmer. This can work for short code written in one file and in one class. If the code to examine covers many pages or many files, it could be aweful and tiresome to examine code with your eyes line by line. Fortunately, to assist you with this operation, Microsoft Visual Studio provides various tools and windows you use, one window, or a combination of objects. One of the tools you can use is the Standard toolbar that is equipped with various debugging buttons:

Standard Toolbar

Starting and Continuing With Debugging

There are different ways you can launch the debugging process:

In later sections, we will see other ways of starting or proceeding with debugging. In some cases, we will see how you can suspend debugging. When this has happened, to resume debugging:

We will see other ways of continuing with debugging.

Stopping the Debugging

As we will see in later sections, there are various debugging approaches available in Microsoft Visual Studio. Sometimes you will want to suspend or stop debugging.

To end debugging at any time:

The Locals Window

One of the primary pieces of information you want to get is the value that a variable is holding. A window named Locals is used to show that value. Normally, when you start debugging, the Locals window shows automatically. During debugging, if the Locals window is hidden, to display it:

As its name indicates, the Locals window shows the values of local variables as they are changed. If there is more than one variable, the Locals window displays their names and gives a row to each variable. The Locals window organizes its information in a table or grid:

Locals

The Name column shows the name of each variable declared in the function or the section that is being debugged. Imagine that the variable is a structure, a record, or a class. Here are examples:

using static System.Console;

Square sqr = new();
Rectangle rect = new(496.84, 825.97);

sqr.Side = 927.93;

WriteLine("Geometry - Square");
WriteLine("----------------------------");
WriteLine($"Side:      {sqr.Side}");
WriteLine($"Perimeter: {sqr.Perimeter}");
WriteLine($"Area:      {sqr.Area}");
WriteLine("============================");
WriteLine("Geometry - Rectangle");
WriteLine("----------------------------");
WriteLine($"Width:     {rect.Width}");
WriteLine($"Height:    {rect.Height}");
WriteLine($"Perimeter: {rect.CalculatePerimeter()}");
WriteLine($"Area:      {rect.CalculateArea()}");
WriteLine("============================");

// A Structure
internal record struct Square
{
    public double Side { get; set; }

    public double Perimeter
    {
        get
        {
            return Side * 4.00;
        }
    }

    public double Area
    {
        get
        {
            return Side * Side;
        }
    }
}

public class Rectangle
{
    public Rectangle(double width, double height)
    {
        (Width, Height) = (width, height);
    }

    public double Width  { get; init; }
    public double Height { init; get; }

    public double CalculatePerimeter()
    {
        return (Width * Height) * 2.00;
    }

    public double CalculateArea()
    {
        return Width * Width;
    }
}

In this case, when debugging, in the Locals window, an arrow icon would appear to its left, indicating that it has properties:

Debugging

Debugging

Debugging

In this case, to show the variables, that is, to expand the node, click the arrow icon. This would show the fields under the variable name and the name of the class between curly brackets under the Value column:

Debugging

The Value columns shows the value of each variable. When debugging starts, each variable shows its default or initial value. As debugging progresses, when a variable acquires a new value, the Locals window updates it in the Value column. In some cases, instead of the debugger changing the value, you can manually select and change it in the Locals window and press Enter.

The Type column shows the data type of the variable. If the variable is a class, the name of the class shows in the Type column.

Debugging Statements

Executing One Statement at a Time

Just as done when reading code with your eyes, the most basic way to monitor code is to execute one line at a time and see the results displayed before your eyes. To support this operation, the debugger provides what is referred to as stepping into.

To execute code one line at a time, while the file that contains it is displaying:

When code is being stepped into, the margin corresponding to the line that is being examined displays a right-pointing yellow arrow:

Syntax Error

This lets you know what line is currently considered.

Practical LearningPractical Learning: Examining Local Variables

  1. While the Code Editor is displaying the code you typed, on the main menu, click Debug -> Step Into.
    If the Locals window is not displaying, on the main menu, click Debug -> Window -> Locals
  2. To perform the instructions in this section, move the windows to make sure you can see the Code Editor, the Locals window, and the DOS window. Here is an example:

    Debugging

  3. Notice the yellow arrow button on the left of the declaration of the principal variable.
    Notice that the Locals window displays one entry: the principal variable.
    Notice that the DOS window shows a blinking caret.
    To end debugging, on the main menu, click Debug -> Stop Debugging
  4. Change the document as follows:
    using static System.Console;
    
    double principal;
    double interestRate;
    double period;
    double interestAmount;
    double futureValue;
            
    Title = "Watts A Loan?";
    
    Console.WriteLine("This application allows you to evaluate a loan");
    WriteLine("To proceed, enter the following values");
    
    Write("Enter the principal: ");
    principal = double.Parse(ReadLine()!);
    Write("Enter the interest rate: ");
    interestRate = double.Parse(ReadLine()!) / 100;
    Write("Enter the number of months: ");
    period = double.Parse(ReadLine()!) / 12;
    
    interestAmount = principal * interestRate * period;
    futureValue    = principal + interestAmount;
    
    Clear();
    
    WriteLine("============================");
    WriteLine("Loan Summary");
    WriteLine("=--------------------------=");
    WriteLine("Principal:       {0:F}", principal);
            Console.WriteLine("Interest Rate:   {0:P}", interestRate);
    WriteLine("Period For:      {0} months", period * 12);
    WriteLine("Interest Amount: {0:F}", interestAmount);
    WriteLine("Future Value:    {0:F}", futureValue);
    WriteLine("============================");
  5. To restart debugging, on the main menu, click Debug -> Step Into.
    Notice that the Locals window displays a list of the local variables and their starting values:

    Locals - Variables

  6. To continue debugging, on the Standard toolbar, click the Step Into button Step Into
  7. To continue debugging, press F11.
    At some point, text will display in the DOS window
  8. Keep pressing F11 until the focus moves to the DOS window. You will know when the lines in the Locals window are disabled and the caret is blinking in the DOS window while requesting a value

    Debugging

  9. When asked to enter a value for the principal, type 6500 and press Enter
  10. Notice that the focus moves back to the Code Editor.
    Notice that the value of the principal has changed in the Locals grid:

    Debugging

    Press F11
  11. Press F11 again
  12. When the caret sttars blinking in the DOS window as you are asked to provide the interest rate, type 12.65 and press Enter.
    The focus moves back to the Code Editor.
    Notice that the value of the interestRate variable in the Locals window has changed

    Debugging

  13. press F11
  14. press F11 again
  15. The focus moves to the DOS window. When the number of months is requested, type 36 and press Enter.
    The focus moves back to the Code Editor.
    Notice that the value of the period variable in the Locals window has been changed

    Debugging

  16. press F11

    Debugging

  17. press F11

    Debugging

  18. press F11

    Debugging

  19. Continue pressing F11 until the DOS window displays a summary of the calculations:
    ============================
    Loan Summary
    =--------------------------=
    Principal:       6500.00
    Interest Rate:   12.65 %
    Period For:      36 months
    Interest Amount: 2466.75
    Future Value:    8966.75
    ============================
  20. Press Enter to get back to your programming environment
  21. press F11 to end

Executing One Method at a Time

The Step Into feature is a good tool to monitor the behavior of variables inside a method. This also allows you to know if a method is behaving as expected. Once you have established that a method is alright, you may want to skip it. Instead of executing one line at a time, the debugger allows you to execute a whole method at a time or to execute the lines in some methods shile skipping the others. To support this, you use a feature named Step Over.

To step over a method, while debugging:

As its name suggests, the Step Over feature allows you to skip a method if you know it doesn't have any problem. When debugging, you choose what methods to step into and which ones to step over.

Practical LearningPractical Learning: Stepping Over

  1. Change the file as follows:
    using static System.Console;
    
    double GetPrincipal()
    {
        double value = 0.00D;
    
        Write("Enter the principal: ");
        value = double.Parse(ReadLine()!);
    
        return value;
    }
    
    double GetInterestRate()
    {
        double value = 0.00D;
    
        Write("Enter the interest rate: ");
        value = double.Parse(ReadLine()!);
    
        return value;
    }
    
    double GetPeriods()
    {
        double value = 0;
    
        Console.Write("Enter the number of months: ");
        value = double.Parse(ReadLine()!);
    
        return value;
    }
    
    void ShowLoanSummary(double presentValue,
                         double annualRate,
                         double periods,
                         double interestCollected,
                         double totalCollected)
    {
        WriteLine("============================");
        WriteLine("Loan Summary");
        WriteLine("=--------------------------=");
        WriteLine("Principal:       {0:F}", presentValue);
        WriteLine("Interest Rate:   {0:P}", annualRate);
        WriteLine("Period For:      {0} months", periods * 12);
        WriteLine("Interest Amount: {0:F}", interestCollected);
        WriteLine("Future Value:    {0:F}", totalCollected);
        WriteLine("============================");
    }
    
    double principal;
    double interestRate;
    double period;
    double interestAmount;
    double futureValue;
    
    Title = "Watts A Loan?";
    
    WriteLine("This application allows you to evaluate a loan");
    WriteLine("To proceed, enter the following values");
    
    principal = GetPrincipal();
    interestRate = GetInterestRate() / 100;
    period = GetPeriods() / 12;
    
    interestAmount = principal * interestRate * period;
    futureValue = principal + interestAmount;
    
    Clear();
    
    ShowLoanSummary(principal, interestRate, period, interestAmount, futureValue);
  2. To debug, on the main menu, click Debug -> Step Into.
    Notice that the yellow button is positioned in the first line of the code
  3. To continue, on the Debug toolbar, click the Step Into button Step Into three times until the debugger gets to the principal = GetPrincipal(); line:

    Debugging

  4. Press F11 to continue:

    Debugging

    Notice that the debugger is positioned on the first line of the GetPrincipal() method
  5. Press F11

    Debugging

    Notice that the debugger gets in the body of the GetPrincipal() method
  6. Keep pressing F11 while inside the GetPrincipal() method, until the DOS window receives focus
  7. When asked to provide a Principal, type 21650 and press Enter:

    Debugging

  8. When focus is back to your programming environment, to end debugging, on the main menu, click Debug -> Stop Debugging
  9. In the code, move the functions to the bottom of the document:
    using static System.Console;
    
    double principal;
    double interestRate;
    double period;
    double interestAmount;
    double futureValue;
    
    Title = "Watts A Loan?";
    
    WriteLine("This application allows you to evaluate a loan");
    WriteLine("To proceed, enter the following values");
    
    principal = GetPrincipal();
    interestRate = GetInterestRate() / 100;
    period = GetPeriods() / 12;
    
    interestAmount = principal * interestRate * period;
    futureValue = principal + interestAmount;
    
    Clear();
    
    ShowLoanSummary(principal, interestRate, period, interestAmount, futureValue);
    
    double GetPrincipal()
    {
        double value = 0.00D;
    
        Write("Enter the principal: ");
        value = double.Parse(ReadLine()!);
    
        return value;
    }
    
    double GetInterestRate()
    {
        double value = 0.00D;
    
        Write("Enter the interest rate: ");
        value = double.Parse(ReadLine()!);
    
        return value;
    }
    
    double GetPeriods()
    {
        double value = 0;
    
        Console.Write("Enter the number of months: ");
        value = double.Parse(ReadLine()!);
    
        return value;
    }
    
    void ShowLoanSummary(double presentValue,
                         double annualRate,
                         double periods,
                         double interestCollected,
                         double totalCollected)
    {
        WriteLine("============================");
        WriteLine("Loan Summary");
        WriteLine("=--------------------------=");
        WriteLine("Principal:       {0:F}", presentValue);
        WriteLine("Interest Rate:   {0:P}", annualRate);
        WriteLine("Period For:      {0} months", periods * 12);
        WriteLine("Interest Amount: {0:F}", interestCollected);
        WriteLine("Future Value:    {0:F}", totalCollected);
        WriteLine("============================");
    }
  10. To start debugging again, on the main menu, click Debug -> Step Into and notice that the yellow button is positioned in the first processed line of code
  11. To continue, on the Debug toolbar, click the Step Into button Step Into button four times:

    Debugging

  12. This time, to skip the GetPrincipal() function, on the main menu, click Debug -> Step Over
  13. Notice that, instead of the debugging moving to the function, the DOS window received focus.
    When asked to provide the principal, type 8725 and press Enter
    Notice that the focus moves back to the Code Editor at the end of the GetPrincipal() function
  14. To continue, on the Debug toolbar, click the Step Into button Step Into button three times until the line that calls the GetInterestRate() is accessed:

    Debugging

  15. To execute the function, on the Debug toolbar, click the Step Over button Step Over
  16. As the focus has moved to the DOS window, when the interest rate is requested, type 14.05 and press Enter.
    The focus moves back to the Code Editor where the GetPeriod() function is called:

    Debugging

    Notice the value of the interestRate in the Locals window
  17. To execute the GetPeriod() function, press F10 (Step Over)
  18. As the focus has moved to the DOS window, when asked to provide the number of months, type 48 and press Enter.
    The focus moves back to the Code Editor:

    Debugging

    In the Locals window, notice the period value that corresponds to 4 years
  19. To continue with Step Into, press F11 continually. Observe the Code Editor and the DOS window:

    Debugging

    Debugging

  20. Continue stepping into code until the DOS window displays Press any key to close this window . . .
  21. To end debugging, on the Debug toolbar, click the Stop button Stop button

Running to a Point

Introduction

When executing a program, you can specify a section or line where you want the execution to pause, for any reason you judge necessary. This approach is useful if you have checked code up to a certain point and it looked alright. If you are not sure about code starting at a certain point, this can be your starting point.

To execute code up to a certain point

Practical LearningPractical Learning: Running to a Point

  1. In the code, right-click the salesTotal = totalOrder + taxAmount; line and click Run To Cursor.
    Make sure you can see the Code Editor, the Locals window, and the DOS window
  2. When requested, enter the following values. Press Enter after entering each value. While you are entering them, check the moving button in the Code Editor and observe the values in the Locals window:
    Customer Name: Steve Longley
    Customer Phone: 301-208-2333
    Order Date: 10/08/2014
    Order Time: 08:14 AM
    Number of Shirts: 5
    Number of Pants: 2
    Number of Other Items: 8

    The focus moves to Microsoft Visual Studio and the Locals window:
     
    Locals
  3. Press F5 to continue
  4. For the amount tended, type 60 and press Enter
    ====================================
    -/- Georgetown Cleaning Services -/-
    ====================================
    Customer:    Steve Longley
    Home Phone:  301-208-2333
    Order Date:  Wednesday, October 08, 2014
    Order Time:  8:14 AM
    ------------------------------------
    Item Type  Qty Unit/Price Sub-Total
    ------------------------------------
    Shirts      5      0.95      4.75
    Pants       2      2.95      5.90
    Others      8      4.55     36.40
    ------------------------------------
    Total Order:   47.05
    Tax Rate:      5.75 %
    Tax Amount:    2.71
    Net Price:     49.76
    ------------------------------------
    Amount Tended: 60.00
    Difference:    10.24
    ====================================

Breakpoints

A breakpoint on a line is the code where you want the exection to suspend. You must explicitly specify that line by creating a breakpoint. You can as well create as many breakpoints as you want. You can also remove a breakpoint you don't need anymore.

To create a breakpoint, first identify the line of code where you want to add it. Then:

A breakpoint is represented by a red circular button Breakpoint. After creating a breakpoint, when code executes and reaches that line, it would pause and let you know by drawing a right-pointing yellow button Breakpoint.

After using a breakpoint, you can remove it. To delete a breakpoint:

Remember that you can create more than one breakpoint. If you have more than one breakpoint in your code, execution would pause at each one of them. At any time, you can remove one or all breakpoints. To delete all breakpoints, on the main menu, click Debug -> Delete all Breakpoints.

Practical LearningPractical Learning: Using Breakpoints

  1. In the Code Editor, click the margin on the left side of subTotalOtherItems = numberOfOtherItems * PriceOtherItem;
     
    Inserting a Breakpoint
  2. On the main menu, click Debug -> Start Debugging.
    Notice that the DOS window displays
  3. When asked, enter the values as follows and press Enter after each
     
    Customer Name: Hermine Simms
    Customer Phone: 410-573-2031
    Order Date: 10/08/2014
    Order Time: 09:22 AM
    Number of Shirts: 3
    Number of Pants: 3
    Number of Other Items: 12

    The focus moves back to Microsoft Visual Studio
     
    Locals
  4. Press F5 to continue
  5. For the amount tended, type 100 and press Enter twice
  6. In the code, click the Console.Write("Number of Shirts: "); line
  7. On the main menu, click Debug -> Toggle Breakpoint
  8. In the code, right-click the salesTotal = totalOrder + taxAmount; line, position the mouse on Breakpoint, and click Insert Breakpoint
     
    Inserting Breakpoints
  9. To start debugging, press F5
  10. When asked, enter the values as follows and press Enter after each
     
    Customer Name: Ginette Rhoads
    Customer Phone: 301-217-9494
    Order Date: 11/06/2014
    Order Time: 02:07 PM

    The focus moves back to Microsoft Visual Studio
  11. Press F5 to continue
  12. When asked, enter the values as follows and press Enter after each
     
    Number of Shirts: 0
    Number of Pants: 2
    Number of Other Items: 0
    Order Time: 14:07

    The focus moves back to Microsoft Visual Studio
  13. Press F5 to continue
  14. For the amount tended, type 20 and press Enter

Stepping to Breakpoints

You can combine the Step Into and/or the Step Over feature with breakpoints. That is, you can examine each code line after line until you get to a specific line. This allows you to monitor the values of variables and see their respective values up to a critical section. To do this, first create one or more breakpoints, then proceed with steps of your choice.

Practical LearningPractical Learning: Stepping to Breakpoints

  1. Make sure the previous two breakpoints are still selected.
    To start debugging, on the main menu, click Debug -> Step Into.
    Make sure you can see the Code Editor, the Locals window, and the DOS window
  2. press F11 continually until the DOS window receives focus
  3. For the customer name, type James Sandt and press Enter
  4. press F11 twice to continue
  5. For the customer phone, type 301-870-7454 and press Enter
    The focus moves back to Microsoft Visual Studio:

    Locals

  6. Press F11 to continue
  7. For the order date, type 10/10/14 and press Enter
  8. Press F11 to continue
  9. For the order time, type 06:02 PM and press Enter
  10. For the number of shirts, enter 8 and press Enter
  11. Press F11 to continue
  12. For the number of pants, enter 2 and press Enter
  13. Press F11 to continue
  14. For the number of other items, enter 12 and press Enter
  15. Press F11 continually until the DOS window receives focus
  16. For the amount tended, type 80 and press Enter
  17. Continue pressing F11 to the end
    ====================================
    -/- Georgetown Cleaning Services -/-
    ====================================
    Customer:    James Sandt
    Home Phone:  301-870-7454
    Order Date:  Friday, October 10, 2014
    Order Time:  6:02 PM
    ------------------------------------
    Item Type  Qty Unit/Price Sub-Total
    ------------------------------------
    Shirts      2      0.95      1.90
    Pants       8      2.95     23.60
    Others     12      4.55     54.60
    ------------------------------------
    Total Order:   80.10
    Tax Rate:      5.75 %
    Tax Amount:    4.61
    Net Price:     84.71
    ------------------------------------
    Amount Tended: 80.00
    Difference:    -4.71
    ====================================
  18. On the main menu, click File -> Close Solution
  19. When asked whether you want to save, click No

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