Programación Orientada A Objetos

Descargar como pdf o txt
Descargar como pdf o txt
Está en la página 1de 124

PROGRAMACIÓN

ORIENTADA A OBJETOS
Métodos, Interfaces.
2

Métodos
• La razón de establecer los atributos en vista privada, es
poder controlar la modificación del estado del objeto y no
permitir que este evolucione hacía estados incoherentes.
• Pero debemos establecer los mecanismos para poder
modificar el objeto, y son los métodos, el sistema
adecuado.
3

Métodos
Podemos organizar los métodos en tres tipos, teniendo en
cuenta aspectos sintácticos y semánticos:
• Constructores. Métodos que inicializan la instancia
• Métodos genéricos. Realizan acciones utilizando los
atributos
• Métodos para accesos directo a los atributos (getters &
setters). Estos métodos son opcionales y suelen utilizarse
más en clases de tipo “entidad”, es decir, que tienen
persistencia en bases de datos, o en los “beans”, donde
tienen una correspondencia con formularios. Aunque
estos métodos son realmente generales, los distinguimos
por separado por tener unas normas concretas de estilo.
4

MÉTODOS GENÉRICOS
Comencemos con el formato básico de un método:

• El primer calificador es public o private, se establece el tipo de


vista.
• La segunda palabra es el tipo devuelto, con el calificador void
(vacío) se indica que no devuelve nada, o se establece el tipo
devuelto.
• La siguiente palabra representa el nombre del método.
• A continuación vienen los parámetros. Hay que resaltar, en
general, los datos con que actúan los métodos son sus propios
atributos, y por lo tanto no es necesario pasarlos por
parámetro.
• Por parámetros se pasan valores externos al objeto que
condicionan la acción a realizar.
5

MÉTODOS GENÉRICOS
Existen dos tipos de parámetros:
• Tipos primitivos (byte, short, int...). Estos se pasan por
valor, es decir, se realiza una copia del contenido en la
variable que sustenta el parámetro, y por lo tanto, si el
método modifica el parámetro no afecta a la variable del
mensaje entrante.
• Clases y arrays. Estos se pasan por referencia, es decir,
se pasa la dirección en el parámetro, así que,
modificaciones dentro del método afectan externamente.
Con la sentencia return, se finaliza el método. Si además
queremos devolver un valor se utiliza la sentencia return
valor.
6

MÉTODOS GENÉRICOS
Ampliando la clase Punto, se podrían añadir los siguientes
métodos:

• Dentro de los métodos podemos definir variables locales, las


cuales serán creadas en la ejecución del método y destruidas
al finalizar la ejecución. Las variables de método, sólo son
visibles desde el propio método en que se definen.
7

MÉTODOS PARA ACCESOS A


ATRIBUTOS (GETTERS & SETTERS)
Realmente, a nivel gramatical, se pueden considerar como
métodos genéricos. Pero semánticamente, tienen
connotaciones especiales que debemos especificar.
Cada atributo tiene dos posibles acciones: leer su valor o
establecerlo. No tenemos por qué realizar ambas acciones,
depende del diseño que se busque, de hecho, muchas
clases no tiene estos métodos.
8

Existe una nomenclatura especial para el identificador del


método:
• Establecer el valor de un atributo: set + NombreAtributo
(la primera letra del nombre del atributo debe estar en
mayúsculas). Por ejemplo, en la clase Punto sería setX y
setY, en la clase NumeroNatural sería setValor, en la
clase Usuario sería setNombre, setEMail, setCiudad,
setMovil y setMayorEdad.
• Leer el valor del atributo: get + NombreAtributo o is +
NombreAtributo si devuelve un tipo boolean. Por ejemplo,
en la clase Punto sería getX y getY, en la clase
NumeroNatural sería getValor, en la clase Usuario sería
getNombre, getEMail, getCiudad, getMovil y
isMayorEdad.
9

A continuación, presentamos la estructura completa del


código de la clase Punto:
10

CONSTRUCTORES
• Los constructores son métodos especiales que reúnen
las tareas de inicialización de los objetos de una clase;
por lo tanto, el constructor establece el estado inicial de
todos los objetos que se instancian.
• No es obligatorio definir constructores, si no se realiza,
existe un constructor por defecto sin parámetros.
• La ejecución del constructor es implícita a la creación de
una instancia.
• La restricción sintáctica del constructor es que debe
llamarse igual que la clase y no devuelve ningún tipo ni
lleva la palabra void.
11

Como ejemplo añadimos un constructor a la clase Punto:


12

