Debugging Tactics Flashcards
Basic debugging tactics
- commenting out the code
- validating the code flow
- printing values
Validating code flow
Used when the program is calling to a function too many or few times.
Using validating code flow
By placing statements at the top of the function to print the functions name
Use std::cerr and do not indent the debug statements
Printing values
Used when the program is calculating or passing wrong value
Using printing values
By outputting the value of variables and expressions by using std::cerr
Why not to use debug statements
- clutter the code
- clutter the output of the program
- must be removed, it is not reusable
- requires modification which can introduce new bugs
Conditionalizing the debugging code
To make the debugging statements conditional using preprocessor directives
Using conditionalizing the debugging code
With #define ENABLE_DEBUG by commenting/uncommenting it
Conditionalizing the debugging code in multi-file program
Use #define ENABLE_DEBUG in a header file in all code files
Log file
File stored in disk, that records events that occur in the software
Logging
Process of writing information to a log file
Advantages of log file
- aviding clutter, because it is stored on a separate file
* can be sent to others for diagnosis
Program state
Tracked value of variables , called function and current point of execution within the program
Debugger
Computer program that allows the programer to control how a program executes and examin the program state while the program is running
Advantige of debugger
Ability to precisily control execution of the program and ability to view/modify program’s state
Stepping
Set of related debugger features that let us execute (step through) our code statement by statement
Step into
That executes the next statement in the normal execution path of the program, and pauses so you can inspect the code
Stepping into without running
- the program will recompile
- will begin to run
- will open diagnostic windows
Step over
Executes an entire function without stepping, use it when you know a function works
Step out
Executes remaining code in the function currently being executed, use it when you accidentally stepped into a function that you don’t want to debug
Step back
Rewind the last step, so you can return the program to a prior state
Run to cursor
Executes the prigram until the execution reaches the statement selected by your cursor, then it returns the control to you
Continue
Continues runnung the program as per normal, either until the program terminates or until something triggers control to return back
Start
Running the program fromthe begining, it can be invoked when not already in a debug session
Breakpoint
Tells the debugger to stop executing the program at the breakpoint when running in debug mode
Advantages of breakpoints
- return control everytime they are encountered
* you can set a breakpoint and it will persist until you remove it
Set next statement
Changes the point of execution to some other statement (jumping), you can jump forwards and backwards
Watching variables
Process of inspecting the value of a variable while the program is executing in debug mode.
Ways to watch variables
- hovering the mouse over variables
* quickWatch in the right-click menu
Watch window
Window where you can add variables you like to continually inspect, will be updated as you step trough
Ways to add variables to watch window
- trough watch window by typing the name of the variable in the left column
- trough code window by right-click on variable and add watch
Watching out of scope variables
Will stay on the window, but will be marked as not available or show the last kmown value but it will be grayed out
Other watch window use
Evaluating simple expressions
Local watches
Quick way to watch value of local variables in scope
Call stack
List of all the active functions that have been called to get to the current point of execution
Included in the call stack
Entry for each function called and which line of code will be returned to when the function returns
Adding functions to the call stack
The function is added to the top of the stack, when it’s returned the function is removed from the top of the stack and the control is returned to the function below it
Call stack window
Debugger window that shows the current call stack
Use for call stacks
In conjunction with breakpoints, when your breakpoint is hit and you want to know what functions were called to get to that point of code
Things that can help avoid making errors
- follow the best practices
- don’t program when tired
- understand where the common pitfalls are in a language
- keep your programs simple
- don’t let your functions to get too long
- preder using the standard library to writing your own code
- comment liberary
Refacturing
Process of making strucrltural changes to your code without changing it’s behaviour
Making behavioural and structural changes
At the same time tends to lead to more errors
Defensive programming
Practice where the programmer tries to anticipate all the ways the software could be mis-used by end-users or developers using the code
Programming a little bit a time
Fast way to find errors especialy if you test the code and make sure it works
Testing functions
Primitive form of unit testing, by which a small unit of source code are tested to determine whether they are correct
Use of testing functions
Ex. testadd(), tests the function by calling it with different values
Constrains
Involve the addition of some extra code to check that some set of assumptions or expectations are not violated
Static analysis tools (linters)
Programs that analyze your code to identify specific semantic issues