Aqua 2024.2 Help

Examine suspended program

After the debugger session has started, the Debug tool window appears, and the program runs normally until one of the following happens:

After that, the program is suspended, allowing you to examine its current state, control its further execution, and test various scenarios at runtime.

Examine frames

The state of the program is represented by frames. When the program is suspended, the current frame stack is displayed in the Frames pane located on the Threads & Variables tab.

Frames pane

A frame corresponds to an active method or function call. It stores the local variables of the called method or function, its arguments, and the code context that enables expression evaluation.

Threads and frames

To better understand the concept of frames, let's look into what happens when a program is run. The execution of the program starts from the main method, which in turn calls other methods. Each of these methods may make additional method calls. The set of local variables and parameters for each method call is represented by a frame.

Each time a method is called, a new frame is added to the top of the stack. When the execution of a method is complete, the corresponding frame is removed from the stack (in the last in, first out fashion).

Examining frames helps you understand why particular parameters were passed to a method and what the state of the caller was at the time of calling.

Thread Statuses

Thread statuses are provided by Java to reflect what is currently happening to the thread:

Thread status

Description

MONITOR

The thread is waiting on a Java monitor.

NOT_STARTED

The thread has not yet been started.

RUNNING

The thread is active and running.

SLEEPING

The thread is sleeping because Thread.sleep() or JVM_Sleep() was called.

UNKNOWN

The thread status is unknown.

WAIT

The thread is waiting after Object.wait() or JVM_MonitorWait() was called.

ZOMBIE

The thread has completed execution.

Thread Icons

Icons near each thread indicate the status of the thread:

Icon

Description

Current thread

The current thread in suspended state.

Running thread

An active thread.

Thread at breakpoint

The thread that has hit the current breakpoint.

Suspended thread

A suspended thread. Threads are marked as suspended when they were paused by the debugger.

Frozen thread

A frozen thread. Threads are marked as frozen when they were manually paused.

By default, Aqua hides the frames that correspond to framework and library calls.

Show frames from libraries

  • To reveal the hidden frames, press the Show All Frames toggle button show Frames from Libraries button located in the top-right corner of the Frames pane.

    Frames pane with 88 collapsed framework calls and Show All Frames button in the top-right corner

Configure packages to hide

  1. Go to Settings Ctrl+Alt+S | Build, Execution, Deployment | Debugger | Stepping.

  2. Specify the packages that you want to hide under Do not step into classes.

  3. Make sure the Hide stack frames using stepping filters is enabled.

Copy stack to clipboard

  • To copy the call stack for the current thread, right-click anywhere on the Frames tab and select Copy Stack.

Export threads

If you need to get a report containing the state of each thread and its stack trace, use the Export Threads option. This is useful when you need to share the information about the threads in text format.

  1. Right-click anywhere on the Frames pane and select Export Threads from the menu.

    Export Threads menu item
  2. To save a report as a text file, specify the path to the file in the Export Threads dialog and click Save, or click Copy to copy it to the clipboard.

    Export Preview dialog

Examine/update variables

The Variables pane shows the list of the variables in the selected frame/thread. Examining the variables can help you understand why the program operates in a certain way.

Variables pane

The icon on the left of each variable indicates its type.

Variable types

Icon

Description

Static members

Static members of the enclosing class

A field

Fields of an object (both static and nonstatic)

A field containing a self-reference

Fields containing a self-referencing object (for example, cause in Throwable before the field gets initialized)

A final field

Final fields

A static field

Static fields

A thrown exception

A thrown exception (only displayed when an exception breakpoint was hit)

A method return value

A method return value (only displayed when the Show Method Return Values option is enabled)

A parameter

Method parameters

An enum constant

Enum constants

An array

Local arrays

A primitive

Local primitive types

A watch

Watches and auto-variables.

A primitive

Local reference variables

Pin fields

If an object has numerous fields, you can pin some of them, so that they are always displayed at the top of the list. This priority will apply to all instances of the corresponding class.

  • In the Variables pane, click the icon that indicates the variable type.

    Pinned field is marked with blue flag