SOBRECARGA DE MÉTODOS
• La sobrecarga es definir dos o más métodos con el mismo
nombre, pero con parámetros diferentes por cantidad o tipo.
• El objetivo de la sobrecarga es reducir el número de
identificadores distintos para una misma acción pero con
matices que la diferencian. La sobrecarga se puede realizar
tanto en métodos generales, como en constructores.
• La sobrecarga es un polimorfismo estático, ya que es el
compilador quien resuelve el conflicto del método a referenciar.
• Si definimos un constructor con parámetros, el constructor sin
parámetros deja de estar disponible; así que, si nos interesa,
se debe definir para su utilización.
13

Como ejemplo añadimos sobrecarga a los constructores


de la clase Punto:
14

Objetos
Recordemos que la clase representa el modelo para crear
objetos, pero son los objetos los elementos con los que
podemos trabajar. Los objetos ocupan memoria y tiene un
estado, que representa el conjunto de valores de sus
atributos.

Podemos crear todos los objetos que queramos, todos


ellos tiene existencia propia, pudiendo evolucionar por
caminos diferentes e independientes.
15

• Cuando se crea un objeto se debe asociar con una referencia, por lo


tanto, el proceso debe tener dos
• partes:
• 1. Declarar una variable que pueda referenciar al objeto en cuestión
• 2. Crear el objeto y asociarlo a la variable. Para crear el objeto, se
realiza con la sentencia new,
• recordar que lleva implícito llamar al método constructor para que
inicialice el objeto
• Un ejemplo sería:
16

• En este ejemplo declaramos cuatro puntos de la clase


Punto. Cada vez que se utiliza la palabra new, quiere
decir que se crea una nueva instancia, por lo tanto,
tenemos un total de dos instancias. En el ejemplo se han
utilizado dos constructores diferentes para inicializar las
instancias. Mostramos a continuación, una gráfica una
vez ejecutado todo el código.
17

• Fijarse que las variables punto1 y punto4 referencian la


misma instancia, es una manera redundante deacceder.
El contenido de una variable de instancia es la dirección
de memoria donde se encuentra el objeto. Cuando
comparamos variables de instancia, realmente
comparamos direcciones de memoria.
• Se ha declarado la variable punto3, pero no se ha
asociado a ninguna instancia, en ese caso, su contenido
es null. Podemos asignar a cualquier variable de
instancia el valor null, por ejemplo punto2=null.
18

• En Java, no existe ninguna forma explícita para destruir


un objeto, simplemente cuando un objeto se queda
huérfano (no tiene ninguna referencia) se destruye. Esta
labor la realiza el recolector de basura
• (garbage collector) de forma automática. En el ejemplo
anterior, si asignamos a punto2 un null, queda su
instancia sin referencia, por consiguiente, cuando se
lance el recolector de basura destruirá el objeto huérfano.
• Notar que cuando creamos un array NO implica la
creación de los objetos. Veamos un ejemplo en tres
pasos para entender mejor el proceso.
19

• En primer lugar se declara una variable de tipo array de


Punto.

El contenido de la variable tablaPuntos es null.


• En segundo lugar se crea el array.

• Gráficamente se describiría como:


20

• En tercer lugar se crean los objetos.

• Gráficamente se describiría como:


21

Si una variable de Clase lleva el calificador final, quiere decir que no se


permite que se vuelva a crear una nueva instancia con el calificador new, pero
si se permite cambiar el contenido de la instancia.
Mostramosun ejemplo para comprender este concepto:
22

Mensajes
• Una vez creado los objetos es posible pasarles mensajes,
o invocarles métodos, para solicitar una acción.
• El objeto establece a que método le corresponde el
mensaje recibido.
• Para enviar un mensaje se realiza con la notación punto.
Dependiendo si almacenamos el valor devuelto del
método, tenemos dos posibilidades:
23

• La segunda línea es para guardar el valor devuelto por el


mensaje.
Ponemos a continuación un ejemplo con la clase Punto:
24

THIS
• Dentro de la implementación de un método, a veces se
necesita referenciar a la propia instancia a la cual
pertenece. Para ello está la palabra reservada this. Un
ejemplo dentro de Punto sería: this.x = 3.
Resulta recomendable calificar con this a todas las
referencias a los atributos y métodos de la propia
clase; y resulta obligatorio cuando queramos distinguir
entre un parámetro y una variable de instancia con
el mismo nombre.
También referenciamos con la palabra this a nuestros
propios constructores, por ejemplo this(3,4).
25

• A continuación presentamos el código completo de la


clase Punto:
26
27

• Se debe evitar la redundancia de código, para ello sólo se


implementa un solo constructor y el resto referencian a
este, mediante valores por defecto de los parámetros.
• Los constructores deben hacer uso de los métodos para
el acceso a los atributos, y así, se aplica los filtros o
controles pertinentes.
28

