Professional Documents
Culture Documents
Interfaces e
polimorfismo
Helder da Rocha (helder@acm.org) argonavis.com.br1
O que é polimorfismo
2
Objetos substituíveis
freia() Onibus
acelera() Veiculo
vira(1) freia() Jipe Subclasses
acelera() de Veiculo!
vira(direcao) Jegue (herdam
todos os
Aviao métodos)
Por exemplo: objeto do tipo Manobrista
sabe usar comandos básicos para controlar Usuário de Veiculo
Veiculo (não interessa a ele saber como ignora existência desses
cada Veiculo diferente vai acelerar, frear objetos substituíveis
ou mudar de direção). Se outro objeto tiver
a mesma interface, Manobrista saberá usá-lo
3
Programas extensíveis
Novos objetos podem ser usados em programas que não
previam a sua existência
Garantia que métodos da interface existem nas classes novas
Objetos de novas classes podem ser criados e usados (programa pode
ser estendido durante a execução)
...
Veiculo Jipe
freia() freia() LandRover
acelera() acelera()
vira(0) vira(direcao) AirBus LandRover
Concorde
Mesmo nome. Aviao
Implementações Veiculo v1 = new Veiculo();
diferentes. Veiculo v2 = new Aviao();
Veiculo v3 = new Airbus();
v1.acelera(); // acelera Veiculo Concorde AirBus
v2.acelera(); // acelera Aviao
v3.acelera(); // acelera AirBus
4
Interface vs. implementação
Estabelece uma
Polimorfismo permite separar a interface comum
interface da implementação
A classe base define a interface Veiculo
freia() {}
comum acelera() {}
Não precisa dizer como isto vai ser feito vira(dir) {}
Não diz: eu sei como frear um Carro ou
um Ônibus
Diz apenas que os métodos existem, que
eles retornam determinados tipos de
dados e que requerem certos parâmetros Onibus Carro
Diz: Veiculo pode acelerar, frear e virar freia() {} freia() {}
para uma direção, mas a direção deve ser acelera() {} acelera() {}
vira(dir) {} vira(dir) {}
fornecida
Implementações
da interface
(dizem como fazer)
5
Como funciona
6
Ligação de chamadas de função
herdam interface
Herdam interface
mas sobrepõem
Jegue implementação
LandRover
(cada objeto
executa o método
freia() {...} freia() {...}
de forma particular)
acelera() {...} acelera() {...}
vira(direcao) {...} vira(direcao) {...}
8
Exemplo (2)
Trecho de programa que usa Manobrista: Manobrista usa a classe
Em tempo de execução passa implementação de Veiculo (e ignora a existência
Jegue e LandRover no lugar da implementação de tipos específicos de Veiculo
original de Veiculo como Jegue e LandRover
(aproveita apenas a interface de Veiculo)
Veiculo Manobrista
(...)
(...)
Manobrista
Manobrista mano
mano freia() {...}
== new
new Manobrista (); acelera() {...}
Manobrista ();
vira(direcao) {...}
Veiculo
Veiculo v1
v1 == new
new Jegue();
Jegue();
Veiculo
Veiculo v2
v2 == new
new LandRover();
LandRover(); herdam interface
mano.estaciona(v1);
mano.estaciona(v1); public void
estaciona (Veiculo v) {
mano.estaciona(v2);
mano.estaciona(v2); Jegue freia() {...}
LandRover ...
acelera() {...} v.freia();
(...)
(...) ...
vira(direcao) {...}
freia() {...} freia() {...} }
acelera() {...} acelera() {...}
vira(direcao) {...} vira(direcao) {...}
9
Detalhes (1) Manobrista
public void
estaciona (Veiculo v) {
...
Pilha (referências) Heap (objetos) v.freia();
...
03b7 }
Veiculo v1 0xffa903b7 Jegue
10
Como funciona (3) Manobrista
public void
estaciona (Veiculo v) {
...
v.freia();
...
03b7 }
Veiculo v1 0xffa903b7 Jegue
12
Métodos e classes abstratos
13
Classes abstratas
14
Classes abstratas (2)
dois()
}
ClasseConcretaUm ClasseConcretaDois
16
Template method: implementação
public abstract class Template {
public abstract String link(String texto, String url);
public String transform(String texto) { return texto; }
18
Upcasting
19
Downcasting
20
Upcasting e downcasting
• Upcasting • Downcasting
– sobe a hierarquia métodos visíveis – desce a hierarquia
– não requer cast na referência v – requer operador de cast
Veiculo v = new Jegue(); Jegue j = (Jegue) v;
Veiculo
freia()
j freia()
acelera()
Veiculo
freia()
acelera() vira() acelera()
vira(direcao) corre() vira(direcao)
v morde()
21
ClassCastException
O downcasting explícito sempre é aceito pelo compilador se
o tipo da direita for superclasse do tipo da esquerda
Veiculo v = new Onibus();
Carro c = (Carro) v; // passa na compilação
Object, portanto, pode ser atribuída a qualquer tipo de referência
Em tempo de execução, a referência terá que ser ligada ao
objeto
Incompatibilidade provocará ClassCastException
Para evitar a exceção, use instanceof
if (v instanceof Carro)
c = (Carro) v;
22
Herança Pura vs. Extensão
• Herança pura: referência têm • Extensão: referência apenas tem
acesso a todo o objeto acesso à parte definida na
interface da classe base
Veiculo
freia() Veiculo
acelera()
Referência Veiculo freia()
vira(direcao)
acelera()
não enxerga estes vira(direcao)
métodos
Jegue LandRover
freia() freia() Jegue LandRover
acelera() acelera()
freia() freia()
vira(direcao) vira(direcao)
acelera() acelera()
vira(direcao) vira(direcao)
Veiculo v = new Jegue(); corre() diesel()
v.freia() // freia o Jegue morde() tracao(tipo)
v.acelera(); // acelera o Jegue
Veiculo v = new Jegue();
v.corre() // ERRADO!
v.acelera(); //OK
23
Ampliação da referência
Uma referência pode apontar para uma classe estendida, mas
só pode usar métodos e campos de sua interface Object
Para ter acesso total ao objeto que estende a interface equals()
original, é preciso usar referência que conheça toda sua
interface pública ERRADO: raio não faz parte da
Circulo
Exemplo interface de Object
raio
equals()
class
class Circulo
Circulo extends
extends Object
Object {{
public
public int int raio;
raio; class
class Circulo
Circulo extends
extends Object
Object {{
public
public boolean
boolean equals(Object
equals(Object obj)
obj) {{ public
public int int raio;
raio;
ifif (this.raio
(this.raio ==
== obj.raio)
obj.raio) public
public booleanboolean equals(Object
equals(Object obj)obj) {{
return
return true;
true; ifif (obj
(obj instanceof
instanceof Circulo)
Circulo) {{
verifica se obj Circulo
return
return false;
false; Circulo kk = = (Circulo)
(Circulo) obj;
obj;
}} realmente ifif (this.raio
(this.raio ==
== k.raio)
k.raio)
}} //// CÓDIGO
CÓDIGO ERRADO! ERRADO! é um Circulo return
return true;
true;
}}
cria nova referência return
return false; false;
que tem acesso a toda Como k é Circulo
}}
a interface de Circulo possui raio
}}
24
Interfaces Java
26
Herança múltipla em Java
Classe resultante combina todas as interfaces, mas só possui
uma implementação: a da classe base <<interface>>
Empregado
<<interface>> <<interface>>
Animal Cidadao Contribuinte trabalha()
come() vota() pagaIR()
respira() getRG() getCPF() <<interface>>
Professor
ensina()
come()
respira() Este objeto pode ser usado
Pessoa ensina() em qualquer lugar onde
vota() for aceito um Animal, um
ensina() getRG()
vota() Professor, um Cidadao, um
trabalha()
getRG() pagaIR() Empregado, um Contribuinte,
trabalha() getCPF() um java.lang.Object
pagaIR() equals()
getCPF() toString()
... 27
Exemplo
32
Curso J100: Java 2 Standard Edition
Revisão 17.0
argonavis.com.br
33