When a field is pinned, a blue flag replaces the original icon. To unpin the field, click this flag.

Copy variables

When examining variables, you may need to copy a variable name or value to paste it somewhere else or compare it with another variable.

  • To copy the name of a variable, right-click the variable and select Copy Name.

  • To copy the value that a variable holds, right-click the variable and select Copy Value Ctrl+C. For types other than String, the toString representation is copied.

Compare variables with clipboard

To compare a variable value with some other value, use the Compare Value with Clipboard option. This is helpful, for example, when a variable holds a long string, and you need to compare it with another long string.

  1. Copy the content you want to compare, for example, from a text file.

  2. In the Variables pane, right-click a variable and select Compare Value with Clipboard.

  3. Examine the differences in the Diff Viewer that opens. For more information about Diff Viewer, refer to Comparing Files and Folders.

View variables in a dedicated dialog

Aqua allows you to inspect variables in a dedicated dialog. This is useful when you need to keep track of some variable (or the object whose reference it holds) and at the same time be able to navigate between frames and threads.

  • Right-click a variable or a watch and select Inspect.

    Inspect dialog

Set variable values

If you want to test how the program would behave with certain data or change its flow at runtime, you can achieve that by changing the variable values.

  1. Select a variable and press F2. Alternatively, select Set Value from the context menu.

  2. Enter the value for the variable and press Enter.

    Enter new value for the variable in the field right next to its name

You can navigate to declarations from the Variables pane.

  • To navigate to the code where the variable is declared, right-click the variable and select Jump to Source F4.

  • To navigate to the class declaration of the variable type, right-click the variable and select Jump to Type Source Shift+F4.

  • To navigate to a method body from a stack trace element, click Navigate near the stack trace element on the Variables pane.

    Clicking Navigate takes you to the code of the corresponding method

Examine incoming references

Aqua provides you with the information on the currently existing objects, which hold a reference to the objects on the Variables tab. The feature also detects indirect references, like those in anonymous classes using outside variables.

  • To view the list of referring objects, right-click a variable on the Variables tab and select Show Referring Objects.

The Referring Objects dialog opens, allowing you to inspect the references to the selected object and trace reference chains down to the GC roots.

Evaluate expressions

Aqua lets you evaluate expressions during a debugging session to obtain additional details about the program state or test various execution scenarios at runtime.

This feature only works if the program was suspended after hitting a breakpoint (not paused).

If there are breakpoints inside methods called within the expression, they will be ignored.

Evaluate a simple expression in the editor

To quickly evaluate an expression, point at it in the editor. Note that method calls cannot be evaluated this way.

  1. Point at the expression you want to evaluate. The result of the expression appears in a tooltip.

    Value tooltip
  2. To view child elements of the resulting object, click .

    Resulting objects are represented by trees. This helps you view their internal state

If you find value tooltips distracting, you can increase the delay or disable them altogether. To do this, in the Settings dialog (Ctrl+Alt+S) , go to Build, Execution, Deployment | Debugger | Data Views and set the Show value tooltip and Value tooltip delay options according to your preference.

Evaluate a complex expression in the editor

If you want to evaluate an expression in the code that involves a method call, or you want to be specific about which portion of expression to evaluate, use the Quick Evaluate Expression option.

  1. Place the caret at the expression (to evaluate the closest matching expression) or select a portion of it (if you want to be specific about which part of a complex expression to evaluate).

  2. Go to Run | Debugging Actions | Quick Evaluate Expression Ctrl+Alt+F8. Alternatively, hold Alt and click the selection.

    Value tooltip appears

You can configure Quick Evaluate Expression to work for a piece of code on just selecting it (without using the menu/shortcut). Use this option carefully, as you can accidentally call methods when it is enabled.

Evaluate expressions on selecting code

  • Go to Settings | Build, Execution, Deployment | Debugger | Data Views and set the Show value tooltip on code selection option.

Evaluate arbitrary expressions