• Otro ejemplo es la clase NumeroNatural:


29

Tarea
• Realizar la clase Rango y PruebaRango, representa un
número natural limitado hasta un
máximo.
• Atributos:
▫ valor (lectura y escritura). En la escritura se deben filtrar lo
valores fuera de rango
▫ maximo (sólo lectura). Sólo lectura significa que el método
setMaximo está calificado conprivate o, simplemente, no se
define el método.
• Constructores:
▫ Rango(int maximo, int valor)
▫ Rango()
• Métodos:
▫ void reset(), pone valor a cero
30

• Realizar la clase ContadorCircular y


PruebaContadorCircular, representa un contador
similar a un cuentakilómetros de un coche, con cada
petición incrementa en uno el contador y al
llegar al valor máximo vuelve a cero.
• Atributos: maximo (sólo lectura) representa el valor máximo
de cuenta, sólo se puede establecer con
el constructor
• ▫ contador (sólo lectura)
• Constructores:
▫ ContadorCircular(int maximo)
• Métodos:
▫ void reset(), pone a cero el contador
▫ void incrementar(), incrementa en una unidad el contador.
Cuando se pase del máximo pasa automáticamente a cero
31

Elementos estáticos
• Todos los conceptos expuestos giran en torno a una idea
principal: la clase describe el comportamiento de
cada uno de sus objetos.
El presenta apartado pretende dar solución a cuestiones
de la clase y de todos los objetos en su generalidad. Por
ejemplo, si definimos una constante de una clase que
tenga siempre el mismo valor, es común a todos los
objetos, y por lo tanto, debe estar asociado a la clase y
no a las instancias.
Podemos definir atributos estáticos y métodos estáticos.
32

ATRIBUTOS ESTÁTICOS
• Para definirlos se antepone al tipo la palabra static; se
deben inicializar independientemente a los
constructores.
• Estos atributos están compartidos por todos los objetos
de la clase y son accesibles desde cada uno de ellos.
• Cualquier modificación por parte de un objeto afectará al
resto.
• Para el acceso a un atributo, en lugar de utilizar this, se
utiliza el nombre de la clase: Clase.atributo.
• Normalmente, las constantes asociadas a la clase son
buenas candidatas para realizarlas estáticas.
33

• Un ejemplo sería:
34

MÉTODOS ESTÁTICOS
• Al igual que los atributos, se antepone la palabra static al
tipo devuelto.
No puede utilizar en el código la palabra this, ya que está
asociado exclusivamente a las instancias. Se pueden
acceder a los atributos estáticos del mismo modo que
pueden las instancias, anteponiendo el nombre de la
clase: Clase.atributo.
Se permite definir un código estático para inicializar los
atributos estáticos. Su ejecución se dispara una sola vez
al principio. Tiene el siguiente formato:
35

• Vamos añadir algunos aspectos estáticos a la clase


Punto:
36
37

Clases de Java fundamentales


• CLASE STRING
• CLASE STRINGBUFFER
• BOOLEAN, CHARACTER, MATH, SYSTEM...
38

CLASE STRING
• La clase String representa una cadena de caracteres. Su
contenido no puede cambiar, por lo tanto, cada vez que
se modifica, se crea un nuevo String. Consume menos
recursos para el almacenamiento pero las operaciones de
modificación resultan costosas en el tiempo. Si van a
existir numerosas modificaciones se deberá utilizar la
clase StringBuffer.
Todos los literales de cadena como “abc” se implementan
como una instancia de String, podemos realizar la
siguiente asignación:
39

• Si se quiere crear cadena a partir de valores o variable


nativas, debemos utilizar el método estático valueOf(...).
• Algunos ejemplos son:
40

• El operador “+” entre cadenas se interpreta como


concatenación. De forma automática, al concatenar un
String con cualquier otro valor, lo convierte a String.
Algunos ejemplos son:
41

La clase String tiene un gran número de métodos que nos


facilita el trabajo con cadenas. Algunos de los
más útiles son:

• Longitud de la cadena (length).


• Comparar cadenas (equals, equalsIgnoreCase,
compareTo).
• Busca subcadenas (indexOf, lastIndexOf)
• Extrae subcadenas (substring, toLowerCase,
toUpperCase, trim)
42

• Algunos ejemplos de uso son:


43

