Structural search and replace examples
As you know the main difference between regular search and the structural search is that in the structural search we are looking for a structural template in a programming language.
The beauty of a structural search is that you can create a pattern based on the existing template and save yourself time when searching and replacing code.
The extensive list of existing templates covers a lot of use-cases from simple patterns to more complex ones.
Each item in a pattern consists of variables that are limited by $ sign on both sides.
Let's start with a simple task and search for an open lock objects using the synchronized
keyword. The synchronized
keyword can have two cases:
as a block statement
class ClassA { public void someMethod() { synchronized(this) { // ... } } }
as a method modifier
class ClassA { public synchronized void someMethod() { // ... } }
Let's look for something like that
synchronized ($parameter$){
$statement$;
}
We can add the Count modifier for the $parameter$ variable form 0 to infinity. So, the pattern can search for all the synchronizable methods with an arbitrary number of parameters, but with only one line of code in the body. We also set the same zero-to-infinity limitation for the $statement$ variable in order to correct this.
$Statement$;
Increasing the number of occurrences count to a certain number, you can find sequences of statements that contain up to the specified number of elements.
$Instance$.$MethodCall$($Parameter$)
This template matches method call expressions. If the number of occurrences is zero, it means that a method call can be omitted.
@Deprecated
$Instance$.$MethodCall$($Parameter$)
You can use this template to find deprecated methods and use it as prototype for creating other annotated method templates. Specifically for searching method calls to deprecated methods you can select the already existing template method calls to deprecated methods.
For more information about creating a search template, refer to Structural search and replace.
if ($Expr$) {
$ThenStatements$;
}
else {
$ElseStatements$;
}
Consider one wants to find comments or literal containing 'foo'. Search template would be like $SomethingWeWantToFind$
or "$SomethingWeWantToFind$"
. In case one wants to find comments/string containing some particular words (say, foo as a word), this should be specified as a text constraint.
If one wants to replace a statement with a try/catch/finally
construct, the following pair of search and replace templates can be suggested. The search template is:
$Statements$;
with a certain maximum number of occurrences specified as a constraint.
The replacement template is:
try {
$Statements$;
}
catch(Exception ex) {
}
Consider the following search templates:
class $Clazz$ extends $AnotherClass$ {}
or
class $Clazz$ implements $SomeInterface$ {}
As the text constraint for the variables $AnotherClass$ or $SomeInterface$, specify the name of the base class or implemented interface.
To look for the different implementations of the same interface method, use the following search template:
class $a$ {
public void $show$();
}
Specify text constraint for the $show$
variable, and enable the option This variable is the target of the search.
IntelliJ IDEA suggests pre-defined templates for the package local and instance fields of a class. These templates make use of the @Modifier annotation, which helps describe search target, when there is no way to express it using the natural language means.
However, if you need to search for package local or instance methods, you will have to create the corresponding search templates yourself, applying the @Modifier annotation.
To specify criteria for finding all methods with the visibility modifiers package local and instance, use the following search template:
class
$Class$ {
@Modifier("packageLocal") @Modifier("Instance" ) $ReturnType$ $MethodName$($ParameterType$ $Parameter$);
}
}
The existing example uses the following template:
LOG.debug($params$);
Placing if('_a) { '_st*; }
where _a
and _st
are variables and *
denotes zero or more occurrences in Contained in Constraints field and selecting Invert condition checkbox of Complete Match variable will result a search of logging statements that are not contained in the if
statement.
The following examples show how you can use structural search in HTML and XML code.
The simplest template to search for a tag is <$tag$/>
.
By placing constraints on the variable
$tag$
, you can specify tags that you want to find. For example, if you specifyli
, you will get allli
tags.Consider the following template for searching in XML and HTML:
<$tag$ $attribute$=$value$ />
. For example, if you specify the text filterid
for the$attribute$
variable and the\d+
regular expression as the text filter for the$value$
variable, you can find all tags that have numeric values in theid
attribute.
Create an HTML file and paste the following code:
HTML code
{...}In the main menu, go to Edit | Find | Search Structurally.
From the Language list, select HTML.
Paste the following string to the Search template field:
<$tag$ $attribute$="$value$">
Click the
$tag$
variable.In the filter panel, click Add modifier, select Text and type
li
in the value field.If the filter panel is not visible, click Toggle Modifier Panel.
Click the
$attribute$
variable.In the filter panel, click Add modifier, select Text and type
id
in the value field.Click the
$value$
variable.In the filter panel, click Add modifier, select Text and type
\d+
in the value field.The
\d+
regular expression limits search results to numeric values. So, the line with theid="a"
will be filtered out.Without switching the focus from the filter panel, click the Add button, select Script and paste the following code:
value.getText().replaceAll (/"/, '').toInteger() > 2
The script reads the content of the
$value$
variable and returns it as a string (for example,"1"
). Then the script replaces all the quotes and converts the string value to integer and compares it with2
.
note
To switch between different selection scopes, select the necessary value from the Target list. For example, to select the whole line that matches the template, select Complete match.
Create an HTML file and paste the following code:
HTML code
{...}In the main menu, go to Edit | Find | Replace Structurally.
From the Language list, select HTML.
Paste the following string to the Search template field:
<$tag$ $attribute$="$value$">
Select the Match case checkbox.
Click the
$tag$
variable.In the filter panel, click Add modifier, select Text and type
li
in the value field.Click the
$attribute$
variable.In the filter panel, click Add modifier, select Text and type
class
in the value field.Click the
$value$
variable.In the filter panel, click Add modifier, select Text and type
[A-Z].*
in the value field.The
[A-Z].*
regular expression limits search results to uppercase values.note
Sometimes you might need to use inline flags. In this case the regular expression will be the following:
(?-i)\b[A-Z]+\b
.(?-i)
ensures that case-insensitive mode is turned off.From the Target list, select value. This procedure highlights all the uppercase values of the
class
attribute.In the Replace template field, paste the
$to_lower_case$
variable.Click the
$to_lower_case$
variable.In the filter panel, click Add modifier, select Script and paste the following code:
value.getText().toLowerCase()
Click Find.
In the Find tool window, preview the found results and click Replace All.
Thanks for your feedback!