Whether you are looking for bugs, figuring out how the code flows, or experimenting your new feature, knowing how to use the debugger will save you a lot of time. I use the debugger basically every single time I run my application. It is also an excellent tool to use to learn how programming works. I just wish school taught this in my early beginner computer science classes. Instead, I was first introduced this after two years when I did my first internship.
These are some debugging techniques that I use the most. I highly recommend reading through objc.io’s debugging tips for a more inclusive reading.
Quick overview of the basics. You need to know this. Period.
- Step Over - execute the current line and stop before executing the next line.
- Step In - go inside the method at the current line. If it’s already run, finish executing the current line and stop before executing the next line.
- Step Out - finish executing the current block and return to the caller.
- Conditional Breakpoints - hit breakpoint only if it satisfies the condition.
- Automatically Continue Breakpoints - executes the expression in the breakpoint and then continue automatically.
po- print out details about the variable.
expression- execute the command.
e myVariable = @"new".
e UIView *$view = (UIView *)0x0000- create a variable while debugging.
All Objective-C Exceptions Breakpoint
Catch it before the crash. This exception breakpoint is a major life saver. When there is an exception thrown, the application will just crash and give you the error stack trace in the Xcode output. When this is enabled, you will hit a breakpoint right at the line before the application crashes. Then you can look at your variables and see what’s exactly going on that’s causing the crash.
To turn it on:
- Open up the Navigator
- Click on the ‘Breakpoint navigator’ tab
- Click on the ‘+’ sign at the bottom to make a new ‘Exception Breakpoint’
That’s all it is! But you can do more to get the most out:
- When the breakpoint hits, set it so that the exception will print out the
error message from running this command:
- Set exception on ‘Objective-C’ only. This will ignore any C++ exceptions if you don’t want to see them.
- Right click and move breakpoint to User to keep this breakpoint across all your projects!
Return from a Method
thread return NO,
thread return myVariable.
Great for stubbing methods. Quickly return from a method with the value that you want. Let’s say that you wanted to test this method that can return multiple values. Instead of changing the value, re-compiling and running it, you can just put a breakpoint here and return the value that you want!
Skip/Jump Executing Lines
thread jump --by 2,
thread jump --line 102
I use this when I want to skip something from executing. This doesn’t always work flawlessly though as LLDB doesn’t like skipping lines too much. Also, Xcode IDE will not know that you skipped line until you step over to the next line.
e @import UIKit
You hit a breakpoint and you want to print a view’s bounds by calling
po view.bounds. Instead of getting the view’s bounds, you may have gotten some
garbage output such as:
- error: unsupported expression with unknown type
- error: ‘bounds’ has unknown return type; cast the call to its declared return type
To fix it, you may have called it by calling it as a method and also specify the
return type by doing
po (CGRect)[self bounds]. Well, instead of typing all that
mess every time you want to print something, you can just do do
e @import UIKit
and you will be allowed to do
po view.bounds and all the other methods that comes
Update UI Changes
e (void)[CATransaction flush],
caflush (with chisel installed)
Great for testing UI changes on the fly without having to restart after each change.
Snapshot of a View
visualize myView (with chisel installed)
Wondering what the view looks like? Just run this command and it’ll open a snapshot image of your view on Preview.app.
I don’t use all of these all the time, but they are used more often than other techniques that you may have seen. I will update these as my preferences are changed or I learn something new. Let me know if I should be using something and it’s missing from this list.