CLASE STRINGBUFFER
• Representa una cadena de caracteres con una estructura
interna que facilita su modificación, ocupa más que la
clase String, pero las operaciones de modificación son
muchos más rápidas.
• Para la modificación del contenido, cuenta con los
métodos insert y append. Mirar el API para mayor
documentación.
• En el siguiente código, el bucle realizado con StringBuffer
es aproximadamente 1000 veces más rápido que el bucle
realizado con String. Es importante tener en cuenta, que
cuando se requiera modificar una cadena de forma
dinámica, es mucho mejor la clase StringBuffer.
44
45

CLASE NUMBER: INTEGER,


DOUBLE...
• Este conjunto de clases son útiles para trabajar con los valores
primitivos, pero mediante objetos. A estas clases se les llama
Wrappers (envolventes). Cada clase especializada, almacena un tipo
de valor primitivo numérico.

• Las clases tienen constantes que nos pueden ser útiles, por ejemplo,
todas tienen MAX_VALUE y MIN_VALUE, Double además tiene NaN,
POSITIVE_INFINITY. NEGATIVE_INFINITY...

• Otra utilidad interesante es que pueden convertir una cadena a un


valor numérico, por ejemplo, Integer.parseInt(“123”) devuelve el valor
int asociado.

• Para facilitar su uso, en las versiones actuales de Java existe una


conversión automática entre el tipo
primitivo y el objeto (boxing y unboxing).
46

• El siguiente código funciona correctamente:


47

BOOLEAN, CHARACTER, MATH,


SYSTEM...
• Mirar el API para mayor documentación.
48

Javadoc
• Los comentarios JavaDoc están destinados a describir,
principalmente, las clases y los métodos. Existe una
utilidad de Sun Microsystems llamada javadoc.exe para
generar APIs en formato HTML de un documento de
código fuente Java.
• Recordar que los comentarios javadoc son /** ... */. Se
utiliza el carácter @ como anotación especial,
para describir las etiquetas de documentación.
• Los comentarios se deben colocar justo antes del
elemento a documentar.
• Se permite añadir elementos HTML a la documentación
para enriquecer su presentación.
49

ANOTACIONES
• Generales
@see referencia. Realiza un enlace. Ejemplos de
referencia son: #metodo(); clase#metodo();
paquete.clase; paquete.clase#metodo(); <a href=”...”>
enlace estándar de html </a>
@since. Indica en la versión que apareció este
elemento respecto a la secuencia de versiones
aparecidas a lo largo del tiempo
50

• Paquetes
@see, @since
@deprecated. Indica que este elemento es antiguo y
que no se recomienda su uso porque
posiblemente desaparecerá en versiones posteriores. Se
debe indicar la alternativa correcta
51

• Clases e interfaces
@see, @since, @deprecated
@version. Versión y fecha
@author. Nombre del autor
52

• Variables y constantes
@see, @deprecated
53

• Métodos
@see, @since, @deprecated
@param. Definición de un parámetro, es obligatorio.
*@param parámetro descripción
@return. documenta el dato devuelto, es obligatorio.
*@return descripción
@throws. Excepción lanzada por el método. *@throws
clase descripción
54

• Un ejemplo de uso podría ser:


55
56
57
58

• Para generar la documentación en html, se realiza de


forma parecida a compilar, desde la consola se teclea:
javadoc Mes.java. Conviene realizarlo en una carpeta
independiente debido a la gran cantidad de ficheros que
genera. Si queremos realizarlos con varias clases sería:
59

TAREA
• Implementar por completo la clase Mes
• Realizar la clase Pila, gestiona una pila de cadenas
(String) (último en entrar, primero en
salir). Para su implementación se apoya en un array
60

TAREA
• Atributos:
▫ capacidad (sólo lectura) representa el tamaño del array. Sólo se
puede establecer en el
constructor
▫ array (sin acceso). Significa que los métodos get/set son private o
no se implementan
▫ cima (sin acceso) apunta al último elemento apilado, -1 si no hay
elementos
• Constructores:
▫ Pila(int capacidad), Pila()
• Métodos:
▫ void borrarTodo(), borra toda la pila
▫ boolean apilar(String elemento). Añade un elemento en la cima de
la pila, si no está llena.
Devuelve un valor lógico indicando si se ha podido añadir
▫ String desapilar(). Devuelve el String que se encuentra en la cima
de la pila, devuelve null
si la pila está vacía
▫ int tamaño(). Devuelve el número de String que contiene la pila
61

• La clase TestPila debe tener el siguiente contenido:


62
63

• La clase IO, mediante el método addController(), crea un


botón por cada método que aparezca en la
prueba. Estos métodos devuelven void y no pueden tener
parámetros. Así, podremos probar de una
manera gráfica todos los métodos de nuestra clase.
HERENCIA Y
POLIMORFISMO
65

