WebSphere eXtreme Scale Programming Guide > Access data in WebSphere eXtreme Scale > Query API > Query performance tuning
Query plan
All eXtreme Scale queries have a query plan. The plan describes how the query engine will interact with ObjectMaps and indexes. Display the query plan to determine if the query string or indexes are being used appropriately. The query plan can also be used to explore the differences that subtle changes in a query string make in the way eXtreme Scale runs a query.
The query plan can be viewed one of two ways:
- EntityManager Query or ObjectQuery getPlan API methods
- ObjectGrid diagnostic trace
getPlan method
The getPlan method on the ObjectQuery and Query interfaces return a String which describes the query plan. This string can be displayed to standard output or a log to display a query plan. Note: In a distributed environment, the getPlan method does not run against the server and will not reflect any defined indexes.
To view the plan, use an agent to view the plan on the server.
Query plan trace
The query plan can be displayed using ObjectGrid trace.
To enable query plan trace, use the following trace specification:
QueryEnginePlan=debug=enabled
See Logs and trace for details on how to enable trace and locate the trace log files.
Query plan examples
Query plan uses the word for to indicate that the query is iterating through an ObjectMap collection or through a derived collection such as: q2.getEmps(), q2.dept or a temporary collection returned by an inner loop. If the collection is from an ObjectMap, the query plan shows whether a sequential scan (denoted by INDEX SCAN), unique or non-unique index is used. Query plan uses a filter string to list the condition expressions applied to a collection.
A Cartesian product is not commonly used in object query. The following query scans the entire EmpBean map in the outer loop and scans the entire DeptBean map in the inner loop:
SELECT e, d FROM EmpBean e, DeptBean d Plan trace: for q2 in EmpBean ObjectMap using INDEX SCAN for q3 in DeptBean ObjectMap using INDEX SCAN returning new Tuple( q2, q3 )
The following query retrieves all employee names from a particular department by sequentially scanning the EmpBean map to get an employee object. From the employee object, the query navigates to its department object and applies the d.no=1 filter. In this example, each employee has only one department object reference, so the inner loop runs once:
SELECT e.name FROM EmpBean e JOIN e.dept d WHERE d.no=1 Plan trace: for q2 in EmpBean ObjectMap using INDEX SCAN for q3 in q2.dept filter ( q3.getNo() = 1 ) returning new Tuple( q2.name )
The following query is equivalent to the previous query. However, the query below performs better because it first narrows the result down to one department object by using the unique index that is defined over the DeptBean primary key field number. From the department object, the query navigates to its employee objects to get their names:
SELECT e.name FROM DeptBean d JOIN d.emps e WHERE d.no=1 Plan trace: for q2 in DeptBean ObjectMap using UNIQUE INDEX key=(1) for q3 in q2.getEmps() returning new Tuple( q3.name )
The following query finds all the employees that work for development or sales. The query scans the entire EmpBean map and performs additional filtering by evaluating the expressions: d.name = 'Sales' or d.name='Dev'
SELECT e FROM EmpBean e, in (e.dept) d WHERE d.name = 'Sales' or d.name='Dev' Plan trace: for q2 in EmpBean ObjectMap using INDEX SCAN for q3 in q2.dept filter (( q3.getName() = Sales ) OR ( q3.getName() = Dev ) ) returning new Tuple( q2 )
The following query is equivalent to the previous query, but this query runs a different query plan and uses the range index built over the field name. In general, this query performs better because the index over the name field is used for narrowing down the department objects, which run quickly if only a few departments are development or sales.
SELECT e FROM DeptBean d, in(d.emps) e WHERE d.name='Dev' or d.name='Sales' Plan trace: IteratorUnionIndex of for q2 in DeptBean ObjectMap using INDEX on name = (Dev) for q3 in q2.getEmps() for q2 in DeptBean ObjectMap using INDEX on name = (Sales) for q3 in q2.getEmps()
The following query finds departments that do not have any employees:
SELECT d FROM DeptBean d WHERE NOT EXISTS(select e from d.emps e) Plan trace: for q2 in DeptBean ObjectMap using INDEX SCAN filter ( NOT EXISTS ( correlated collection defined as for q3 in q2.getEmps() returning new Tuple( q3 ) returning new Tuple( q2 )
The following query is equivalent to the previous query but uses the SIZE scalar function. This query has similar performance but is easier to write.
SELECT d FROM DeptBean d WHERE SIZE(d.emps)=0 for q2 in DeptBean ObjectMap using INDEX SCAN filter (SIZE( q2.getEmps()) = 0 ) returning new Tuple( q2 )
The following example is another way of writing the same query as the previous query with similar performance, but this query is easier to write as well:
SELECT d FROM DeptBean d WHERE d.emps is EMPTY Plan trace: for q2 in DeptBean ObjectMap using INDEX SCAN filter ( q2.getEmps() IS EMPTY ) returning new Tuple( q2 )
The following query finds any employees with a home address matching at least one of the addresses of the employee whose name equals the value of the parameter. The inner loop has no dependency on the outer loop. The query runs the inner loop once.
SELECT e FROM EmpBean e WHERE e.home = any (SELECT e1.home FROM EmpBean e1 WHERE e1.name=?1) for q2 in EmpBean ObjectMap using INDEX SCAN filter ( q2.home =ANY temp collection defined as for q3 in EmpBean ObjectMap using INDEX on name = ( ?1) returning new Tuple( q3.home ) ) returning new Tuple( q2 )
The following query is equivalent to the previous query, but has a correlated subquery; also, the inner loop runs repeatedly.
SELECT e FROM EmpBean e WHERE EXISTS(SELECT e1 FROM EmpBean e1 WHERE e.home=e1.home and e1.name=?1) Plan trace: for q2 in EmpBean ObjectMap using INDEX SCAN filter ( EXISTS ( correlated collection defined as for q3 in EmpBean ObjectMap using INDEX on name = (?1) filter ( q2.home = q3.home ) returning new Tuple( q3 ) returning new Tuple( q2 )
Parent topic
Query performance tuning