Effective
Coding Strategies
1. Use
UPPER-lower case method to make code more readable.
PL/SQL code is made up of many
different components:
PL/SQL data structures such as
PL/SQL tables, bind
variables like Oracle Forms
items, procedures, functions,
loops, declarations, control
constructs, etc. All of these
elements break down roughly into
two types of text:
reserved words and
application-specific identifiers. The
PL/SQL compiler treats these two
kinds of text very
differently. You can improve the
readability of your code
greatly by reflecting this
difference in the way the text is
displayed. I employ the following
UPPER-lower rule to
highlight the distinction:
Enter all reserved words in
UPPERCASE and all
application identifiers in
lowercase.
Using a consistent mixture of
upper- and lowercase
words increases the readability
of your code by giving a
sense of dimension to the code.
The eye can more easily
cruise over the text and pick the
different syntactical
elements of each statement. The
uppercase words act as
signposts directing the activity
in the code. You can
focus quickly on the lowercase
words for the
application-specific content.
There are other widely-used
standards for use of case
in programming, most notably the
Visual Basic style
of UsingInitCapInLongNames.
Whichever you
approach you take, make sure it
aids in readability --
and use it consistently!
2. Use
consistent indentation.
You should use indentation -- in
a ruthlessly consistent
manner -- to reflect the logical
structure of your
program. All of the following
statements should cause
indentation to another
"level": IF statements, loops,
sections of PL/SQL Block, and
exception handlers.
Watch out for indentations which
are too short to
differentiate and too long to
hold together related
statements. I recommend an
indentation of three spaces.
3. Avoid
hard-coded literals of any kind in your programs.
Every application has its own set
of special or "magic"
values. These values might be
configuration parameters
or user information. They are
constants throughout the
application -- but that doesn’t
mean they never change. In
fact, such values will almost
always change eventually. To
avoid getting burned by literals
which have been littered
throughout your code, follow
these guidelines:
¥ Remove all literals (within
reason) from your code.
Instead, declare constants which
hold those literal values.
4. Make
sure your name describes the module accurately.
A name can make a big difference
in the usability of
your variables, modules and
parameters. Your names
should improve the
self-documenting nature of your
code. Follow these guidelines for
procedures and
functions:
¥ A procedure is an executable
statement, a command
to the PL/SQL compiler.
Consequently, the grammar
of the procedure name should be
similar to a
command: Verb_Subject.
¥ A function is used like an
expression in an executable
statement. It returns or
"represents" a value, so the
grammar of a function name should
be a noun:
Description_of_Returned_Value.
5. Test
all of your assumptions as you debug your code.
Obvious, is it? Most of the time
we aren’t even aware of all
the assumptions we make about our
programs and the
environment. I have often wasted
hours trying to find the
source of a problem because I
muttered to myself "All
right, I know that the value of
that variable is set..." or
something like that. And after I
have tested everything
else, I will finally say
"Oh, what the heck! Might as check
that variable." And lo and
behold, that was my problem.
Don’t assume anything. Test everything.
Data
Structures
6. Use
anchored declarations whenever possible.
You can use the %TYPE and
%ROWTYPE declaration
attributes to anchor the datatype
of one variable to that
of a previously-existing variable
or data structure. The
anchoring data structure can be a
column in a database
table, the entire table itself, a
programmer-defined
record or a local PL/SQL
variable. In the following
example I declare a local
variable with the same
structure as the company name:
my_company
company.name%TYPE;
In this second example I declare
a record based on a
cursor’s structure:
CURSOR company_cur
IS
SELECT company_id,
name, incorp_date
FROM company;
company_rec
company_cur%ROWTYPE;
Anchored types offer the
following benefits:
1. Synchronization with database
columns. Many PL/SQL
variables "represent"
database information inside the
program. By using %TYPE, I am
guaranteed that the
local variable’s data structure
matches that in the
database. If I instead hard-coded
my declaration, the
program could get "out of
synch" with my data
dictionary and generate errors.
For example, if the
previous declaration had declare
my_company as
VARCHAR2(30) and then I expanded
the column size
in the table to 60, my program is
likely to cause
VALUE_ERROR exceptions.
2. Normalization of local
variables. You use PL/SQL
variables to store calculated
values used throughout
the application. You can use
%TYPE to base all
declarations of these variables
against a single,
centralized variable datatype. If
you need to modify
that datatype, perhaps to expand
the maximum size
of a number to reflect higher
revenue, you only need
to change that single
declaration. All anchored
declarations will then pick up
the new constraint
when the programs are recompiled.
with the record TYPE statement.
Records offer the following
benefits:
¥ Write less code: instead of
declaring, assigning and
referencing individual variables,
you can work the
record as a whole.
¥ Protect your code from changes
in database
structures. Cursor and table
records automatically
adapt to the definitions of the
columns in the
underlying query.
Consider the program shown below.
I wrote in while
sitting in a coach airline seat.
There was very little room to
maneuver and so I wanted to
minimize my typing. I was
also unfamiliar with the details
of the data structures, such
as column names and datatypes. As
a result, I worked
almost exclusively with records
and was able to focus on
the logical structure of the
program. I filled in the details
when back on the ground.
FOR imp_rec IN imp_cur
LOOP
/* Remove after
compiles successfully! */
IF imp_cur%ROWCOUNT
> &1 THEN RETURN; END
IF;
IF NVL
(imp_rec.prft_ovrd_seq_nu, 0) = 0
THEN
get_site_prof_data
(imp_rec, li_rec);
ELSE
get_provrd_data
(imp_rec, li_rec);
END IF;
insert_line_001
(li_rec);
insert_line_062
(li_rec);
IF
duplicate_pl_site (imp_rec)
THEN
create_new_override
(imp_rec, li_rec,
dup_count, new_pl_seq_nu);
END IF;
END LOOP;
8.
Optimize foreign key lookups with PL/SQL tables.
Given the normalized nature of
our databases, you will
have to perform many foreign key
lookups (given this
key, get the entity’s name) in
your screens and reports.
You can use PL/SQL tables to
cache already-retrieved
values and thereby avoid
repetitive SQL access. This
will speed up your programs. You
build the PL/SQL
table incrementally with each
query against the
database. On subsequent calls,
however, you check the
table first and use that value.
Here is the pseudo-code
which describes the logical flow
behind this approach:
1 FUNCTION
company_name
2 (id_in IN
company.company_id%TYPE)
3 RETURN VARCHAR2
4 IS
5 BEGIN
6
get-data-from-table;
7
return-company-name;
8 EXCEPTION
9 WHEN
NO_DATA_FOUND
10 THEN
11
get-data-from-database-table;
12
store-in-PL/SQL-table;
13
return-company-name;
14 END;
9. Use
Booleans to improve readability.
You can use Boolean variables and
functions to greatly
improve the readability of your
programs. When you have
complex Boolean expressions in IF
statements and loops
(the WHILE loop condition and the
simple loop’s EXIT
statement), hide that complexity
behind either a local
PL/SQL variable or a call to a
Boolean function. The name
of the variable/function will
state clearly a summation of
all the complex logic. You can
also more easily re-use that
logic by hiding behind this kind of
interface.
To get a feel for the
improvement, compare the
following two IF statements. In
the first conditional
statement, a raise is triggered
by a complex series of
Boolean expressions. It is hard
to make sense of this
logic and, even worse, it exposes
the formula directly in
this code. What if the business
rule changes?
IF total_sal
BETWEEN 10000 AND 50000 AND
emp_status
(emp_rec.empno) = ‘N’ AND
(MONTHS_BETWEEN
(emp_rec.hiredate, SYSDATE)
> 10)
THEN
give_raise
(emp_rec.empno);
END IF;
In this second IF statement, the
details are hidden
behind the eligible_for_raise
function. As a result, the
code is more readable and the
business rule is
encapsulated within the module.
IF
eligible_for_raise (emp_rec.empno)
THEN
give_raise
(emp_rec.empno);
END IF;
Built-in
Functions and Packages
10.
Leverage fully the built-in functions.
PL/SQL offers dozens of built-in
functions to help you
get your job done with the
minimum amount of code
and fuss possible. Some of them
are straightforward,
such as the LENGTH function,
which returns the length
of the specified string. Others
offer subtle variations
which will aid you greatly -- but
only when you are
aware of those variations.
Two of my favorites in this
category of hidden talents
are INSTR and SUBSTR, both character
functions.
SUBSTR returns a sub-portion of a
string. Most
developers only use these
functions to search forward
through the strings. By passing a
negative starting
location, however, SUBSTR will
count from the end of
the string. The following expression
returns the last
character in a string:
SUBSTR (my_string,
-1, 1)
INSTR returns the position in a
string where a substring
is found. INSTR will actually
scan in reverse
through the string for the Nth
occurrence of a substring.
In addition, you can easily use
INSTR to count the
number of times a substring
occurs in a string, using a
loop of the following nature:
LOOP
substring_loc :=
INSTR (string_in,
substring_in, 1, return_value);
/* Terminate loop
when no more occurrences are found.
*/
EXIT WHEN
substring_loc = 0;
/* Found match, so
add to total and continue. */
return_value :=
return_value + 1;
END LOOP;
RETURN return_value
- 1;
11.Get
familiar with the new built-in packages.
In addition to the many built-in
functions provided by
PL/SQL, Oracle Corporation also
offers many built-in
packages. These packages of
functions, procedures and data
structures greatly expand the
scope of the PL/SQL language.
It is no longer sufficient for a
developer to become
comfortable simply with the basic
PL/SQL functions
like TO_CHAR and ROUND and so
forth. Those
functions have now become only
the inner-most layer of
useful functionality. Oracle
Corporation has built upon
those functions, and you should
do the same thing.
Just to give you a taste of what
the built-in packages
offer consider the following
possiblities:
DBMS_UTILITY.GET_TIME
Function
Returns the ealpsed time in
100ths of seconds since
an arbitrary time. You can use it
to measure subsecond
response time -- and also analyze
the impact
of your coding practices on
application performance.
DBMS_LOCK.SLEEP
Procedure
Blocks the current program from
continuing
execution for the specified number of
seconds.
Server. (PL/SQL Release 2.1 and
above)
UTL_FILE.GET_LINE
Procedure
Read a line of text from an
operating system file. And
use the PUT_LINE procedure to
write text back out
to a file. (PL/SQL Release 2.3
only)
The possibilities and
capabilities aren’t quite endless,
but they are getting there! With
each new release of the
Oracle Server, we get new
packages with which to
improve our own programs.
Loops
12. Take
advantage of the cursor FOR loop.
The cursor FOR loop is one of my
favorite PL/SQL
constructs. It leverages fully
the tight and effective
integration of the procedural
aspects of the language
with the power of the SQL
database language. It
reduces the volume of code you
need to write to fetch
data from a cursor. It greatly
lessens the chance of
introducing loop errors in your
programming Ñ and
loops are one of the more
error-prone parts of a
program. Does this loop sound too
good to be true?
Well, it isn’t Ñ it’s all true!
Suppose I need to update the
bills for all pets staying
in my pet hotel, the
Share-a-Din-Din Inn. The
example below contains an
anonymous block that
uses a cursor, occupancy_cur, to
select the room
number and pet ID number for all
occupants at the
Inn. The procedure update_bill
adds any new
changes to that pet’s room
charges.
1 DECLARE
2 CURSOR
occupancy_cur IS
3 SELECT pet_id,
room_number
4 FROM occupancy
WHERE occupied_dt =
SYSDATE;
5 occupancy_rec
occupancy_cur%ROWTYPE;
6 BEGIN
7 OPEN
occupancy_cur;
8 LOOP
9 FETCH
occupancy_cur
INTO occupancy_rec;
10 EXIT WHEN
occupancy_cur%NOTFOUND;
11 update_bill
12
(occupancy_rec.pet_id,
occupancy_rec.room_number);
13 END LOOP;
14 CLOSE
occupancy_cur;
15 END;
This code leaves nothing to the
imagination. In addition
to defining the cursor (line 2),
you must explicitly
declare the record for the cursor
(line 5), open the cursor
(line 7), start up an infinite
loop, fetch a row from the
cursor set into the record (line
9), check for an end-ofdata
condition with the cursor
attribute (line 10), and
finally perform the update. When
you are all done, you
have to remember to close the
cursor (line 14).
If I convert this PL/SQL block to
use a cursor FOR loop,
then I all I have is:
DECLARE
CURSOR
occupancy_cur IS
SELECT pet_id,
room_number
FROM occupancy
WHERE occupied_dt =
SYSDATE;
BEGIN
FOR occupancy_rec
IN occupancy_cur
LOOP
update_bill
(occupancy_rec.pet_id,
occupancy_rec.room_number);
END LOOP;
END;
Here you see the beautiful
simplicity of the cursor FOR
loop! Gone is the declaration of
the record. Gone are the
OPEN, FETCH, and CLOSE
statements. Gone is need to
check the %FOUND attribute. Gone
are the worries of
getting everything right.
Instead, you say to PL/SQL, in
effect:: ÒYou and I both know
that I want each row and I
want to dump that row into a
record that matches the
cursor. Take care of that for me,
will you?" And PL/SQL
does take care of it, just the
way any modern
programming language integrated
with SQL should.
13.
Don’t declare your FOR loop index.
Whether you use a numeric or
cursor FOR loop, the
loop index is declared for you
and implicitly by
PL/SQL. If you do declare a
variable with the same
name as the loop index, that will
be a different variable.
It will not be used by the loop
and will likely introduce
bugs into your program. Consider,
for example, the
anonymous block below. Even if I
am the only person
stored in the emp table, I will
never get a raise!
DECLARE
CURSOR emp_cur IS
SELECT empno, ename
FROM emp;
emp_rec
emp_cur%ROWTYPE;
BEGIN
FOR emp_rec IN
emp_cur
LOOP
display_emp
(emp_rec.ename);
END LOOP;
IF emp_rec.ename =
‘FEUERSTEIN’
THEN
give_raise
(emp_rec.empno,
1000000);
END IF;
END;
14.
Avoid unstructured exits from loops.
You should follow these
guidelines for terminating a
loop:
¥ Let a FOR loop complete its
specified number of
iterations. Do not use an EXIT
statement to leave
prematurely.
¥ Always make sure you include an
EXIT statement
within a simple loop.
¥ Never use EXIT to leave a WHILE
loop.
¥ Do not issue RETURNs directly
from within a loop.
Modular
Code
15.
Construct abstract data types with PL/SQL
packages.
The term "abstract data
type" is about as dry and technicalsounding
as you can get. Yet the concept
of an abstract
data type, or ADT, is something
we apply -- or should
apply -- in every single one of
our application efforts,
sometimes without even realizing
that we are doing it.
An abstract data type is a
collection of information and
operations which act on that
information. When you
create an ADT, you work with
objects as opposed to
variables, columns, and other
computer-science items.
You perform an abstraction from
the implementation
details to the "thing in
itself" and work on a higher level.
In PL/SQL, this is best done with
a package.
To give you a feel for the ADT,
consider the following
"before and after"
examples of a thermometer
implemented in Oracle Forms. The
"before" shows how
to manage the thermometer as a
sequence of separate
statements. The "after"
shows the interface provided by
the progress package.
1. Set the thermometer to 20%
completion.
Before:
:B_PROGRESS.PERCENT_DONE
:= '20 % Complete.';
:B_PROGRESS.THERMOMETER
:= 'nn';
SHOW_VIEW
('cv_progress');
SYNCHRONIZE;
After:
progress.bar (20,
2);
2. Hide the progress box when the
program completed
and control was returned back to
the user.
Before:
HIDE_VIEW
('cv_progress');
SET_WINDOW_PROPERTY
('PROGRESS_WINDOW',
VISIBLE,
PROPERTY_OFF);
SYNCHRONIZE;
After:
progress.hide;
By treating the progress box as
an object with rules
governing its use and appearance,
I can reduce greatly
the volume of code required. The
resulting statements
are also easier to understand and
maintain.
Here are some guidelines you
should follow when
designing an ADT:
1. Maintain a consistent level of
abstraction. This is
probably the most important
aspect of your ADT
implementation. All the modules
you build to
represent your abstract data
structure should operate
with the same level of data.
2. Provide a comprehensive
interface to the abstract data
type. Make sure that the user (a
programmer, in this
case) can perform all necessary
operations on the
ADT without having to go around
the interface you
build to the ADT. Hide all the
implementational
details of your ADT behind calls
to procedures and
modules -- without exception.
3. Use the package structure, the
most natural repository
for ADT code. The ADT represents
a thing by
presenting a layer of code which
allows you to perform
operations on that thing as a
whole, rather than its
individual components. The
package joins related
objects together and so
corresponds closely to the ADT.
The package clearly distinguishes
between the public
and private parts of the code.
The public objects make
up the interface to the ADT. The
private objects contain
and hide the implementational
details for the ADT.
16.
Enhance scope control with nested blocks.
PROCEDURE
delete_details
IS
BEGIN
BEGIN
DELETE FROM child1
WHERE ...;
EXCEPTION
WHEN OTHERS THEN
NULL;
END;
BEGIN
DELETE FROM child2
WHERE ...;
EXCEPTION
WHEN OTHERS THEN
NULL;
END;
END;
I can in this way use my nested
blocks to allow my
PL/SQL program to continue past
exceptions.
17.
Overload modules to make your software smarter.
Within a package
and within the declaration section of a
PL/SQL block, you
can define more than module with
the same name! The
name is, in other words,
overloaded. In the
following example, I have overloaded
the value_ok
function in the body of my check package,
which is used to
validate or check values used in my
application:
PACKAGE BODY check
IS
/* First version
takes a DATE parameter. */
FUNCTION value_ok
(date_in IN DATE)
RETURN BOOLEAN IS
BEGIN
RETURN date_in
<= SYSDATE;
END;
/* Second version
takes a NUMBER parameter. */
FUNCTION value_ok
(number_in IN NUMBER)
RETURN BOOLEAN IS
BEGIN
RETURN number_in
> 0;
END;
END;
Now I can put both versions of
value_ok to work in my
code as follows:
IF check.value_ok
(hiredate) AND
check.value_ok
(salary)
THEN
...
END IF;
I have found overloading to be
extremely useful when I
am building a layer of code which
will be used by other
developers (my PL/SQL toolbox). I
use module
overloading to hide complexities
of the programmatic
interface from my users (other
programmers). Instead of
having to know the six different
names of procedures
used to, for example, display
various kinds of data, a
developer can rely on a single
module name. In this
fashion, overloading transfers
the burden of knowledge
from the developer to the
software.
18. Use
local modules to reduce code volume and
improve
readability.
A local module is a procedure or
function which is
defined in the declaration
section of a PL/SQL block
(anonymous or named). This module
is considered local
because it is only defined within
the parent PL/SQL
block. It cannot be called by any
other PL/SQL blocks
defined outside of that enclosing block.
There are two key reasons to
create local modules:
¥ Reduce the size of the module
by stripping it of
repetitive code. This is the most
common motivation to
create a local module. The code
reduction leads to
higher-quality code since you
have fewer lines to test
and fewer potential bugs. It
takes less effort to maintain
the code, since there is simply
less to maintain. When
you do have to make a change, you
make it in one
place in the local module and the
effects are felt
immediately throughout the parent
module.
¥ Improve the readability of your
code. Even if you do
not repeat sections of code
within a module, you still
may want to pull out a set of
related statements and
package them into a local module
to make it easier to
follow the logic of the main body
of the parent module.
Consider the following example,
in which I calculate the
net present value for various
categories and then format
the result for display purposes.
PROCEDURE
display_values
(proj_sales_in IN
NUMBER, year_in IN INTEGER)
IS
BEGIN
DBMS_OUTPUT.PUT_LINE
('Total Sales: ' ||
TO_CHAR((npv
('total_sales', year_in) /
proj_sales_in *
100),'999.99'));
DBMS_OUTPUT.PUT_LINE
('Employee Labor: '
||
TO_CHAR((npv
('labor_costs', year_in) /
proj_sales_in *
100),'999.99'));
END;
In this approach, I have exposed
the way I perform the
calculation and formatting. As a
result, whenever a
change is required (different
display format, different
formula, etc.) I must upgrade
each distinct calculation. If,
on the other hand, I hide the
calculation behind the
interface of a callable module,
then the calculation is
coded only once.With the help of
a local module, the
display_values procedure is transformed
as shown below.
PROCEDURE display_values
(proj_sales_in IN
NUMBER, year_in IN INTEGER)
IS
/*-------------------
Local Module -----------------*/
PROCEDURE
display_npv (column_in IN VARCHAR2)
IS
BEGIN
DBMS_OUTPUT.PUT_LINE
(INITCAP (REPLACE
(column_in, '_', ' '))
|| ': ' ||
TO_CHAR((npv
(column_in, year_in) /
proj_sales_in
* 100), '999.99'));
END;
BEGIN
display_npv
('total_cost');
display_npv
('employee_labor');
END;
I have found that few developers
are aware of the
ability to create local modules.
I have also found that
modules-within-a-module play an
important role in
allowing me to write
well-structured, even elegant
programs. Take a look at any of
your more complex
programs and I guarantee you will
quickly identify
segments of the code which would
serve you better
bundled into a local module.
19. Code
a single RETURN for successful function
execution.
The sole purpose of a function
should be to return a
single value (this could be a
PL/SQL table or other
composite structure, of course).
To reinforce this singlemindedness,
you should code your function so
that it
has only one RETURN statement.
This RETURN should
also be the last statement in the
function. I use the
following template to achieve
this effect:
FUNCTION fname ()
RETURN datatype
IS
return_value
datatype;
BEGIN
RETURN
return_value;
END fname;
20. Use
Self-identifying Parameters
(Avoid
Boolean Values)
A common parameter for a module
is a flag which relies
on two-valued logic (TRUE or
FALSE) to pass
information to the module. The
temptation in such a
case is to declare the parameter
to be type Boolean (or to
simulate a Boolean with a
VARCHAR2 ’Y’ or ’N’ value).
With this approach, when you call
the program, you
will pass either TRUE or FALSE.
You will find in these
circumstances that while the
specification for your
program is very readable, the way
you call it in other
programs will be difficult to
understand.
Consider the parameters for the
following procedure,
which is used to generate
reports. It offers flags to
control report printing and file
clean-up.
PROCEDURE
gen_report
(report_id_in IN
NUMBER,
clean_up_in IN
BOOLEAN,
print_in IN
VARCHAR2);
In order to make gen_report more
flexible, the
developer has provided two flag
parameters:
clean_up_in TRUE if the procedure
should clean up
log files, FALSE to keep the files
in place,
usually for purposes of
debugging.
print_in Pass ’Y’ if the output
file of the report
should be printed, ’N’ to skip
the print step.
When one glances over the
procedure’s specification, the
purpose and usage of each
parameter seems clear enough.
But take a look at how I would
call this procedure:
gen_report
(report_id, TRUE, 'Y');
As you can see, these arguments
are not very
descriptive. Without the context
provided by their
names, actual Boolean parameters
cannot "selfdocument"
their effect. A maintainer of the
code must
go back to the source to
understand the impact of a
particular value. That defeats
completely the
information hiding principle of modular
programming.
A much better approach replaces
Boolean and pseudo-
Boolean parameters with character
parameters whose
acceptable values are
descriptions of the action or
situation. With this change, a
call to gen_report states
clearly its intentions:
gen_report
(report_id, 'CLEANUP', 'PRINT');
or:
gen_report
(report_id, 'NO_CLEANUP', 'PRINT');
As you can see from these
examples, I write my
procedures to accept an
unambiguous affirmative value
("CLEANUP") and the
most clearly-defined negative form
("NO_CLEANUP"). One
complication to this style is that
you need to validate the
parameter values; if you were
using a Boolean, the strong
datatyping in PL/SQL would
guarantee that a legal value is
passed. An even better
solution than either of these
approaches is to use constants
which are stored in a package. In
this way, you avoid
hard-coding values, get
compile-time verification of
correctness and make your code
more readable.
The package might look like this:
PACKAGE vals
IS
c_print CONSTANT
VARCHAR2(5) := ‘print’;
c_noprint CONSTANT
VARCHAR2(7) := ‘noprint’;
c_cleanup CONSTANT VARCHAR2(5)
:= ‘cleanup’;
c_nocleanup
CONSTANT VARCHAR2(7) :=
‘nocleanup’;
END vals;
And the calls to gen_report would
now look like this:
gen_report
(report_id, vals.c_clean, vals.c_print);
This paper should provide you
with lots of ideas on
how to improve your code.
No comments:
Post a Comment