Herencia
• La herencia es uno de los mecanismos fundamentales de la
programación orientada a objetos, por medio del cual una clase se
construye a partir de otra. Una de sus funciones más importantes es
proveer el polimorfismo.
La herencia relaciona las clases de manera jerárquica; una clase
padre, superclase o clase base sobre otras clases hijas, subclases o
clase derivada. Los descendientes de una clase heredan todos los
atributos y métodos que sus ascendientes hayan especificado como
heredables, además de crear los suyos propios.
En Java, sólo se permite la herencia simple, o dicho de otra manera,
la jerarquía de clases tiene estructura de árbol. El punto más alto de
la jerarquía es la clase Object de la cual derivan todas las demás
clases.
Para especificar la superclase se realiza con la palabra extends; si no
se especifica se hereda de Object.
66

• Algunos ejemplos son:

• Cuando decimos que se heredan los miembros (atributos


y métodos) de la superclase, quiere decir que se
pueden referenciar con respecto a la clase hija. Por
ejemplo, si la clase SubClase hereda de SuperClase, y
esta tiene un método llamado mtd1, el siguiente mensaje
es correcto:
67

• En este caso, la clase hija recibe el mensaje, y esta dispone de


la funcionalidad por la herencia de la clase
padre.
• Para que un atributo o método, sea directamente utilizable por
la clase hija, debe ser public, protected o friendly (hasta ahora
sólo hemos hablado de public, en el tema de paquetes
aclararemos estos calificadores).
• Los miembros que sean private, no se podrán invocar
directamente mediante this, pero su funcionalidad esta latente
en la herencia.
• Para referirnos a los miembros de la clase padre lo
realizaremos mediante la palabra super. Con “super.”
accedemos a los miembros y con “super( )” accedemos a los
constructores.
68

• Los constructores no se heredan, aunque existe llamadas


implícitas a los constructores de las clases padre.
• Al crear una instancia de una clase hija, lo primero que se
hace es llamar al constructor de la clase padre para que
se inicialicen los atributos de la clase padre, sino se
especifica, se llama al constructor sin parámetros.
• Por ello, si se quiere invocar otro constructor, la primera
sentencia del constructor de la clase
hija debe ser la llamada al constructor de la clase padre.
69

• Veamos un ejemplo completo. Vamos a partir de la


superclase Punto, que si no se especifica su
superclase, hereda de Object.
70

• Ahora queremos crear una clase Punto pero que tenga


en cuenta el tiempo, le llamaremos PuntoTiempo.
71

• Cuando creamos una instancia de PuntoTiempo, se crea


una instancia que lleva la funcionalidad de
Object, Punto y lo añadido en PuntoTiempo. En la clase
PuntoTiempo, en el constructor sin parámetros,
como no hemos especificado el constructor de la
superclase, se llama al super()
72

COMPATIBILIDAD ENTRE VARIABLES


DE CLASES
• Una variable de tipo MiClase, puede contener instancias
de MiClase o de cualquier subclase.
• En definitiva, una variable de tipo Object, puede contener
cualquier instancia. En cambio, está prohibido que una
variable tenga instancias de las superclases.
• En el ejemplo anterior, una variable de tipo Punto, puede
referencia instancias de tipo PuntoTiempo, pero
una variable de tipo PuntoTiempo NO puede tener
instancia de la clase Punto.
• También se puede realizar casting entre clases.
73

• Mostremos un ejemplo con código Java:


74

REDEFINICIÓN DE MÉTODOS
• A veces, los métodos heredados de la superclase no son
adecuados para la subclase. En este caso, debemos
redefinir (sobrescribir) el método.
• Supongamos que queramos realizar una clase Punto3D
que hereda de PuntoTiempo para añadirle una tercera
dimensión. En este caso, el método heredado modulo ya
no es válido y lo debemos redefinir.
• Es recomendable añadir la directiva @Override para
asegurarnos que no realizamos una sobrecarga por
error. A continuación mostramos el código Java.
75
76

• Fijarse que si enviamos un mensaje al método velocidad


de una instancia de la clase Punto3D, esta se resuelve
por la herencia de PuntoTiempo, cuando desde esa
instancia se invoca al método modulo, se lanza la nueva
versión redefinida en Punto3D.
77

CLASE OBJECT
• Esta clase es la superclase de todas. Algunos de sus
métodos son:
public String toString(). Este método devuelve una
cadena con la descripción de la clase. Es importante
redefinir este método para adaptarlo a nuestra
descripción, es muy útil en los procesos de depuración.
Por ejemplo, en la clase Punto podría devolver los
valores x e y entre paréntesis y separados por una coma:
[3,5]. Se recomienda que siempre se redefina este
método.
78

• Su código quedaría:
79