Evaluating arbitrary expressions is the most flexible evaluating option. It lets you evaluate any custom code as long as it is in the context of the current frame. Using it, you can evaluate declarations, method calls, anonymous classes, lambdas, loops, and so on.

  1. To evaluate an arbitrary expression, enter it in the Evaluate expression field in the Variables pane and press Enter

    Expression in the Variables pane
  2. The result is displayed right below. You can also add the expression to watches by clicking in the right-hand part of the expression field.

    Result of an expression in the Variables pane

If you want to evaluate long code blocks, you may want to use a dedicated dialog for that:

  1. If you want to start with some expression or a variable, which is currently in front of you (for example, in the editor or on the Variables pane), select it.

  2. Go to Run | Debugging Actions | Evaluate Expression Alt+F8 or select Evaluate Expression from the context menu. The shortcut may not work on Ubuntu (for correct operation, adjust the shortcut configuration).

  3. In the Evaluate dialog, modify the selected expression or enter a new one in the Expression field. Click Expand Shift+Enter to modify a multiline code fragment.

  4. Click Evaluate (Ctrl+Enter for multiline mode). The expression result appears in the Result field.

    The result of the expression is taken from the return statement. When there is no return statement, the result is taken from the last line of code (it does not even have to be an expression: a single literal works too). When there is no valid line to take the value from, the result is undefined. If the specified expression cannot be evaluated, the Result field indicates the reason.

The Evaluate dialog is non-modal, so you can switch the focus back to the editor to copy other variables and expressions. You can also open multiple Evaluate dialogs.

View values inline

Aqua shows the values of the variables right next to their usage.

Variable values are displayed at the lines where they are used

Once the variable value has changed, the inline view is updated with the new value and changes its color.

Inline values of the variables change with each step

If a line contains a reference to an object, you can examine its fields right in the editor. From this popup, you can also change the variable values and add inline watches.

Inline debugger hints

The inline view is enabled by default. To turn it off, in the Settings dialog (Ctrl+Alt+S) , go to Build, Execution, Deployment | Debugger | Data Views and disable the Show values inline option.

Add an Inline Watch

If you want the result of some expression to appear on a particular line, you can set up an inline watch for that. Inline watches are persistent and remain active after session restart.

  1. Click the inline hint referring to the object whose field you want to track.

  2. In the popup, select the field and click Add as Inline Watch.

    Inline debugger hints
  3. Fine-tune the watch if needed. You can use any valid expression as a watch.

To remove an inline watch, hover over the watch and click the cross near it.

Watches

If you want to keep track of some variable or the result of a more complex expression, set up a watch for this variable or expression. This is useful when you need to evaluate something that is not regularly displayed on the list of variables.

This feature only works if the program was suspended after hitting a breakpoint (not paused).

Watches are evaluated in the context of the selected frame. Watches cannot be evaluated when they are out of context or when they fail to compile. If this is the case, the watch is marked with the error icon Error icon.

By default, watches are shown together with variables in the Variables pane. To hide/reveal the Watches pane, use the Separate watches option in the Layout Settingsthe Restore Layout button menu.

Add a watch

  • Enter the expression in the top part of the Variables pane, then click Add to Watches.

    Watch expression

After you have added a variable/expression to Watches, it stays there and is evaluated for each step, providing you with the result in the current context.

Edit a watch

  • Right-click the desired watch and select Edit.

Delete a watch

  • To remove a single watch, right-click it and select Remove Watch. Alternatively, select the watch and press Delete on the Variables/Watches pane.

  • To remove all watches, right-click anywhere on the Variables/Watches pane and select Remove All Watches.

Watches allow for the same actions as variables do. For example, you can view them in a dedicated dialog or use them to navigate to the source code.

Watches are a part of your project. This means you can stop and rerun the debugging session without risk of losing them.

Execution point

Return to the current execution point

Examining the program state involves navigating in code, and you often need to return to the place where your program is suspended.

Do one of the following:

  • In the main menu, go to Run | Debugging Actions | Show Execution Point.

  • Press Alt+F10.

  • Click The More button on the stepping toolbar of the Debug tool window, then select Show Execution Point The Show Execution Point button from the list.

The current execution point is indicated with a blue line. The code at this line has not been executed yet.

Blue line indicating the current execution point
Last modified: 24 October 2024