In the previous section, we looked briefly at CORBA IDL, the language used to describe objects for remote access. In this next section, we examine IDL in detail—first by looking at each supported construct and its Java equivalent and then by looking at the
shortcomings of the language.
IDL Constructs
As has been stated over and over, IDL allows developers to describe code such that virtually any programming language can understand it. The example of COBOL code invoking methods on a Java object is often used, but this example also implies that the COBOL application fully understands the methods exposed by the Java application. If the Java application has a method that returns a java.awt.Image object, the COBOL application must be able to fully understand that object. Unfortunately, it’s not possible to wave a magic wand and magically have all legacy applications understand modern programming constructs. Many of those legacy applications were written before computers could even display any form of image. To allow all languages to intercommunicate, IDL allows lowest-common-denominator communication, which basically means that all methods must accept as parameters, and return as a return value, either a byte, some form of number, or a string. This, of course, does present some challenges to the developer, but there are possible workarounds. As a conclusion to this section, we’ll look at an application that allows image data to be passed from client to server.
IDL Data Types
All communication with CORBA objects is performed using any of the data types supported by IDL. If some entity does not have a CORBA equivalent, it must be abstracted using one of the supported data types. Table 22.1 contains all data types, along with their Java equivalent. The only type that may not be familiar to you is the any data type. An any , as its name implies, is a holder class that can reference any entity in the CORBA universe.
TABLE 22.1 IDL-TO-JAVA MAPPING
IDL Java
char char octet byte boolean boolean TRUE true FALSE false
string java.lang.String
short short long int float float double double
any CORBA.Any
module
Starting at the highest level, the first IDL construct that we examine is the module. The module, like its Java equivalent the package, allows for the logical grouping of entities.
Listing 22.6 contains an IDL snippet that makes use of the module construct. Note that the construct starts with the keyword module and is immediately followed by the logical name associated with it. The entire module is delimited by curly braces, and a
semicolon immediately follows the closing brace.
LISTING 22.6 THE IDL module CONSTRUCT MAPS TO THE JAVA PACKAGE CONSTRUCT
module userData {
interface Person {
string getName();
};
};
If your application design calls for subpackages, the module should be placed inside of another module. Listing 22.6 contains an example that would map into the Java package org.luke.userData .
LISTING 22.6 USING SUBMODULES TO CREATE A MULTILEVEL PACKAGE STRUCTURE
module org {
module luke {
module userData {
interface Person { string getName();
};
};
};
};
interface
As you may have guessed from the development we’ve done in this chapter, the interface keyword maps to a Java interface. This should not be confused with mapping directly to a Java class, because it’s quite possible that many Java classes might implement the interface defined by a single IDL interface or one Java class might implement many IDL interfaces. When the idl2java compiler compiles an IDL interface, the Java interface is named interfaceNameOperations . In most situations, you’ll not deal directly with this interface. You simply extend the skeleton class, which in turn implements the interfaceNameOperations interface.
Interface inheritance is supported and, like much of IDL, uses syntax borrowed from C++.
Listing 22.7 contains two interfaces, ParentInterface and ChildInterface , where ChildInterface inherits ParentInterface .
LISTING 22.7 INHERITANCE IN IDL USES A SYNTAX SIMILAR TO C++
interface ParentInterface { string parentMethod();
};
interface ChildInterface : ParentInterface { string childMethod();
};
Operations
An interface as a whole is comprised of a collection of operations. An operation maps directly into a Java method; however, an IDL operation signature is slightly more
complicated. All parameters passed into an operation must be specified as either “in,”
“inout,” or “out” parameters. This specification tells the ORB exactly how it needs to manage data as it travels from client to server, and potentially back again. Table 22.2 discusses how each of the modifiers function, and Listing 22.8 contains some IDL that uses each modifier.
In keeping with the goal of supporting the lowest-common-denominator programming language, there’s no support for method overloading in IDL.
TABLE 22.2 IDL OPERATION MODIFIERS
Parameter
Modifier Function
in Specifies a parameter that passes data into a method but is not changed by the method invocation
inout Specifies a parameter that passes data into a method and is potentially changed during the method invocation
out Specifies a parameter that does not pass data into the method but is potentially modified by the method execution
LISTING 22.8 PARAMETER MODIFIERS IN CORBA IDL
interface ParameterModifierExample {
string getDataForIn(in string name);
void getDataForInout(in string name, inout results);
void getDataForOut(in string name, out results);
};
attribute
In addition to associating methods with an interface, it’s also possible to specify an attribute exposed by the interface. An attribute maps to a private member variable, which is obtained using getter and setter methods. Listing 22.9 contains some IDL that makes
use of the attribute keyword. The mapping of this IDL to Java is shown in Listing 22.10.
LISTING 22.9 ATTRIBUTES IN IDL ARE USED TO MODEL THE NONBEHAVIORAL ASPECTS OF AN OBJECT
interface AttributeDemo {
attribute string attribute1;
attribute long attribute2;
};
LISTING 22.10 IDL ATTRIBUTES MAP TO PRIVATE INSTANCE VARIABLES WITH PUBLIC GETTER AND SETTER METHODS
public class AttributeDemoImplementation extends AttributeDemoImplBase { private String attribute1;
private int attribute2;
public String attribute1() { return attribute1;
}
public void attribute1(String attribute1) { this.attribute1 = attribute1;
}
public int attribute2() { return attribute2;
}
public void attribute2(int attribute2) { this.attribute2 = attribute2;
} }
struct
In addition to specifying CORBA entities using the interface keyword, it’s also possible to specify an entity using the struct keyword. Although a struct may have little meaning to the Java developer, those of us with experience in C/C++ have an intimate understanding of its use. A struct is basically a collection of data that includes no behavior at all. A struct maps directly into a Java class with public member
variables matching all struct attributes. Listing 22.11 contains a simple struct defined using IDL. Listing 22.12 contains the mapping of that struct into Java.
LISTING 22.11 THE struct CONSTRUCT IS USED TO CREATE ENTITY OBJECTS
struct StructDemo {
string attribute1;
string attribute2;
long attribute3;
};
LISTING 22.12 AN IDL struct MAPS TO A CLASS WITH PUBLIC MEMBER
VARIABLES
public class StructDemo {
public String attribute1;
public String attribute2;
public int attribute3;
}
enum
The enum construct allows for the creation of enumerated data types. For example, if you have an interface called “rainbow,” you might want to model each of its colors using a data type that can only take on a value of red, orange, yellow, green, blue, indigo, or violet. A CORBA enum maps to a Java class with public final static int member variables matching each element in the enumerated data type. Listing 22.13 contains the IDL for the rainbow example, and Listing 22.14 contains the Java code that maps to the color enum .
LISTING 22.13 ENUMERATED DATA TYPES ARE CREATED IN CORBA USING THE enum CONSTRUCT
enum Color {red, orange, yellow, green, blue, indigo, violet};
interface Rainbow {
attribute Color color1;
attribute Color color2;
attribute Color color3;
attribute Color color4;
attribute Color color5;
};
LISTING 22.14 THE MAPPING OF AN IDL enum TO A JAVA CLASS final public class Color {
final public static int _red = 0;
final public static int _orange = 1;
final public static int _yellow = 2;
final public static int _green = 3;
final public static int _blue = 4;
final public static int _indigo = 5;
final public static int _violet = 6;
final public static Color red = new Color(_red);
final public static Color orange = new Color(_orange);
final public static Color yellow = new Color(_yellow);
final public static Color green = new Color(_green);
final public static Color blue = new Color(_blue);
final public static Color indigo = new Color(_indigo);
final public static Color violet = new Color(_violet);
private int __value;
private Color(int value) { this.value = value;
}
public int value() { return __value;
}
public static Color from_int(int $value) { switch($value) {
case _red : return red;
case _orange : return orange;
case _yellow : return yellow;
case _green : return green;
case _blue : return blue;
case _indigo : return indigo;
case _violet : return violet;
default : throw new org.omg.CORBA.BAD_PARAM
("Enum out of range: [0.." ++ (7 -1) + "]: " ++
$value);
} }
public java.lang.String toString() { org.omg.CORBA.Any any =
org.omg.CORBA.ORB.init().create_any();
ColorHelper.insert(any, this);
return any.toString();
} }
union
The IDL union construct allows for a single entity to store any one of many different data types. When the union is declared, all possible data types are listed, and during usage, it may take on one and only one at a single point in time. During its life cycle, the active data type may change. An IDL union maps into a Java class with getter and setter methods corresponding to each available data type. Listing 22.15 contains the IDL declaration for a simple union, and Listing 22.16 contains the generated Java code.
LISTING 22.15 A SIMPLE UNION IN IDL
enum Color {red, orange, yellow, green, blue, indigo, violet};
union ColorUnion switch (Color) { case red:long rgbValue;
case orange:long rgbValue;
case yellow:long rgbValue;
case green:long rgbValue;
default:boolean bNoColor;
};
LISTING 22.16 THE GENERATED JAVA CODE ASSOCIATED WITH THE IDL IN LISTING 22.15
final public class ColorUnion {
private .Object _object;
private Color _disc;
public ColorUnion() { }
public Color discriminator() { return _disc;
}
public int rgbValue() {
if(_disc != (Color) Color.red &&
_disc != (Color) Color.orange &&
_disc != (Color) Color.yellow &&
_disc != (Color) Color.green &&
true) {
throw new org.omg.CORBA.BAD_OPERATION("rgbValue");
}
return ((.Integer) _object).intValue();
}
public boolean bNoColor() {
if(_disc == (Color) Color.red ||
_disc == (Color) Color.orange ||
_disc == (Color) Color.yellow ||
_disc == (Color) Color.green ||
false) {
throw new org.omg.CORBA.BAD_OPERATION("bNoColor");
}
return ((.Boolean) _object).booleanValue();
}
public void rgbValue(int value) { _disc = (Color) Color.red;
_object = new .Integer(value);
public void rgbValue(Color disc, int value) { _disc = disc;
_object = new .Integer(value);
}
public void bNoColor(Color disc, boolean value) { _disc = disc;
_object = new .Boolean(value);
}
public .String toString() { org.omg.CORBA.Any any =
ColorUnionHelper.insert(any, this);
return any.toString();
} }
sequence and array
The IDL sequence and array data types are used to specify collections of elements.
Both data types map to Java arrays. When declaring either a sequence or an array, you first declare it using the typedef keyword and then use it in context. Listing 22.17 demonstrates how each is used. Note that because the sequence and array data types map directly into Java arrays, no Java class is generated by the idl2java compiler for them.
LISTING 22.17 CREATING SEQUENTIAL COLLECTIONS OF INFORMATION USING
typedef string StringArray[1013];
interface LotsOfStrings {
attribute StringSequence StringCollection1;
attribute StringArray StringCollection2;
};
As Java developers, we’re accustomed to programming in an environment that allows for the use of exceptions. Just as in Java, an IDL exception is raised when some exceptional situation occurs during the invocation of a method. For example, a server might expose a login() method that returns a reference to a now-logged-in user. If that login were to fail, the method could raise an exception that indicates the failed login. An IDL
exception maps to a Java exception and is declared using the raises clause
LISTING 22.18 MODELING EXCEPTIONAL SITUATIONS USING THE exception
string reason;
};
interface User {
attribute string name;
attribute string password;
};
interface LoginServer {
User loginUser(in string name, in string password) raises (InvalidLoginException);
}