Beruflich Dokumente
Kultur Dokumente
Components
Author: Steve Muench, ADF Development Team
Date: January 19, 2006
Revision 1.9 ( Revision History)
Abstract
This paper provides a high-level description of the key ADF Business Components classes in the Oracle
Application Development Framework, summarizing the methods that ADF developers write, call, and override
most frequently.
Contents
Overview
Logical Architecture of Services Built with ADF Business Components
Most Commonly Used Methods in the Client Tier
ApplicationModule Interface
Transaction Interface
ViewObject Interface
RowSet Interface
RowSetIterator Interface
Row Interface
StructureDef Interface
AttributeDef Interface
AttributeHints Interface
Most Commonly Used Methods In the Business Service Tier
Controlling Custom Java Files For Your Components
ApplicationModuleImpl Class
DBTransactionImpl2 Class
EntityImpl Class
EntityDefImpl Class
ViewObjectImpl Class
ViewRowImpl Class
Setting Up Your Own Layer of Framework Base Classes
Conclusion
Overview
Service-oriented J2EE Applications built using the ADF framework encapsulate their business logic into one or
more business services. Clients access these business services either as web services or, when the client is a
user interface, using the ADF data-binding layer that implements the Data Controls and Bindings from JSR-227.
The data controls abstract the back-end business service implementations and binding objects link front-end user
interface controls in a declarative way to back-end data.
ADF supports business services implemented as simple Java objects, web services, EJB Session beans, and
framework-based Java objects that inherit built-in functionality from the ADF Business Components base classes.
These ADF Business Components are business-tier building-block classes, and related design-time extensions
to JDeveloper, that give you a prescriptive architecture for building richly-functional and cleanly layered J2EE
business services with great performance. This paper offers a high-level description of the key components that
comprise ADF Business Components, and provides a summary of the methods that developers leveraging ADF
BC base classes write, call, and override most frequently while building their J2EE-compliant business services.
Logical Architecture of Services Built with ADF Business Components
You can use business services you build with ADF Business Components in one of three ways:
1.
2.
3.
With the J2EE web service option, your web service client interacts with the service using a web service proxy
class generated by the client tool environment based on the service's WSDL service description. The methods
that appear on your web service are the custom methods you've written on your business service component.
With the local Java service class and the EJB session bean options, your Java client code interacts with either
the base ADF component interfaces or custom component interfaces that JDeveloper 10g generates for you
automatically after indicating which custom methods should appear on the component's client interface.
Since ADF adopts this best-practice approach of having your client work with interfaces instead of directly with
implementation classes, it offers the additional benefit of allowing you to change between deploying your
business service as a local Java class or an EJB session bean without changing your client code. Along with the
custom service interface, JDeveloper generates appropriate client-side proxy implementation classes that
implement your custom interface. As such, it's an implementation detail you don't need to worry about yourself.
With either 2-tier or 3-tier deployment of your application modules, thing just work...which is nice.
Figure 1 shows the relationships between the key components in the Oracle Application Development
Framework. The dashed vertical line in the diagram represents the clean separation of component interfaces that
are available on the client, and which ones are only available in the business tier implementation.
YourProjectNameCSMT.jar
This archive contains all of the implementation classes that comprise your project's business service application
module components and other ADF BC components that your application modules use. We picked the MT suffix
in the name to represent Middle Tier. When deploying your application module as an EJB Session Bean, this JAR
file lives on the EJB middle tier. It never is downloaded to the remote client.
YourProjectNameCSCommon.jar
This archive contains any custom client interfaces, generated custom remote client proxy classes, custom
domains, and message bundles that are accessible by both the client-tier and the business service tier. We
picked the suffix Common in the archive name to remind you that this jar is common to both tiers. This file is
generally relatively quite small compared to the middle-tier JAR.
The following sections briefly explain each component and give tips about the methods J2EE developers using
ADF Business Components will use most frequently for each kind of component.
Most Commonly Used Methods in the Client Tier
All of the interfaces described in this section are designed for use by client-layer code and are part of
the oracle.jbo.* package.
NOTE:
The corresponding implementation classes for these oracle.jbo.* interfaces are consciously designed to not be directly a
client code. As we'll see in the Most Commonly Used Methods In the Business Service Tier section below, the implementation
in the oracle.jbo.server.* package and generally have the suffix Impl in their name to help remind you not to using the
client-layer code.
ApplicationModule Interface
The ApplicationModule
other ADF components and coordinateswith them to implement a number of J2EE design patterns important to
business application developers. These design pattern implementations enable your client code to work easily
with updateable collections of value objects, based on fast-lane reader SQL queries that retrieve only the data
needed by the client, in the way the client wants to view it. Changes made to these value objects are
automatically coordinated with your persistent business domain objects in the business service tier to enforce
business rules consistently and save changes back to the database.
If you want
to...
findViewObject()
Access an
existing view
object instance
by name
Creating a new createViewObject()
view object
instance from an
existing definition
Creating a new createViewObjectFromQueryStmt()
view object
instance from a Note This incurs runtime overhead to describe the "shape" of the dynamic query's SELECT list.
SQL Statement :
We recommend using this only when you cannot know the SELECT list for the query at
design-time. Furthermore, if you are creating the dynamic query based on some kind of
custom runtime repository, you can follow this tip to create (both read-only and
updateable) dynamic view objects without the runtime-describe overhead with a little more
work. If only the WHERE needs to be dynamic, create the view object at design time, then
set the where clause dynamically as needed using
ViewObject
API's.
Access a nested findApplicationModule()
application
module instance
by name
createApplicationModule()
Create a new
nested
application
module instance
from an existing
definition
Find a view
findViewObject()
object instance in
a nested
Note To find an instance of a view object belonging to a nested application module you use a
application
:
dot notationnestedAMInstanceName.VOInstanceName
module
getTransaction()
Accessing the
current
transaction
object
In addition to generic ApplicationModule access, JDeveloper 10g can generate you a
custom YourApplicationModuleNameinterface containing service-level custom methods that you've chosen
to expose to the client. You do this by visiting the Client Interface tab of the Application Module editor, and
shuttling the methods you'd like to appear in your client interface into the Selected list. JDeveloper will also
commit()
Commit pending
changes
Rollback pending rollback()
changes
Execute a one-time executeCommand()
database command
or block of PL/SQL Note Commands that require retrieving OUT parameters, that will be executed more than
:
once, or that could benefit by using bind variables should not use this method. Instead,
expose a custom method on your application module class as described here.
Validate all pending validate()
invalid changes in
the transaction
Change the default setLockingMode()
locking mode
Note You can set the locking mode in your configuration by setting the
:
property jbo.locking.mode to one of the four supported
values: none, optimistic, pessimistic, optupdate. If you don't explicitly set it, it
will default to pessimistic. For web applications, we recommend
using optimistic or optupdate modes.
Decide whether to setBundledExceptionMode()
use bundled
exception reporting Note ADF controller layer support sets this parameter to true automatically for web
mode or not.
:
applications.
Decide whether
setClearCacheOnCommit()
entity caches will be
cleared upon a
Note:
Default is false
successful commit
of the transaction.
Decide whether
setClearCacheOnRollback()
entity caches will be
cleared upon a
Note:
Default is true
rollback of the
transaction.
clearEntityCache()
Clear the entity
cache for a specific
entity object.
ViewObject Interface
A ViewObject
encapsulates a database query and simplifies working with the
RowSet
of results it
produces. You use view objects to project, filter, join, or sort business data using SQL from one or more tables
into exactly the format that the user should see it on the page or panel. You can create "master/detail" hierarchies
of any level of depth or complexity by connecting view objects together using view links. View objects can
produce read-only query results, or by associating them with one ore more entity objects at design time, can be
fully updateable. Updateable view objects can support insertion, modification, and deletion of rows in the result
collection, with automatic delegation to the correct business domain objects.
Every ViewObject aggregates a "default rowset" for simplifying the 90% of use cases where you work with a
single
RowSet
of results for theViewObject's query. A ViewObject implements all the methods on
the RowSet interface by delegating them to this default RowSet. That means you can invoke
any
RowSet
methods on any ViewObject as well.
Every ViewObject implements the
StructureDef
interface to provide information about the number and types
of attributes in a row of its rowsets. So you can call StructureDef methods right on any view object.
Note This WHERE clause augments any WHERE clause specified at design time in the
:
base view object. It does not replace it.
setOrderByClause()
createViewCriteria()
Note You then create one or more ViewCriteriaRow objects using
:
the createViewCriteriaRow()method on the ViewCriteria object you
created. Then, you add() these view criteria rows to the view criteria collection
and apply the criteria using the method below.
applyViewCriteria()
insertRow()
Note It's a good habit to always immediately insert a newly created row into the
:
rowset. That way you will avoid a common gotcha of creating the row but
forgetting to insert it into the rowset.
last()
Get the last row of the
iterator's rowset
previous()
Get the previous row of
the iterator's rowset
reset()
Reset the current row
pointer to the slot before
the first row
closeRowSetIterator()
Close an iterator when
done iterating
Set a given row to be the setCurrentRow()
current row
Remove the current row removeCurrentRow()
Remove the current row to removeCurrentRowAndRetain()
later insert it at a different
location in the same
iterator.
Remove the current row removeCurrentRowFromCollection()
from the current collection
but do not remove it from
the transaction.
Set/change the number of setRangeSize()
rows in the range (a
"page" of rows the user
can see)
Scroll to view the Nth page scrollToRangePage()
of rows (1-based)
Scroll to view the range of scrollRangeTo()
rows starting with row
number N
Set row number N in the setCurrentRowAtRangeIndex()
range to be the current
row
Get all rows in the range getAllRowsInRange()
as a Row array
Row Interface
A Row
is generic value object. It contains attributes appropriate in name and Java type for
the ViewObject that it's related to.
getAttribute()
Get the value of an attribute by name
setAttribute()
Set the value of an attribute by name
writeXML()
Produce an XML document for a single row
validate()
Eagerly validate a row
readXML()
Read row attribute values from XML
remove()
Remove the row
Flag a newly created row as temporary (until updated again) setNewRowState(Row.STATUS_INITIALIZED)
Retrieve the attribute structure definition information for a rowgetStructureDef()
getKey()
Get the Key object for a row
In addition to generic Row access, JDeveloper 10g can generate you a
custom YourViewObjectNameRowinterface containing your type-safe attribute getter and setter methods, as
well as any desired row-level custom methods that you've chosen to expose to the client. You do this by visiting
the Client Row Interface tab of the View Object editor, and shuttling the methods you'd like to appear in your
client interface into theSelected list. JDeveloper will also generate an
appropriate YourViewObjectNameRowClientclient proxy implementation class that is used automatically by
your remote client in the case that you deploy your application module as an EJB Session Bean or whenever you
use your application module in Batch Mode.
StructureDef Interface
A StructureDef
is an interface that provides access to runtime metadata about the structure of a
Row
.
In addition, for convenience every ViewObject implements the StructureDef interface as well, providing
access to metadata about the attributes in the resulting view rows that its query will produce.
Access attribute definitions for all attributes in the view object getAttributeDefs()
row
findAttributeDef()
Find an attribute definition by name
getAttributeDef()
Get attribute definition by index
getAttributeCount()
Get number of attributes in a row
AttributeDef Interface
An AttributeDef
provides attribute definition information for any attribute of a View Object row or
Entity Object instance like attribute name, Java type, and SQL type. It also provides access to custom attributespecific metadata properties that can be inspected by generic code you write, as well as UI hints that can assist
in rendering an appropriate user interface display for the attribute and its value.
named oracle.jbo.server.* and have the Impl suffix in their name to help remind you not to using them in
your client-layer code.
In your business service tier implementation code, you can use any of the same methods that are available to
clients above, but in addition you can also:
Safely cast any oracle.jbo.* interface to its oracle.jbo.server.* package implementation class
and use any methods on that Impl class as well.
Override any of the base framework implementation class' public or protected methods to augment
or change its default functionality by writing custom code in your component subclass before or after
calling super.methodName().
This section provides a summary of the most frequently called, written, and overridden methods for the key ADF
Business Components classes.
Controlling Custom Java Files For Your Components
Before examining the specifics of individual classes, it's important to understand how you can control which
custom Java files each of your components will use. When you don't need a customized subclass for a given
component, you can just let the base framework class handle the implementation at runtime.
Each business component you create comprises a single XML component descriptor, and zero or more related
custom Java implementation files. Each component that supports Java customization has a Java tab in its
component editor in the JDeveloper 10g IDE. By checking or unchecking the different Java classes, you control
which ones get created for your component. If none of the boxes is checked, then your component will be an
XML-only component, which simply uses the base framework class as its Java implementation. Otherwise, tick
the checkbox of the related Java classes for the current component that you need to customize. JDeveloper 10g
will create you a custom subclass of the framework base class in which you can add your code.
NOTE:You can setup global IDE preferences for which Java classes should be generated by default for each
ADF business component type by selecting Tools | Preferences... | Business Components and ticking
the checkboxes to indicate what you want your defaults to be.
A best practice is to always generate Entity Object and View Row classes, even if you don't require any custom
code in them other than the automatically-generated getter and setter methods. These getter and setter methods
offer you compile-time type checking that avoids discovering errors at runtime when you accidentally set an
attribute to an incorrect kind of value.
ApplicationModuleImpl Class
The ApplicationModuleImpl
class is the base class for application module components. Since the
application module is the ADF component used to implement a business service, think of the application module
class as the place where you can write your service-level application logic. The application module coordinates
with view object instances to support updateable collections of value objects that are automatically "wired" to
business domain objects. The business domain objects are implemented as ADF entity objects.
Methods You Typically Call on ApplicationModuleImpl
custom application
module state from the
state management XML
snapshot
DBTransactionImpl2 Class
The DBTransactionImpl2
constructed by theDatabaseTransactionFactory
commit()
Commit the transaction
rollback()
Rollback the transaction
validate()
Eagerly validate any pending invalid changes in the
transaction
Create a JDBC PreparedStatement using the transaction's createPreparedStatement()
Connection object
Create a JDBC CallableStatement using the transaction's createCallableStatement()
Connection object
createStatement()
Create a JDBC Statement using the transaction's
Connection object
addWarning()
Add a warning to the transaction's warning list.
Methods You Typically Override in Your Custom DBTransactionImpl2 Subclass
commit()
Perform custom code before or after the
transaction commit operation
rollback()
Perform custom code before or after the
transaction rollback operation
In order to have your custom DBTransactionImpl2 subclass get used at runtime, there are two steps you must
follow:
1.
2.
If you want
to...
If you want
to...
Perform attributespecific validation
method validation
Perform entity-level publicbooleanvalidateSomething()
validation
Note Register the entity-level validator method by adding a "MethodValidator" on the entity in
:
the Validation panel of the Entity Object editor.
Calculate the value Add your calculation code to the generated getAttributeName()method.
of a transient
attribute
Methods You Typically Override on EntityImpl
It is also possible to override the validateEntity() method to write entity-level validation code, however if
you want to maintain the benefits of the ADF bundled exception mode where the framework collects and
reports a maximal set of validation errors back to the client user interface it is recommended to adopt
the MethodValidator approach suggested in the table above. This allows the framework to automatically
collect all of your exceptions that your validation methods throw without your having to understand the bundled
exception implementation mechanism. Overriding the validateEntity() method directly shifts the
responsibility on your own code to correctly catch and bundle the exceptions like the ADF framework would have
done by default, which is non-trivial and a chore to remember and hand-code each time.
EntityDefImpl Class
The EntityDefImpl
class is a singleton, shared metadata object for all entity objects of a given type in
a single Java VM. It defines the structure of the entity instances and provides methods to create new entity
instances and find existing instances by their primary key.
Methods You Typically Call on EntityDefImpl
Perform any of the common view See the ViewObject Interface, RowSet Interface, and RowSetIterator
object, rowset, or rowset iterator
Interface sections above.
operations from inside your class,
which can also be done from the
client
Set an additional runtime WHERE setWhereClause()
clause on the default rowset
setWhereClauseParam()
Set bind variable values on the
default rowset
getFilteredRows()
Retrieved a subset of rows in a
view object's row set based on
evaluating an in-memory filter
expression.
Retrieved a subset of rows in the getFilteredRowsInRange()
current range of a view object's row
set based on evaluating an inmemory filter expression.
Set the number of rows that will be setFetchSize()
fetched from the database per
round-trip for this view object.
Note The default fetch size is a single row at a time. This is definitely not
:
optimal if your view object intends to retrieve many rows, so you should
either set the fetch size higher at design time on the Tuning tab of the
View Object editor, or set it at runtime using this API.
Methods You Typically Write in Your Custom ViewObjectImpl Subclass
:
Perform custom processing of the
setting of a view row attribute
need to write the custom calculation logic inside the method body.
setAttrName()
Note JDeveloper generates the skeleton of the method for you, but you
:
need to write the custom logic inside the method body if required.
isAttributeUpdateable()
Determine the updateability of an
attribute in a conditional way.
Custom methods that expose logical
doSomething()
operations on the current row, optionally
callable by clients
Note Often these view-row level custom methods simply turn around and
:
delegate to a method call on the underlying entity object related to
the current row.
JDeveloper 10g can generate you a custom YourViewObjectNameRowinterface containing view row custom
methods that you've chosen to expose to the client. You can accomplish this by visiting the Client Row
Interface tab of the View Object editor, and shuttling the methods you'd like to appear in your client interface into
the Selected list.
Methods You Typically Override in Your Custom ViewRowImpl Subclass
packagecom.yourcompany.adfextensions;
importoracle.jbo.server.ApplicationModuleImpl;
publicclassCustomApplicationModuleImplextendsApplicationModuleImpl{
/*
*Wemightnotyethaveanycustomcodetoputhereyet,but
*thefirsttimeweneedtoaddagenericfeaturethatallof
*ourcompany'sapplicationmodulesneed,wewillbeveryhappy
*thatwethoughtaheadtoleaveourselvesaconvenientplace
*inourclasshierarchytoadditsothatalloftheapplication
*moduleswehavecreatedwillinstantlybenefitbythatnewfeature,
*behaviorchange,orevenperhaps,bugworkaround.
*/
}
A common set of customized framework base classes in a package name of your own choosing
like com.yourcompany.adfextensions, each importing the oracle.jbo.server.* package, would
consist of the following classes.
publicclassCustomEntityImplextendsEntityImpl
publicclassCustomEntityDefImplextendsEntityDefImpl
publicclassCustomViewObjectImplextendsViewObjectImpl
publicclassCustomViewRowImplextendsViewRowImpl
publicclassCustomApplicationModuleImplextendsApplicationModuleImpl
publicclassCustomDBTransactionImplextendsDBTransactionImpl2
publicclassCustomDatabaseTransactionFactoryextends
DatabaseTransactionFactory
For completeness, you may also want to create customized framework classes for the following classes as well,
but overriding anything in these classes would be a fairly rare requirement.
publicclassCustomViewDefImplextendsViewDefImpl
publicclassCustomEntityCacheextendsEntityCache
publicclassCustomApplicationModuleDefImplextendsApplicationModuleDefImpl
Conclusion
Hopefully this short list of methods will assist you in getting started with Oracle ADF Business Components to
focus your attention on these most frequently used methods instead of feeling bewildered about the many other
methods that could be overridden but which aren't often done in practice.
If you have already been using ADF Business Components for some time and notice some methods you use
frequently that didn't make my list, please send me an email to let me know what I should add to the list. Include
in your email a brief description of how you're using the method, or what functionality you often providing in the
overridden version of the method. Thanks!
Revision History
Date
Comments
18-Nov- Created
2004
19-Nov- Updated diagram to include StructureDef, AttributeDef, and AttributeHints, and added some text on
2004
the generated client proxy classes.
20-Nov- Added a comment about passing null to the createRowSetIterator method
2004
30-Nov- Added getPostedAttribute() and getEntityState()
2004
1-DecClarified getPostState() and getEntityState()
2004
1-JanAdded some use case information for prepareSession()
2005
14-Mar- Clarified when to use VOImpl.create() vs AMImpl.prepareSession()
2005
15-Mar- Clarified when to prepareForDML() vs postChanges()
2005
14-Jun- Clarified that EntityImpl.create() is used for programmatically populating primary key attributes as well
2005
19-Jan- Updated javadoc links
2006
false ,,,,,,,,,,,,,,,