• public boolean equals(Object obj). Este método


devuelve un boolean indicando si la instancia actual es
igual a la que se pasa por parámetro. En Object compara
direcciones, devolviendo true si son la misma dirección.
Normalmente, esta no es la implementación necesitada.
• Por ejemplo, en la clase Punto debemos considerar que
dos instancias de punto son iguales si la coordenada x
e y coinciden.

• Su código quedaría:
80
81

• public int hashCode(). Devuelve un código hash de este


objeto. Normalmente, si se sobrescribe equals se debe sobre
escribir hashcode. Se debe buscar que el código hash sea
diferente entre dos instancias que no se consideren iguales.
• protected void finalize() throws Throwable. Este es el método
destructor, se debe redefinir si queremos realizar alguna
acción para liberar recursos. Cuando un objeto queda
huérfano, el recolector de basura lo destruye; pero esta acción
no es inmediata. Si queremos que actúe el
recolector, lo podemos lanzar mediante la clase System:
System.gc().

• Un ejemplo sería:
82

• protected Object clone() throws CloneNotSupportedException. Este


método nos proporciona una copia de la instancia, pero para ello, la
clase a duplicar debe implementar el interface Cloneable (no tiene
ningún método). Nuestra clase clone, debe encargarse de duplicar
todas las instancias que lo formen. En un apartado posterior,
veremos un ejemplo completo.
83

FINAL
• Si lo aplicamos a una clase, no se permite que existan
subclases.
• Si lo aplicamos a un método, no se permite que
subclases redefinan el método.
84

Clases abstractas
• Una clase abstracta es una clase que no se puede instanciar, pero si
heredar. También, se pueden definir constructores. Se utilizan para
englobar clases de diferentes tipos, pero con aspectos comunes. Se
pueden definir métodos sin implementar y obligar a las subclases a
implementarlos.
Por ejemplo, podemos tener una clase Figura, que representa una
figura general, y subclases con figuras concretas (Triangulo,
Circulo...). Podemos establecer métodos comunes como dibujar, que
sin conocer el
tipo de figura, sea capaz de dibujarla. Pero para que esto funcione
correctamente, cada figura concreta debe implementar su versión
particular de dibujar.
Si añadimos el calificador abstract después del calificativo public del
método, le convertimos en abstracto y no hace falta que lo
implementemos, basta con definirlo. Si añadimos abstract después
del calificativo public de la clase, la convertimos en clase abstracta;
esta acción es obligatoria si algún método es abstracto.
85

• La ventaja en utilizar clases abstractas, es que podemos trabajar con


variables o parámetros de tipo Figura y llamar a los métodos
comunes sin saber a priori el tipo concreto de Figura. Esto permite en
el futuro, añadir nuevas Figuras, sin cambiar las clases ya
desarrolladas. A este concepto se le llama polimorfismo.
• El polimorfismo es el modo en que la POO implementa la polisemia,
es decir, un solo nombre para muchos conceptos.
• Este tipo de polimorfismo se debe resolver en tiempo de ejecución y
por ello hablamos de polimorfismo dinámico, a diferencia del
polimorfismo estático que se puede resolver en tiempo de
compilación.
• Recordemos que éste último se implementaba mediante la
sobrecarga de los métodos.
• A continuación presentamos un ejemplo mediante notación UML
(Lenguaje Unificado de Modelado) y en Java
86
87
88
89

TAREA
• Crear las clases Figura, Triangulo y Cuadrado
90

Interfaces
• Un interface en Java es un tipo abstracto que representa un
conjunto de métodos que las clases pueden implementar y un
conjunto de constantes estáticas, es similar a una clase
abstracta pura. Igual que las clases abstractas, los interfaces
no se puede instanciar. Su utilidad es crear variables de tipo
interface e independizarse de las clases que contiene.
• Para definir un interface se realiza de forma similar a una clase
abstracta pero con la palabra interface en lugar de class.
Pueden existir variables o parámetros de tipo interface, pero
sólo pueden contener instancias de clases que hayan
implementado el interface.
Una clase puede implementar uno o varios interfaces, para ello
se utiliza la palabra implements
91

• Veamos un ejemplo :
92

• Los interfaces pueden heredar de otros mediante la


palabra extends, además se permite herencia múltiple.
Mediante interfaces también se aporta el polimorfismo, de
hecho, una clase abstracta con todos sus métodos
abstractos es similar a un interface.
En general, es mejor utilizar interface; sólo cuando se
necesite añadir código, se debe utilizar las clases
abstractas.
93

COMPATIBILIDAD DE VARIABLES
• Una variable de tipo interface puede contener instancias
de cualquier clase que implemente el interface.
94

