M 2 CH 6
M 2 CH 6
M 2 CH 6
Polimorfismo
Java Bsico
Copyright (c) 2004
J os M. Ordax
Copyright
Este documento puede ser distribuido solo bajo los
trminos y condiciones de la Licencia de
Documentacin de javaHispano v1.0 o posterior.
La ltima versin se encuentra en
http://www.javahispano.org/licencias/
2
Herencia
Mediante la herencia garantizbamos que todas las
subclases de una superclase concreta tienen todos
los mtodos que tiene dicha superclase.
Es decir, definimos una especie de interfaz (API)
para un grupo de clases relacionados mediante la
herencia.
Ejemplo:
Aqu estamos diciendo que cualquier
Animal puede hacer esas cuatro cosas
(incluyendo los parmetros y tipos de
retorno). Cualquier Animal significa
cualquier clase que en la jerarqua de
clases herede de Animal.
Polimorfismo
Significa que podemos referenciar un objeto de
una subclase mediante una referencia declarada
como una de sus superclases.
Es otro de los paradigmas de la Orientacin a
Objetos.
Consiste en que una vez se ha definido una
superclase para un grupo de subclases, cualquier
instancia de esas subclases puede ser usada en el
lugar de la superclase.
Object o =newString(Hola);
3
Ejemplo
anim
lobo
Objeto
Lobo
referencia
referencia
public class Test
{
public static void main(String[] args)
{
Lobo lobo =new Lobo();
Animal anim=lobo;
}
}
Polimorfismo
Por tanto mediante el polimorfismo podemos
asignar a una referencia de un tipo superior en la
jerarqua de herencia, una instancia de un tipo
inferior (que herede).
Algunos usos habituales del polimorfismo en J ava
son:
Implementacin de colecciones genricas.
Implementacin de mtodos genricos.
Ahora bien, que la referencia sea de otro tipo no
significa que los mtodos que se ejecuten sean
otros. Siguen siendo los de la instancia.
4
Polimorfismo
Ejemplo de coleccin genrica:
public class TestPolimorfismo
{
public static void main(String[] args)
{
Animal[] animales =newAnimal[4];
animales[0] =new Lobo();
animales[1] =new Perro();
animales[2] =new Leon();
animales[3] =new Tigre();
for(int i=0; i<animales.length; i++)
{
animales[i].dormir();
animales[i].comer();
}
}
}
Polimorfismo
Ejemplo de mtodo genrico:
public class Matematico
{
public double calcularArea(Figura param)
{
return param.calcularArea();
}
}
public class TestPolimorfismo
{
public static void main(String[] args)
{
Matematico m =new Matematico();
m.calcularArea(new Circulo(1.2,2.4,13.0));
m.calcularArea(new Triangulo(1.1,1.1,2.3,2.3,4.1,4.1));
}
}
5
Polimorfismo
Con el polimorfismo podemos desarrollar cdigo
que no tiene que ser modificado por la introduccin
en el programa de nuevas subclases o tipos
debido a:
Cambio en las especificaciones.
Rediseo.
Ejemplo: la clase Matematico seguir funcionando
aunque desarrollemos nuevas figuras como
Cuadrado, Ameba, etc. siempre y cuando
hereden de la superclase Figura.
Ejemplo
Supongamos que necesitamos
implementar una clase para
almacenar dos Lobos en
nuestro proyecto.
public class MiLista
{
Lobo l1 =null;
Lobo l2 =null;
public boolean add(Lobo param)
{
if(l1 ==null)
{
l1 =param;
return true;
}
else if(l2 ==null)
{
l2 =param;
return true;
}
else
return false;
}
}
6
Ejemplo
Opps! Ahora nos dicen que
en el mismo proyecto tambin
necesitamos almacenar dos
Gatos.
public class MiLista
{
Animal l1 =null;
Animal l2 =null;
public boolean add(Animal param)
{
if(l1 ==null)
{
l1 =param;
return true;
}
else if(l2 ==null)
{
l2 =param;
return true;
}
else
return false;
}
}
Tenemos distintas alternativas:
Crear una clase nueva MiLista2.
Aadir a MiLista dos atributos nuevos
del tipo Gato y otro mtodo add() que
reciba un Gato.
Modificar MiLista para que maneje el
tipo genrico Animal y as nos valga
tanto para Lobos como para Gatos
e incluso otros animales en el futuro.
Ejemplo
Hablando con un colega de
otro proyecto en la mquina
de caf, nos comenta que en
su proyecto necesita
implementar una clase para
almacenar dos Triangulos.
public class MiLista
{
Object l1 =null;
Object l2 =null;
public boolean add(Object param)
{
if(l1 ==null)
{
l1 =param;
return true;
}
else if(l2 ==null)
{
l2 =param;
return true;
}
else
return false;
}
}
Le podramos pasar nuestra
clase si la hubiramos hecho
ms genrica.
No heredaba en J ava todo
de la clase Object?
En J ava encontraremos multitud de
ejemplos que usen el Polimorfismo
con este fin.
7
Castings
Hay dos clases de casting:
El casting es una forma de realizar conversiones
de tipos.
UpCasting: conversin de un tipo en otro superior en la
jerarqua de clases. No hace falta especificarlo.
DownCasting: conversin de un tipo en otro inferior en la
jerarqua de clases.
Se especifica precediendo al objeto a convertir con
el nuevo tipo entre parntesis.
Ejemplo public class Test
{
public static void main (String[] args)
{
Lobo lobo =new Lobo();
// UpCastings
Canino canino =lobo;
Object animal =new Lobo();
animal.comer();
// DownCastings
lobo =(Lobo)animal;
lobo.comer();
Lobo otroLobo =(Lobo)canino;
Lobo error =(Lobo) new Canino();
}
No compila. No puedes llamar al mtodo comer() sobre
un Object. No puedes convertir un Canino en un Lobo.
8
Clases abstractas
Estas clases pueden estar siendo usadas
simplemente para agrupar bajo un mismo tipo a
otras clases, o para contener cdigo reutilizable,
o para forzar un API a sus subclases..
A menudo existen clases que sirven para definir
un tipo genrico pero que no tiene sentido
instanciar (crear objetos de ella).
Puede tener sentido instanciar un Circulo pero a lo mejor
no instanciar una Figura, porqu qu figura es? cul
es su rea? y su permetro?
Por ejemplo:
Clases abstractas
Declaracin de una clase abstracta:
Ejemplo:
modificador_acceso abstract class nom_clase
{
}
public abstract class MiClase
{
}
La clases se definen como abstractas mediante la
keyword: abstract.
9
Ejemplo
public abstract class Animal
{
..
}
public abstract class Canino extends Animal
{
.
}
public class Perro extends Canino
{
.
}
public class Test
{
public static void main(String[] args)
{
Canino c;
c =new Canino();
c.rugir();
}
}
No compila. Canino es una clase abstracta.
Mtodos abstractos
Una clase con uno o varios mtodos abstractos
tiene que ser declarada abstracta.
Adems de clases abstractas, tambin podemos
tener mtodos abstractos.
Una clase abstracta significaba que tena que ser
heredada. No poda ser instanciada.
Un mtodo abstracto significa que tiene que ser
sobrescrito. No est implementado.
No obstante una clase abstracta no tiene porque
tener mtodos abstractos.
10
Mtodos abstractos
Declaracin de un mtodo abstracto:
Ejemplo:
modif_acceso abstract tipo_retorno nombre([tipo param,]);
public abstract void miMetodo();
Los mtodos se definen como abstractos mediante
la keyword: abstract.
El objetivo de un mtodo abstracto es forzar una
interfaz (API) pero no una implementacin.
Ejemplo
public abstract class Figura
{
public abstract double calcularArea();
public abstract double calcularPerimetro();
}
public class Circulo extends Figura
{
private Punto centro =null;
private double radio =0.0;
public double calcularArea()
{
return Math.PI*radio*radio;
}
public double calcularPerimetro()
{
return 2*Math.PI*radio;
}
}
11
Mas Polimorfismo
Diseo del aplicativo SimAnimal 2004.
Mas Polimorfismo
Qu ocurre si queremos reusar el diseo para
un aplicativo de Tienda de Mascotas?
Una primera aproximacin sera aadir a la
clase Animal todos los mtodos especficos de
una mascota.
Automticamente todas las mascotas tendrn los
mtodos necesarios.
Pero tambin los tendrn las no mascotas.
Y seguro que hay que retocar cada mascota
reescribiendo sus mtodos porque tengan alguna
peculiaridad.
12
Mas Polimorfismo
Mas Polimorfismo
Modificamos la primera aproximacin definiendo
los mtodos de las mascotas en la clase Animal
como abstractos de manera que cada mascota los
implemente.
Pero no solo el resto de animales heredarn tambin el
interfaz si no que tienen que implementarlo aunque sea
vaco.
As todas las mascotas heredan el interfaz e implementan
su comportamiento dependiendo de la mascota en
concreto.
13
Mas Polimorfismo
Mas Polimorfismo
Otra aproximacin sera introducir los nuevos
mtodos solo en las mascotas.
Sin embargo esto implica otro tipo de problemas como
que los programadores de mascotas tendrn que ponerse
de acuerdo en el interfaz de estas y siempre llevarlo a
raja tabla puesto que ahora no se hereda y el copilador no
nos ayuda con los posibles errores.
As ya no nos tenemos que preocupar de que haya clases
que sin ser mascotas tengan mtodos de estas.
Otro inconveniente muy importante es que no tenemos
posibilidad de usar el polimorfismo con las mascotas.
14
Mas Polimorfismo
Mas Polimorfismo
La solucin que parece ptima, sera tener otra
clase abstracta llamada Mascota con los mtodos
abstractos de las mascotas. Y que todas las
mascotas heredasen de ella.
Todas las mascotas cumplirn forzosamente el API de las
mascotas y el compilador nos ayudar a asegurarlo.
As, ya no nos tenemos que preocupar de que haya
clases que sin ser mascotas tengan mtodos de estas.
Y tambin tendremos la posibilidad de usar el
polimorfismo con las mascotas.
Pero eso significa que habr clases que heredarn de dos
clases a la vez y en J ava no existe la herencia mltiple.
15
Mas Polimorfismo
Interfaces
Los interfaces en J ava nos solucionan en parte la
no existencia de la herencia mltiple; habilitando
as las posibilidades del polimorfismo en la herencia
mltiple sin los problemas que esta conlleva.
Los interfaces son un tipo de clase especial que
no implementa ninguno de sus mtodos. Todos
son abstractos. Por tanto no se pueden instanciar.
La declaracin de un interface J ava se realiza
mediante la keyword: interface seguido de su
nombre.
16
Interfaces
Declaracin de un interface:
Ejemplo:
modificador_acceso interface nombre_interface
{
}
public interface MiInterface
{
}
Siguen siendo clases J ava por lo que su cdigo
fuente se guarda en un fichero texto de extensin
*.java y al compilarlo se generar un *.class
Interfaces
Declaracin de un mtodo abstracto:
Ejemplo:
modif_acceso abstract tipo_retorno nombre([tipo param,]);
public abstract void miMetodo();
Los mtodos se definen como abstractos mediante
la keyword: abstract.
El objetivo de un mtodo abstracto es forzar una
interfaz (API) pero no una implementacin.
17
Interfaces
Declaracin de la herencia:
Ejemplo:
modif_acceso class nom_clase implements nom_interface[,nom_int.]
{
}
public class MiClase implements MiInterface
{
}
De los interfaces tambin se hereda, aunque se
Suele decir implementa. Y se realiza mediante la
keyword: implements.
Interfaces
Una clase puede heredar de mltiples interfaces.
Un interface puede tambin definir constantes.
Si una clase que hereda de un interface, no
implementa todos los mtodos de este, deber ser
definida como abstracta.
Una clase puede heredar de otra clase y a la vez
heredar de mltiples interfaces.
18
Ejemplo
public interface Mascota
{
public abstract void jugar();
public abstract void vacunar();
}
public class Perro extends Canino implements Mascota
{
public void comer() { }
public void hacerRuido() { }
public void rugir() { }
public void jugar() { }
public void vacunar() { }
}
Por tanto, cuando hablamos de polimorfismo,
significa que una instancia de una clase puede ser
referenciada por un tipo interface siempre y cuando
esa clase o una de sus superclases implemente
dicho interface.
Interfaces
Un interface se trata como un tipo cualquiera.
Un interface puede heredar de otros interfaces.
19
Mas Polimorfismo
Diseo final del aplicativo
de gestin de una Tienda
de Mascotas, reutilizando
el aplicativo SimAnimal 2004.
Interface vs. Clase Abstracta
Una clase puede implementar n interfaces pero
solo una clase.
Un interface no forma parte de la jerarqua de
clases. Clases dispares pueden implementar el
mismo interface.
Un interface no puede implementar ningn
mtodo.
El objetivo de un mtodo abstracto es forzar una
interfaz (API) pero no una implementacin.
20
Haremos una subclase cuando necesitemos hacer
una especializacin de la superclase mediante
sobreescritura o aadiendo nuevos mtodos.
Clases, Subclases, Abstractas
e Interfaces
Haremos una clase que no herede de nadie
cuando la clase no pase la prueba de Es-Un.
Haremos una clase abstracta cuando queramos
definir un grupo genrico de clases y adems
tengamos algunos mtodos implementados que
reutilizar. Tambin cuando no queramos que nadie
instancie dicha clase.
Clases, Subclases, Abstractas
e Interfaces
Haremos un interface cuando queramos definir un
grupo genrico de clases y no tengamos mtodos
implementados que reutilizar. O cuando nos
veamos forzados por la falta de herencia mltiple
en J ava.
21
Bibliografa
Head First J ava
Kathy Sierra y Bert Bates.
OReilly
Learning J ava (2
nd
edition)
Patrick Niemeyer y J onathan Knudsen.
OReilly.
Thinking in J ava (3
rd
edition)
Bruce Eckel.
Prentice Hall.
The J ava tutorial
http://java.sun.com/docs/books/tutorial/