Use the dynamic query service
There are times in the development process when you might prefer to use the dynamic query service rather than the regular EJB query service (which can be referred to as deployment query). During testing, for instance, the dynamic query service can be used at application run time, so we do not have to re-deploy the application.
Following are common reasons for using the dynamic query service rather than the regular EJB query service:
- You need to programmatically define a query at application run time, rather than at deployment.
- You need to return multiple CMP or CMR fields from a query. (Deployment queries allow only a single element to be specified in the SELECT clause.) For more information, see the topic, Example: EJB queries.
- You want to return a computed expression in the query.
- You want to use value object methods or bean methods in the query statement. For more information, see the topic, Path expressions.
- You want to interactively test an EJB query during development, but do not want to repeatedly deploy the application each time you update a finder or select query.
The dynamic query API is a stateless session bean; using it is similar to using any other J2EE EJB application bean. It is included in the com.ibm.websphere.ejbquery in the API package.
The dynamic query bean has both a remote and a local interface. To return remote EJB references from the query, or if the query statement contains remote methods, use the query remote interface:
remote interface = com.ibm.websphere.ejbquery.Query
remote home interface = com.ibm.websphere.ejbquery.QueryHome To return local EJB references from the query, or if the query statement contains local methods, use the query local interface:local interface = com.ibm.websphere.ejbquery.QueryLocal
local home interface = com.ibm.websphere.ejbquery.QueryLocalHomeBecause it uses less application server memory, the local interface ensures better overall EJB performance than the remote.
- Verify that the query.ear application file is installed on the application server on which the application is to run, if that server is different from the default application server created during installation of the product.
The query.ear file is located in the app_server_root directory, where <WAS_HOME> is the location of the WAS. The product installation program installs the query.ear file on the default application server using a JNDI name of
com/ibm/websphere/ejbquery/Query
(You or the system administrator can change this name.)
- Set up authorization for the methods executeQuery(), prepareQuery(), and executePlan() in the remote and local dynamic query interfaces to control access to sensitive data. (This step is necessary only if the application requires security.)
Because we cannot control which ASN names, CMP fields, or CMR fields can be used in a dynamic EJB query, you or the system administrator must place restrictions on use of the methods. If, for example, a user is permitted to run the executeQuery method, he or she can run any valid dynamic query. In a production environment, you certainly want to restrict access to the remote query interface methods.
- Write the dynamic query as part of the application client code. We can refer to the example topics, Remote interface dynamic query example, and Local interface dynamic query example, as query models; they illustrate which import statements to use.
- If the CMP to query is on a different module, you should:
- do a remote lookup on query.ear
- map the query.ear file to the server that the queried CMP bean is installed on.
- Compile and run the client program with the file qryclient.jar in the classpath.
Example
- Use the remote interface for Dynamic query.
When you run a dynamic EJB query using the remote interface, you are calling the executeQuery method on the Query interface. The executeQuery method has a transaction attribute of REQUIRED for this interface; therefore we do not need to explicitly establish a transaction context for the query to run.
When you run a dynamic EJB query using the remote interface, you are calling the executeQuery method on the Query interface. The executeQuery method has a transaction attribute of REQUIRED for this interface; therefore we do not need to explicitly establish a transaction context for the query to run.
Begin with the following import statements:
import com.ibm.websphere.ejbquery.QueryHome; import com.ibm.websphere.ejbquery.Query; import com.ibm.websphere.ejbquery.QueryIterator; import com.ibm.websphere.ejbquery.IQueryTuple; import com.ibm.websphere.ejbquery.QueryException;Next, write the query statement in the form of a string, as in the following example that retrieves the names and ejb-references for underpaid employees:
String query = "select e.name as name , object(e) as emp from EmpBean e where e.salary < 50000";Create a Query object by obtaining a reference from the QueryHome class. (This class defines the executeQuery method.) Note that for the sake of simplicity, the following example uses the dynamic query JNDI name for the Query object:
InitialContext ic = new InitialContext(); Object obj = ic.lookup("com/ibm/websphere/ejbquery/Query"); QueryHome qh = ( QueryHome) javax.rmi.PortableRemoteObject.narrow( obj, QueryHome.class ); Query qb = qh.create();You then must specify a maximum size for the query result set, which is defined in the QueryIterator object, which is included in the Class QueryIterator. This class is included in the You then must specify a maximum size for the query result set, which is defined in the QueryIterator object, which is included in the QueryIterator API package. This example sets the maximum size of the result set to 99:
QueryIterator it = qb.executeQuery(query, null, null ,0, 99 );
The iterator contains a collection of IQueryTuple objects, which are records of the return collection values. Corresponding to the criteria of our example query statement, each tuple in this scenario contains one value of name and one value of object(e). To display the contents of this query result, use the following code:
while (it.hasNext() ) { IQueryTuple tuple = (IQueryTuple) it.next(); System.out.print( it.getFieldName(1) ); String s = (String) tuple.getObject(1); System.out.println( s); System.out.println( it.getFieldName(2) ); Emp e = ( Emp) javax.rmi.PortableRemoteObject.narrow( tuple.getObject(2), Emp.class ); System.out.println( e.getPrimaryKey().toString()); }The output from the program might look something like the following:
name Bob emp 1001 name Dave emp 298003 ...Finally, catch and process any exceptions. An exception might occur because of a syntax error in the query statement or a run-time processing error. The following example catches and processes these exceptions:
} catch (QueryException qe) { System.out.println("Query Exception "+ qe.getMessage() ); }
Handling large result collections for the remote interface queryIf we intend the query to return a large collection, we have the option of programming it to return results in multiple smaller, more manageable quantities. Use the skipRow and maxRow parameters on the remote executeQuery method to retrieve the answer in chunks. For example:
int skipRow=0; int maxRow=100; QueryIterator it = null; do { it = qb.executeQuery(query, null, null ,skipRow, maxRow ); while (it.hasNext() ) { // display result skipRow = skipRow + maxRow; } } while ( ! it.isComplete() ) ;
- Use the local interface for Dynamic query.
When you run a dynamic EJB query using the local interface, you are calling the executeQuery method on the QueryLocal interface. This interface does not initiate a transaction for the method; therefore you must explicitly establish a transaction context for the query to run.
To establish a transaction context, the following example calls the begin() and commit() methods. An alternative to using these methods is simply embedding the query code within an EJB method that runs within a transaction context.
Begin your query code with the following import statements:
import com.ibm.websphere.ejbquery.QueryLocalHome; import com.ibm.websphere.ejbquery.QueryLocal; import com.ibm.websphere.ejbquery.QueryLocalIterator; import com.ibm.websphere.ejbquery.IQueryTuple; import com.ibm.websphere.ejbquery.QueryException;Next, write the query statement in the form of a string, as in the following example that retrieves the names and ejb-references for underpaid employees:
String query = "select e.name, object(e) from EmpBean e where e.salary < 50000 ";Create a QueryLocal object by obtaining a reference from the QueryLocalHome class. (This class defines the executeQuery method.) Note that in the following example, ejb/query is used as a local EJB reference pointing to the dynamic query JNDI name (com/ibm/websphere/ejbquery/Query):
InitialContext ic = new InitialContext(); QueryLocalHome qh = ( LocalQueryHome) ic.lookup( "java:comp/env/ejb/query" ); QueryLocal qb = qh.create();The last portion of code initiates a transaction, calls the executeQuery method, and displays the query results. The QueryLocalIterator class is instantiated because it defines the query result set. This class is included in the Class QueryIterator API package. Keep in mind that the iterator loses validity at the end of the transaction; use the iterator in the same transaction scope as the executeQuery call.
userTransaction.begin(); QueryLocalIterator it = qb.executeQuery(query, null, null); while (it.hasNext() ) { IQueryTuple tuple = (IQueryTuple) it.next(); System.out.print( it.getFieldName(1) ); String s = (String) tuple.getObject(1); System.out.println( s); System.out.println( it.getFieldName(2) ); EmpLocal e = ( EmpLocal ) tuple.getObject(2); System.out.println( e.getPrimaryKey().toString()); } userTransaction.commit();In most situations, the QueryLocalIterator object is demand-driven. That is, it causes data to be returned incrementally: for each record retrieval from the database, the next() method must be called on the iterator. (Situations can exist in which the iterator is not demand-driven. For more information, consult the "Local query interfaces" subsection of the Dynamic query performance considerations topic.)
Because the full query result set materializes incrementally in the application server memory, we can control its size. During a test run, for example, you may decide that return of only a few tuples of the query result is necessary. In that case you should use a call of the close() method on the QueryLocalIterator object to close the query loop. Doing so frees SQL resources that the iterator uses. Otherwise, these resources are not freed until the full result set accumulates in memory, or the transaction ends.
- Example: Use the remote interface for Dynamic query
When you run a dynamic EJB query using the remote interface, you are calling the executeQuery method on the Query interface. The executeQuery method has a transaction attribute of REQUIRED for this interface; therefore we do not need to explicitly establish a transaction context for the query to run.
- Example: Use the local interface for Dynamic query
When you run a dynamic EJB query using the local interface, you are calling the executeQuery method on the QueryLocal interface. This interface does not initiate a transaction for the method; therefore you must explicitly establish a transaction context for the query to run.
- Dynamic query performance considerations
While using a dynamic query can be convenient, there are times when it can have an impact on the application performance.
- Access intent implications for dynamic query
WebSphere Application Server gives you the option to set access intent policies for your entity enterprise beans as a way of managing their transfer of data with the underlying data store. An access intent policy controls the isolation level used on the data source connection, as well as the database locks used during data retrieval. By manipulating these elements, we can maximize the efficiency of the application's data flow.
- Dynamic query API: prepareQuery() and executePlan() methods
Use these methods to more efficiently allocate the overhead associated with dynamic query. They are equivalent in function to the prepareStatement() and executeQuery() methods of the JDBC API.
- Dynamic and deployment EJB query services comparison
Use the dynamic query service to build and execute queries against entity beans constructed dynamically at run time, rather than defining them at deployment time. By using dynamic query you gain the flexibility of queries defined at run time and utilize the power of EJB-Query Language (QL). Apart from supporting all of the capabilities of an EJB-QL query, dynamic query adds functionality not available to standard static query. Two examples are the ability to select multiple data fields directly from the bean itself (static queries currently only allow one) and executing business methods directly in the query.
Related concepts
EJB query language Path expressions