TAREA
• Implementar las clases CA y CB, donde CA hereda
de CB
95

TAREA
• Dado el código: CA a = new CB(1,2,3); ¿qué valor
devuelve “a.suma();”?
Implementar las clases CC y CD
96

TAREA
• Añadir e implementar el método int dobleSuma() en
CC
Definir el interface IA, IB e IC
97

TAREA
• Ampliar CA para que implemente IC
interface Comparable, método compareTo(Object
uno, Object dos) (si uno es <, == o > que
dos, devuelve -1, 0, +1 respectivamente
Ampliar CA para que implemente Comparable
98

Colaboración entre clases


• Llegados a este punto, se recomienda pasar a un entorno
de desarrollo avanzado, como por ejemplo Eclipse.
• En el área de documentación del MIW se encuentra un
tutorial rápido sobre Eclipse.
99

Relación de clases
• Una aplicación orientada a objetos es una colección de
objetos, algunos jerárquicos, que colaboran entre sí para dar
una funcionalidad global. Cada objeto debe asumir una parte
de las responsabilidades y delegar otras a los objetos
colaboradores.
Existirá un objeto que asume las funciones de la aplicación
principal, será el encargado de crear las instancias oportunas y
tener un único punto de entrada (main).
Cabe hacernos una pregunta fundamental, ¿en cuántos
objetos debemos descomponer la aplicación? Si realizamos
objetos muy pequeños, cada uno de ellos será fácil de
desarrollar y mantener, pero resultará difícil la integración. Si
realizamos objetos grandes, la integración será fácil, pero el
desarrollo de los objetos complicado. Se debe buscar un
equilibrio.
100

• En definitiva, y sin adentrar más en la estructuración de


las clases de una aplicación, nuestra aplicación tendrá un
conjunto de clases que están relacionadas entre ellas.
Se considera que existe una relación entre clases si sus
objetos colaboran, es decir, se mandan mensajes.
Tenemos, principalmente, tres tipos de relaciones entre
clases: uso, agregación y composición, aparte de la
herencia que ya vimos.
101

RELACIÓN DE USO
• Se dice que existe una relación de uso entre la clase A y la
clase B, si un objeto de la clase A lanza mensajes al objeto de
la clase B, para utilizar sus servicios.
Es una relación que se establece entre un cliente y un servidor,
en un instante dado, a través de algún método. No es una
relación duradera, sino esporádica.
102

RELACIÓN DE AGREGACIÓN
• Se dice que la clase A tiene una relación de agregación
con la clase B, si un objeto de la clase A delega
funcionalidad en los objetos de la clase B.
El ciclo de vida de ambos no coincide. La clase A delega
parte de su funcionalidad, pero no crea el objeto B. Varios
objetos pueden estar asociados con el objeto B. Al objeto
A, le deben pasar la referencia del objeto B. Es una
relación duradera entre cliente (A) y servidor (B), en la
vida del cliente.
103
104

RELACIÓN DE COMPOSICIÓN
• Se dice que existe una relación de composición entre la
clase A y la clase B, si la clase A contiene un
objeto de la clase B, y delega parte de sus funciones en
la clase B.
Los ciclos de vida de los objetos de A y B coinciden. El
objeto A es el encargado de crear la instancia de
B y de almacenarla, siendo su visibilidad privada. Cuando
se destruye A, también se destruye B.
Por ejemplo, se pretende realizar una clase llamada
Agenda y para ello se apoya en la clase Lista, es
decir, dentro de la clase Agenda se crea una instancia de
la clase Lista, y algunas de las funcionalidades
que aporta la clase Agenda se realizan con los métodos
de la clase Lista.
105
106

Beneficio de la herencia
• Es importante realizar un uso de la herencia para
optimizar el código y evitar las redundancias.

Ejemplo incorrecto. Veamos un mal uso de las clases,


donde aparece código redundante y un mal uso de
la herencia. Resulta incorrecto que el mensaje lo interprete
el emisor
107
108
109

Beneficio de los interfaces


• En una relación de agregación entre la clase A y la clase B, la clase A
debe lanzar mensajes a la clase B, y para ello es necesario que a la
clase A le pasen la referencia de la clase B.
Una vez implementada la clase A, cualquier cambio en la clase B (por
ejemplo, una mejora o cambio en la implementación de B) provoca
que debiéramos cambiar todas las referencias de la clase A sobre la
B.
Incluso puede que cuando se desarrolla la clase A, no se dispone de
la implementación de la clase B.
• Para ello disponemos de dos mecanismos:
Mediante herencia de clases. En este caso, la clase A referencia
los métodos de una superclase B, pudiendo existir varias subclases
de B que sirvan a tal fin.
Mediante interfaces. Aquí, la clase A referencia los métodos de un
interface, pudiendo existir un conjunto de clases que implementan
ese interface.
Fijarse, que en ambos casos, cualquier cambio en las
implementaciones de B no afectan al código de la
clase A.
110

• Veamos un ejemplo concreto. En esta asociación existe una


dependencia clara entre la clase Servidor y la clase Cliente.

• Pero en este ejemplo tenemos un problema, ya que se


pretende construir un servidor para que de servicio
a muchos clientes y se debe romper esa dependencia. Obligar
a los clientes a heredar de una clase no es una buena
solución, ya que le limitamos en su diseño. La mejor solución
es utilizar los interfaces. A continuación presentamos el nuevo
diseño:
111
112

• En este nuevo diseño, hemos roto la dependencia del


servidor respecto a las diferentes implementaciones de
los clientes.
Veamos el código en Java:
113
114

Paquetes
• Los paquetes agrupan un conjunto de clases que trabajan
conjuntamente sobre el mismo ámbito.
• Es una facilidad ofrecida por Java para agrupar sintácticamente
clases que van juntas conceptualmente y definir un alto nivel de
protección para los atributos y los métodos.
• La ventaja del uso de paquetes es que las clases quedan ordenadas
y no hay colisión de nombres. Si dos programadores llaman igual a
sus clases y luego hay que juntar el código de ambos, basta explicitar
a qué paquete nos referimos en cada caso.
• La forma de nombrar los paquetes es con palabras (normalmente en
minúsculas) separadas por puntos, estableciendo una jerarquía de
paquetes; la jerarquía de paquetes es independiente de la jerarquía
de clases.
• Las clases deben almacenarse en una estructura de carpetas que
coincidan con la estructura de paquetes.
115

• Para que una clase se asocie a un paquete se realiza con


la sentencia package, se debe situar antes de la clase.
Algunos ejemplos de uso son:

Tanto los ficheros *.java como los ficheros *.class, se deben almacenar
en una jerarquía de carpetas que coincidan con los nombres de los
paquetes. En la Figura 4. Paquetes en JavaFigura 4 se muestra cómo
quedaría la jerarquía de carpetas.
116
117

• Para su distribución, se genera un fichero jar, con la estructura


de carpetas y clases. Recordar que este fichero no hace falta
descomprimirlo. Para su utilización, se debe referenciar
mediante la variable de entorno CLASSPATH.
Para referenciar cualquier clase en Java, se debe poner el
paquete al que pertenece.
• Por ejemplo:
java.lang.String, java.lang.Integer... Hasta ahora no ha sido
necesario, porque el paquete java.lang se importa de forma
automática.
Para evitar poner siempre la jerarquía de paquetes, se puede
utilizar la sentencia import, con ello se puede importar un
paquete entero (pero no los paquetes inferiores) o una clase.
118

• Mostramos un ejemplo de uso:


119

• Los paquetes permiten utilizar los modos de protección


protected y friendly, que veremos en el siguiente
apartado.
Nota. Se recomienda utilizar un IDE, como por ejemplo
Eclipse. Para ejecutar una clase de un paquete desde la
consola que no está referenciado mediante la variable de
entorno CLASSPATH, nos debemos situar en la carpeta
raíz de los paquetes y referenciar al fichero class
mediante la ruta completa.

Un ejemplo es: java paquete/MiClase
120

Visibilidad
• Hasta ahora hemos trabajado con dos tipos de visibilidad:
public y private. Recordemos que un elemento público es
visible para todos y privado sólo para uso interno.
• En Java se distinguen dos tipos más de visibilidad: protected y
sin calificador (friendly).
• Cuando calificamos a un elemento como protected, significa
que es visible dentro del mismo paquete o cualquier subclase
de forma directa (mediante super.elemento).
• Si pretendemos que una subclase acceda de forma directa a
los atributos de la superclase, el calificador apropiado es
protected.
• Cuando dejamos a un elemento sin calificar (friendly), significa
que sólo es visible dentro del mismo paquete.
121

TAREA
• Probar las clases anteriores
• AVANZADO. Implementar la aplicación AGENDA.
Lleva una agenda de usuarios. Permite
las acciones básicas: añadir, borrar, buscar y ver toda
la agenda.
122
FIN CAPITULO
GRACIAS POR SU ATENCIÓN
Referencia Bibliográfica:
• Jesús Bernal Bermúdez, Luis Fernández Muñoz, (2012)
Programación Orientada a Objetos con Java, Master
Universitario en Ingeniería Web, Universidad Politécnica de
Madrid.

También podría gustarte