Monday, 30 January 2012

Java Object Typecasting


Object Reference Type Casting

In java object typecasting one object reference can be type cast into another object reference. The cast can be to its own class type or to one of its subclass or superclass types or interfaces. There are compile-time rules and runtime rules for casting in java.

How to Typecast Objects with a dynamically loaded Class ? – The casting of object references depends on the relationship of the classes involved in the same hierarchy. Any object reference can be assigned to a reference variable of the type Object, because the Object class is a superclass of every Java class.
There can be 2 casting java scenarios

· Upcasting
· Downcasting

When we cast a reference along the class hierarchy in a direction from the root class towards the children or subclasses, it is a downcast. When we cast a reference along the class hierarchy in a direction from the sub classes towards the root, it is an upcast. We need not use a cast operator in this case.

The compile-time rules are there to catch attempted casts in cases that are simply not possible. This happens when we try to attempt casts on objects that are totally unrelated (that is not subclass super class relationship or a class-interface relationship) At runtime a ClassCastException is thrown if the object being cast is not compatible with the new type it is being cast to.

Below is an example showing when a ClassCastException can occur during object casting

//X is a supper class of Y and Z which are sibblings.
public class RunTimeCastDemo {

          public static void main(String args[]) {
                   X x = new X();
                   Y y = new Y();
                   Z z = new Z();
                   X xy = new Y(); // compiles ok (up the hierarchy)
                   X xz = new Z(); // compiles ok (up the hierarchy)
                   //                 Y yz = new Z();   incompatible type (siblings)
                   //                 Y y1 = new X();   X is not a Y
                   //                 Z z1 = new X();   X is not a Z
                   X x1 = y; // compiles ok (y is subclass of X)
                   X x2 = z; // compiles ok (z is subclass of X)
                   Y y1 = (Y) x; // compiles ok but produces runtime error
                   Z z1 = (Z) x; // compiles ok but produces runtime error
                   Y y2 = (Y) x1; // compiles and runs ok (x1 is type Y)
                   Z z2 = (Z) x2; // compiles and runs ok (x2 is type Z)
                   //                 Y y3 = (Y) z;     inconvertible types (siblings)
                   //                 Z z3 = (Z) y;     inconvertible types (siblings)
                   Object o = z;
                   Object o1 = (Y) o; // compiles ok but produces runtime error
          }
}

Casting Object References: Implicit Casting using a Compiler

In general an implicit cast is done when an Object reference is assigned (cast) to:

* A reference variable whose type is the same as the class from which the object was instantiated.
An Object as Object is a super class of every Class.
* A reference variable whose type is a super class of the class from which the object was instantiated.
* A reference variable whose type is an interface that is implemented by the class from which the object was instantiated.
* A reference variable whose type is an interface that is implemented by a super class of the class from which the object was instantiated.

Consider an interface Vehicle, a super class Car and its subclass Ford. The following example shows the automatic conversion of object references handled by the compiler

interface Vehicle {
}
class Car implements Vehicle {
}
class Ford extends Car {
}

Let c be a variable of type Car class and f be of class Ford and v be an vehicle interface reference. We can assign the Ford reference to the Car variable:
I.e. we can do the following

Example 1
c = f; //Ok Compiles fine

Where c = new Car();
And, f = new Ford();
The compiler automatically handles the conversion (assignment) since the types are compatible (sub class – super class relationship), i.e., the type Car can hold the type Ford since a Ford is a Car.

Example 2
v = c; //Ok Compiles fine
c = v; // illegal conversion from interface type to class type results in compilation error

Where c = new Car();
And v is a Vehicle interface reference (Vehicle v)

The compiler automatically handles the conversion (assignment) since the types are compatible (class – interface relationship), i.e., the type Car can be cast to Vehicle interface type since Car implements Vehicle Interface. (Car is a Vehicle).

Casting Object References: Explicit Casting

Sometimes we do an explicit cast in java when implicit casts don’t work or are not helpful for a particular scenario. The explicit cast is nothing but the name of the new “type” inside a pair of matched parentheses. As before, we consider the same Car and Ford Class

class Car {
void carMethod(){
}
}
class Ford extends Car {
void fordMethod () {
}
}

We also have a breakingSystem() function which takes Car reference (Superclass reference) as an input parameter.
The method will invoke carMethod() regardless of the type of object (Car or Ford Reference) and if it is a Ford object, it will also invoke fordMethod(). We use the instanceof operator to determine the type of object at run time.

public void breakingSystem (Car obj) {
obj.carMethod();
if (obj instanceof Ford)
((Ford)obj).fordMethod ();
}

