|
|
|
1000 Java Tips ebook
|
|
 

Free "1000 Java Tips" eBook is here! It is huge collection of big and small Java
programming articles and tips. Please take your copy here.
Take your copy of free "Java Technology Screensaver"!. |
|
Easy Learn Java: Programming Articles, Examples and Tips - Page 300
Previous
1060 Stories (530 Pages, 2 Per Page)
Next
The Java Lesson 24: abstract classes and polymorphism
|
abstract
classes and polymorphism
Introduction
Some classes are designed to be extended but NOT
instantiated.
Consider, for example, a dog, a cat, a whale, and
a human. Each is a mammal. In object-oriented design, Mammal would be the superclass and Dog, Cat, Whale, and Human would be classes that each extend Mammal. But, since there is no such thing as a
generic mammal, there is no reason to instantiate a Mammal object.
The Mammal class is an example of a class that should be declared
abstract.
An abstract class
public abstract class Mammal
which indicates that an object of the class can
never be instantiated. The class can, however, be extended by a subclass that
would inherit all its features (to be covered shortly). For example,
public class Cat extends Mammal
would be the header of a Cat class that extends the
Mammal class.
-
Can have features. Even though the class is
abstract it can have
instance variables, one or more constructor methods, "set" and "get" methods,
and special purpose methods that will be inherited by its descendent classes.
-
Can optionally have one or more abstract methods. An abstract method is defined with
the abstract keyword in
the method header and has no body. For example,
public abstract double someMethod(double parm1, int
parm2);
declares an abstract method named someMethod that requires two parameters (a double and an int) and returns a double value to the caller.
The purpose of an abstract method is to force certain features to
exist within a subclass. An extending class must either override the abstract method to define the
method body or declare itself to be abstract. If it does not override the abstract method or declare itself to be abstract, a compile error will
occur.
public class App { public static void main(String[] args)
{ Cat myCat = new Cat("Fluffy",
10); System.out.println(myCat.toString());
} }
abstract class Mammal {
private String name; public Mammal(String iName)
{ setName(iName); } public boolean
setName(String nName) { if (nName.length() > 0)
{ name =
nName; return true;
} else { name =
"unknown"; return
false; } } public String getName()
{ return name; } public abstract boolean
setAge(int nAge); public abstract int getAge(); public
abstract String toString(); }
class Cat extends Mammal { private
int age; public static final int MAX_LIFE = 25; public
Cat(String iName, int iAge) {
super(iName); setAge(iAge);
} public
boolean setAge(int nAge) { if (nAge >= 0 &&
nAge <= MAX_LIFE) { age =
nAge; return true;
} else return
false; } public int getAge() {
return age; } public String toString()
{ return getName() + " (cat), " + age;
} }
Notes:
-
The Mammal class encapsulates a mammal's name and supports the
"setting" and "getting" of its name. The first two abstract methods will force
subclasses to implement methods to "set" and "get" the mammal's age. The
third abstract method
will force subclasses to implement a meaningful toString() method.
-
The Cat class subclasses Mammal to encapsulate information about a cat's age. It
defines the setAge()
and getAge() methods to store and
retrieve the age of the cat. It also defines the toString() method to return a meaningful String of information about
the cat.
Polymorphism
- Webster's Collegiate Dictionary
-
Is implemented by the fact that an object
reference can reference an object of its own class or any descendent class. In
other words, if Cat
descends from Mammal (as
defined above), the following code is perfectly valid:
Mammal x = new Cat("Fluffy",
10); System.out.println(x.toString());
This works because the object being constructed
is a Cat and a Cat is a Mammal. Regardless of how we
reference the object, however, it will always remain a Cat and have the features of a
Cat.
Because an object reference can reference
different kinds of objects, we can write some powerful code. For example,
assume that Cat, Mouse, Sheep, and Wolf are subclasses of Mammal with each having its own implementation of the toString() method. If someMammal is a Mammal object reference that
currently references either a Cat, a Mouse, a Sheep, or a Wolf, the expression
someMammal.toString()
will call the toString() method of the correct class. The object
reference, someMammal, is
polymorphic because it has the ability to take on different forms. It will
"act like" an object of the class it currently references.
public class App { public static void main(String[] args)
{
// Declare an array of superclass object
references.
Mammal[] pets = new Mammal[5];
// Load the
array with objects that descend from the
superclass.
pets[0] = new Cat("Fluffy",
10); pets[1] = new Mouse("Speedy", 1); pets[2] =
new
Mouse("Mickey",
2); pets[3] = new Cat("Tiger", 3); pets[4] =
new
Cat("Garfield",
15);
// Use polymorphism to display the contents
of the array.
Utility.skip(); for (int i = 0; i < pets.length;
i++) { System.out.println(" " +
pets[i].toString()); } } }
abstract class
Mammal { private String name; public Mammal(String
iName) { setName(iName); } public
boolean setName(String nName) { if (nName.length()
> 0) { name =
nName; return true;
} else { name =
"unknown"; return
false; } } public String getName()
{ return name; } public abstract
boolean setAge(int nAge); public abstract int getAge();
public abstract String toString(); }
class Cat extends Mammal
{ private int age; public static final int MAX_LIFE =
25; public Cat(String iName, int iAge) {
super(iName); setAge(iAge); }
public boolean setAge(int nAge) { if (nAge >= 0
&& nAge <= MAX_LIFE) { age =
nAge; return true;
} else return
false; } public int getAge() {
return age; } public String toString()
{ return getName() + " (cat), " + age;
} }
class Mouse extends Mammal { private int
age; public static final int MAX_LIFE = 5; public
Mouse(String iName, int iAge) {
super(iName); setAge(iAge); }
public boolean setAge(int nAge) { if (nAge >= 0
&& nAge <= MAX_LIFE) { age =
nAge; return true;
} else return
false; } public int getAge() {
return age; } public String toString()
{ return this.getName() + " (mouse), " +
age; } }
More information
Java ranch has an excellent
"campfire story" relating to this lesson. Be sure to read How my Dog learned
Polymorphism.
Looking ahead
When using polymorphism, it is
sometimes necessary to determine the class of an object being referenced. It is
also sometimes necessary to convert (cast) one object as another. These topics,
and more, will be covered in the next lesson.
Lab exercise for Ferris
students
E-mail your answers to this
assignment no later than
the due date listed in the class schedule.
Review questions
-
True or False: An abstract class must contain at least one
abstract method.
-
True
-
False
-
True or False: An abstract class can have one or more
constructors, instance variables, and instance methods.
-
True
-
False
-
What will happen when an
attempt is made to compile and execute the following program? Line numbers are
for reference purposes only.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
public
class App { public static void main(String[] args)
{ Base x = new Sub();
System.out.println(x.toString());
x.hello(); } } abstract class Base { public
String toString() { return "I am a
Base"; } public abstract void
hello(); } class Sub extends Base { public String
toString() { return "I am a Sub";
} } |
-
A compile error will occur
at line 3
-
A compile error will occur
at line 14
-
Compiles successfully but
a runtime error will occur at line 5
-
Compiles and runs to
display: "I am a Base"
-
Compiles and runs to
display: "I am a Sub"
-
What will happen when an
attempt is made to compile and execute the following program? Line numbers are
for reference purposes only.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |
public
class App { public static void main(String[] args)
{ Base[] subs = new
Base[2]; subs[0] = new
Sub(); subs[1] = new
SubSub(); subs[1].hello();
} } abstract class Base { public abstract void
hello(); } class Sub extends Base { public void
hello() { System.out.println("I am a
Sub"); } } class SubSub extends Sub { public
void hello() { System.out.println("I am a
SubSub"); } } |
-
A compile error will occur
at line 3
-
A compile error will occur
at line 4
-
Compiles successfully but
a runtime error will occur at line 5
-
Compiles and runs to
display: "I am a Sub"
-
Compiles and runs to
display: "I am a SubSub"
13236 bytes more | 1 comment | | Score: 4
|
Posted by jalex on Thursday, February 03, 2005 (00:00:00) (14122 reads)
|
Java Virtual Machine
|
Java: Java Virtual Machine After you read this section, you should be able to understand how it's possible for a Java applet/application to run on many different machines. The key to Java's portability and security is the Java Virtual Machine. The Java Virtual Machine Most compilers translate from the source language (eg, C or Pascal) into machine language for one specific type of machine (eg, Pentium or Sun). The machine language for a Sun computer is not the same as for a Macintosh, or a Windows machine. You cannot use the same machine code program on different kinds of machines. |  Most compilers produce code for a real machine |  The Java compiler produces code for a virtual machine. | Java compilers translate applets in two steps. The first part of the translation is from Java source to Java Virtual Machine (JVM) machine code. This may seem strange because there isn't any real machine that can use JVM instructions yet! JVM code is designed so that it is easy to translate it into machine instructions for real machines, so the second part of the translation to real machine instructions is done in the browser on the user's machine. The browser takes the applet JVM code that it gets from the server and translates from JVM code to the machine code the browser is using, eg, a Pentium. This compiler in the browser is often called a "JIT compiler". JIT stands for "Just In Time", and it means that the last part of the compilation is done just before running the program. The diagram below shows how the same applet JVM code is used by browsers on different machines.

Click on the picture to see full size image. Two step compilation allows Java applets to run on many kinds of machines. Other languages can use the JVM See www.robert-tolksdorf.de/vmlanguages.html for a list of languages that generate code for the Java Virtual Machine. Java portability So applets are really JVM code that is translated into the correct machine code by the browser. That's why applets can run on any machine, and C and C++ programs cannot. This same two step translation can be used for Java applications as well as applets. If a program is in JVM code, then it is portable, and can be run on any machine that has a JIT compiler on it. C and C++ object programs are not portable because they are always translated into the machine language for a specific machine. If you are very careful when you write your C programs, you may be able to compile and run the source program on many machines, but there are often big problems. For example, how big is a C int? On some machines it is 16 bits, on some machines it is 32 bits, and some it is 64 bits. Java has defined the basic types very exactly (eg, ints are always 32 bits). Another big problem is that lack of standard library functions in C/C++. There are some standard library functions for C/C++, but they are very small compared to Java, and don't do many of the things that programmers need to do (eg build a graphical user interface). Portability is extremely important to many companies, since large companies usually have many kinds of computers. If a company can "write once and run everywhere" they can save an enormous amount of money. If a developer can do this, they will have a much larger market for their software.
15 comments | | Score: 0
|
Posted by jalex on Thursday, February 03, 2005 (00:00:00) (3884 reads)
|
|
|