Aug 26, 2011

Interfaces

          In Java, this multiple inheritance problem is solved with a powerful construct called interfaces. Interface can be used to define a generic template and then one or more abstract classes to define partial implementations of the interface. Interfaces just specify the method declaration (implicitly public and abstract) and can only contain fields (which are implicitly public static final). Interface definition begins with a keyword interface. An interface like that of an abstract class cannot be instantiated.
Multiple Inheritance is allowed when extending interfaces i.e. one interface can extend none, one or more interfaces. Java does not support multiple inheritance, but it allows you to extend one class and implement many interfaces.
If a class that implements an interface does not define all the methods of the interface, then it must be declared abstract and the method definitions must be provided by the subclass that extends the abstract class.

Below is an example of a Shape interface

                     interface Shape {

                     public double area();
                     public double volume();
                     }

Below is a Point class that implements the Shape interface.

                     public class Point implements Shape {

                     static int x, y;
                     public Point() {
                     x = 0;
                     y = 0;
                     }
                     public double area() {
                     return 0;
                     }
                     public double volume() {
                     return 0;
                     }
                     public static void print() {
                     System.out.println("point: " + x + "," + y);
                     }
                     public static void main(String args[]) {
                     Point p = new Point();
                     p.print();
                     }
                   }

Similarly, other shape objects can be created by interface programming by implementing generic Shape Interface.
Example:
Below is a java interfaces program showing the power of interface programming in java.Listing below shows 2 interfaces and 4 classes one being an abstract class.
Note: The method toString in class A1 is an overridden version of the method defined in the class named Object. The classes B1 and C1 satisfy the interface contract. But since the class D1 does not define all the methods of the implemented interface I2, the class D1 is declared abstract.
Also,
i1.methodI2() produces a compilation error as the method is not declared in I1 or any of its super interfaces if present. Hence a downcast of interface reference I1 solves the problem as shown in the program. The same problem applies to i1.methodA1(), which is again resolved by a downcast.
When we invoke the toString() method which is a method of an Object, there does not seem to be any problem as every interface or class extends Object and any class can override the default toString() to suit your application needs. ((C1)o1).methodI1() compiles successfully, but produces a ClassCastException at runtime. This is because B1 does not have any relationship with C1 except they are “siblings”. You can’t cast siblings into one another.
When a given interface method is invoked on a given reference, the behavior that results will be appropriate to the class from which that particular object was instantiated. This is runtime polymorphism based on interfaces and overridden methods.

                        interface I1 {

                        void methodI1(); // public static by default
                        }

                        interface I2 extends I1 {

                        void methodI2(); // public static by default
                        }

                        class A1 {

                         public String methodA1() {
                         String strA1 = "I am in methodC1 of class A1";
                         return strA1;
                         }
                         public String toString() {
                         return "toString() method of class A1";
                         }
                       }

                         class B1 extends A1 implements I2 {

                          public void methodI1() {
                          System.out.println("I am in methodI1 of class B1");
                          }
                          public void methodI2() {
                          System.out.println("I am in methodI2 of class B1");
                          }
                       }

                         class C1 implements I2 {

                         public void methodI1() {
                         System.out.println("I am in methodI1 of class C1");
                         }
                         public void methodI2() {
                         System.out.println("I am in methodI2 of class C1");
                         }
                      }

                         // Note that the class is declared as abstract as it does not
                        // satisfy the interface contract
                       abstract class D1 implements I2 {

                       public void methodI1() {
                       }
                      // This class does not implement methodI2() hence declared abstract.
                    }

                    public class InterFaceEx {

                     public static void main(String[] args) {
                     I1 i1 = new B1();
                     i1.methodI1(); // OK as methodI1 is present in B1
                   // i1.methodI2(); Compilation error as methodI2 not present in I1
                   // Casting to convert the type of the reference from type I1 to type I2
                     ((I2) i1).methodI2();
                     I2 i2 = new B1();
                     i2.methodI1(); // OK
                     i2.methodI2(); // OK
                   // Does not Compile as methodA1() not present in interface reference I1
                   // String var = i1.methodA1();
                   // Hence I1 requires a cast to invoke methodA1
                      String var2 = ((A1) i1).methodA1();
                      System.out.println("var2 : " + var2);
                      String var3 = ((B1) i1).methodA1();
                      System.out.println("var3 : " + var3);
                      String var4 = i1.toString();
                      System.out.println("var4 : " + var4);
                      String var5 = i2.toString();
                      System.out.println("var5 : " + var5);
                      I1 i3 = new C1();
                      String var6 = i3.toString();
                      System.out.println("var6 : " + var6); // It prints the Object toString() method
                      Object o1 = new B1();
                      // o1.methodI1(); does not compile as Object class does not define
                      // methodI1()
                      // To solve the probelm we need to downcast o1 reference. We can do it
                      // in the following 4 ways
                       ((I1) o1).methodI1(); // 1
                        ((I2) o1).methodI1(); // 2
                        ((B1) o1).methodI1(); // 3
                       /*
                        *
                        * B1 does not have any relationship with C1 except they are "siblings".
                        *
                        * Well, you can't cast siblings into one another.
                        *
                        */
                        // ((C1)o1).methodI1(); Produces a ClassCastException
                     }
                  }
 
Output

I am in methodI1 of class B1
I am in methodI2 of class B1
I am in methodI1 of class B1
I am in methodI2 of class B1
var2 : I am in methodC1 of class A1
var3 : I am in methodC1 of class A1
var4 : toString() method of class A1
var5 : toString() method of class A1
var6 : C1@190d11
I am in methodI1 of class B1
I am in methodI1 of class B1
I am in methodI1 of class B1
 

0 comments :

Post a Comment