To invoke the fordMethod(), the operation (Ford)obj tells the compiler to treat the Car object referenced by obj as if it is a Ford object. Without the cast, the compiler will give an error message indicating that fordMethod() cannot be found in the Car definition.


The following program shown illustrates the use of the cast operator with references.

Note: Classes Honda and Ford are Siblings in the class Hierarchy. Both these classes are subclasses of Class Car. Both Car and HeavyVehicle Class extend Object Class. Any class that does not explicitly extend some other class will automatically extends the Object by default. This code instantiates an object of the class Ford and assigns the object’s reference to a reference variable of type Car. This assignment is allowed as Car is a superclass of Ford. In order to use a reference of a class type to invoke a method, the method must be defined at or above that class in the class hierarchy. Hence an object of Class Car cannot invoke a method present in Class Ford, since the method fordMethod is not present in Class Car or any of its superclasses. Hence this problem can be colved by a simple downcast by casting the Car object reference to the Ford Class Object reference as done in the program. Also an attempt to cast an object reference to its Sibling Object reference produces a ClassCastException at runtime, although compilation happens without any error.


class Car extends Object {

          void carMethod() {
          }
}

class HeavyVehicle extends Object {
}

class Ford extends Car {

          void fordMethod() {
                   System.out.println("I am fordMethod defined in Class Ford");
          }
}

class Honda extends Car {

          void fordMethod() {
                   System.out.println("I am fordMethod defined in Class Ford");
          }
}

public class ObjectCastingEx {

          public static void main(String[] args) {
                   Car obj = new Ford();
                   //    Following will result in compilation error
                   //    obj.fordMethod();        //As the method fordMethod is undefined for the Car Type
                   //  Following will result in compilation error
                   // ((HeavyVehicle)obj).fordMethod();
                                                //fordMethod is undefined in the HeavyVehicle Type
                   //  Following will result in compilation error
                   ((Ford) obj).fordMethod();
                   //Following will compile and run
                   //       Honda hondaObj = (Ford)obj;        Cannot convert as they are sibblings
          }
}

One common casting that is performed when dealing with collections is, you can cast an object reference into a String.

import java.util.Vector;

public class StringCastDemo {

          public static void main(String args[]) {
                   String username = "asdf";
                   String password = "qwer";
                   Vector v = new Vector();
                   v.add(username);
                   v.add(password);
                   //               String u = v.elementAt(0); Cannot convert from object to String
                   Object u = v.elementAt(0); //Cast not done
                   System.out.println("Username : " + u);
                   String uname = (String) v.elementAt(0); // cast allowed
                   String pass = (String) v.elementAt(1); // cast allowed
                   System.out.println();
                   System.out.println("Username : " + uname);
                   System.out.println("Password : " + pass);
          }
}

Output

Username : asdf
Username : asdf
Password : qwer

instanceof Operator

The instanceof operator is called the type comparison operator, lets you determine if an object belongs to a specific class, or implements a specific interface. It returns true if an object is an instance of the class or if the object implements the interface, otherwise it returns false.

Below is an example showing the use of instanceof operator

class Vehicle {

          String name;
          Vehicle() {
                   name = "Vehicle";
          }
}

class HeavyVehicle extends Vehicle {

          HeavyVehicle() {
                   name = "HeavyVehicle";
          }
}

class Truck extends HeavyVehicle {

          Truck() {
                   name = "Truck";
          }
}

class LightVehicle extends Vehicle {

          LightVehicle() {
                   name = "LightVehicle";
          }
}

public class InstanceOfExample {

          static boolean result;
          static HeavyVehicle hV = new HeavyVehicle();
          static Truck T = new Truck();
          static HeavyVehicle hv2 = null;
          public static void main(String[] args) {
                   result = hV instanceof HeavyVehicle;
                   System.out.print("hV is an HeavyVehicle: " + result + "\n");
                   result = T instanceof HeavyVehicle;
                   System.out.print("T is an HeavyVehicle: " + result + "\n");
                   result = hV instanceof Truck;
                   System.out.print("hV is a Truck: " + result + "\n");
                   result = hv2 instanceof HeavyVehicle;
                   System.out.print("hv2 is an HeavyVehicle: " + result + "\n");
                   hV = T; //Sucessful Cast form child to parent
                   T = (Truck) hV; //Sucessful Explicit Cast form parent to child
          }
}
Download instanceof operator Source Code

Output

hV is an HeavyVehicle: true
T is an HeavyVehicle: true
hV is a Truck: false
hv2 is an HeavyVehicle: false

Note: hv2 does not yet reference an HeavyVehicle object, instanceof returns false. Also we can’t use instanceof operator with siblings

No comments:

Post a Comment