Buscar

ltp2_parte6

Esta é uma pré-visualização de arquivo. Entre para ver o arquivo original

*
*
*
Herança
_____________________
Imagine um Banco que possui funcionários, com as seguintes características em comum:
public class Funcionario {
 private String nome; 
 private double salario;
 
 public void setSalario (double sal) {
 this.salario= sal;
 } 
 
 public double getSalario () {
 return this.salario;
 }
}
*
*
*
Mais um funcionário...
_____________________
Além de um funcionário comum, há também outros cargos, como os gerentes.
Os gerentes possuem os mesmos atributos de um funcionário comum, como matricula e salario, e métodos setSalario e getSalario, mas estes devem possuir outras funcionalidades, tais como, usar o método setFormarEquipe (). 
public class Gerente{
 private String nome;
 private double salario;
 private String equipe;
 
 public void setSalario (double sal) {
 this.salario= sal;
 } 
 public float getSalario () {
 return this.salario;
 }
 
 public void setFormarEquipe (String nome) {
 this.equipe = this.equipe + nome + “\n”;
 }
}
*
*
*
Se houver outros tipos de funcionários?
_____________________
Sempre que houver outros tipos de Funcionarios (Diretor, operador, etc) com atributos e funcionalidades próprias, será necessário criar novamente uma nova classe e copiar o todo o código com as propriedades em comum, e incluir as novas.
Além disso, se um dia precisarmos adicionar uma nova informação para todos os funcionários, precisaremos passar por todas as classes de funcionário e adicionar esse atributo. 
O problema acontece novamente por não centralizar as informações principais do funcionário em um único lugar!
*
*
*
Usando a herança
_____________________
Existe um jeito, em Java, de relacionarmos uma classe de tal maneira que uma delas herda tudo que a outra tem. 
Isto é uma relação de classe mãe e classe filha. 
No nosso caso, gostaríamos de fazer com que o Gerente tivesse tudo que um Funcionario tem, gostaríamos que ela fosse uma extensão de Funcionario. 
Fazemos isto através da palavra chave extends.
*
*
*
Classe com herança
_____________________
public class Gerente extends Funcionario{
 private String equipe; 
 
 public void setFormarEquipe (String nome) {
 this.equipe = this.equipe + nome + “\n”;
 }
}
*
*
*
Gerente herda Funcionario
_____________________
Em todo momento que criarmos um objeto do tipo Gerente, este objeto possuirá também os atributos definidos na classe Funcionario, pois agora um Gerente é um Funcionario:
*
*
*
Instanciando a classe Gerente
_____________________
Dizemos que a classe Gerente herda todos os atributos e métodos da classe mãe, no nosso caso, a Funcionario. 
Para ser mais preciso, ela também herda os atributos e métodos privados, porém não consegue acessá-los diretamente.
 
public class Gerente extends Funcionario{
 private String equipe; 
 public void setFormarEquipe (String nome) {
 this.equipe = this.equipe + nome + “\n”;
 }
}
public class Funcionario {
 private String nome; 
 private double salario;
 
 public void setSalario (double sal) {
 this.salario= sal;
 } 
 public double getSalario () {
 return this.salario;
 }
}
public class ImplementaGerente{
 public static void main (String args[]) {
 Gerente ger = new Gerente ();
 ger.setSalario(5000.0); 
 System.out.println(ger.getSalario());
 ger.setFormarEquipe (“Marcos”);
}
*
*
*
Super e Sub classe
_____________________
A nomenclatura mais encontrada é que Funcionario é a superclasse de Gerente, e Gerente é a subclasse de Funcionario. 
Dizemos também que todo Gerente é um Funcionário.
*
*
*
Encapsulamento em Herança
_____________________
E se precisamos acessar os atributos que herdamos? 
Não gostaríamos de deixar os atributos de Funcionario, public, pois dessa maneira qualquer um poderia alterar os atributos dos objetos deste tipo. 
Existe um outro modificador de acesso, o protected, que fica entre o private e o public. 
Um atributo protected só pode ser acessado (visível) pela própria classe ou suas subclasses.
public class Funcionario {
 protected String nome; 
 protected double salario;
 // métodos devem vir aqui
 
}
*
*
*
Sempre usar protected?
_____________________
Então porque usar private? 
Depois de um tempo programando orientado a objetos, você vai começar a sentir que nem sempre é uma boa idéia deixar que a classe filha acesse os atributos da classe mãe, pois isso quebra um pouco a idéia de que só aquela classe deveria manipular seus atributos.
*
*
*
Mais Herança...
_____________________
Da mesma maneira podemos ter uma classe Diretor que estenda Gerente, e a classe Presidente pode estender diretamente de Funcionario.
Uma classe pode ter várias filhas, mas pode ter apenas uma mãe, é a chamada herança simples do java.
*
*
*
Mais Herança...
_____________________
*
*
*
Reescrita de método
_____________________
Vamos supor que os Gerentes de um Banco ganham além do salário base, também uma comissão de R$ 2.000,00.
Não podemos incluir essa comissão na classe Funcionário, pois poderá modificar incorretamente outras classes filhas, como a classe Diretor, Secretária, etc.
Para solucionar esse problema, há necessidade de se reescrever o método getSalario () na classe Gerente. 
*
*
*
Exemplo de reescrita
_____________________
No Java, quando herdamos um método, podemos alterar seu comportamento. 
Podemos reescrever (sobrescrever, override) este método:
public class Gerente extends Funcionario{
 private String equipe; 
 
 public double getSalario () {
 return this.salario + 2000.0;
 }
 public void setFormarEquipe (String nome) {
 this.equipe = this.equipe + nome + “\n”;
 }
}
public class Funcionario {
 protected String nome; 
 protected double salario;
 
 public void setSalario (double sal) {
 this.salario= sal;
 } 
 public double getSalario () {
 return this.salario;
 }
}
*
*
*
Super
_____________________
Aqui teríamos um problema: o dia que o getSalario do Funcionario mudar, precisaremos mudar o método do Gerente para acompanhar o novo salario. 
Para evitar isso, o getSalario do Gerente pode chamar o do Funcionario utilizando-se da palavra chave super.
public class Gerente extends Funcionario{
 private String equipe; 
 
 public double getSalario () {
 return super.getSalario() + 2000.0;
 }
 public void setFormarEquipe (String nome) {
 this.equipe = this.equipe + nome + “\n”;
 }
}
public class Funcionario {
 protected String nome; 
 protected double salario;
 
 public void setSalario (double sal) {
 this.salario= sal;
 } 
 public double getSalario () {
 return this.salario + this.salario*0.2;
 }
}
*
*
*
Chamada do método da Superclasse
_____________________
A invocação usando o Super vai procurar o método com o nome getSalario de uma super classe de Gerente, que será Funcionario. 
Essa é uma prática comum, pois muitos casos o método reescrito geralmente faz “algo a mais” que o método da classe mãe. 
Chamar ou não o método de cima é uma decisão sua e depende do seu problema. 
Algumas vezes não faz sentido invocar o método que reescrevemos.
*
*
*
Exercício 17
_____________________
A partir da classe super classse Funcionario, crie uma subclasse chamada Diretor que possui atributo lucro (double), e os métodos void setBonusLucro (double luc), como operação do método, a variável luc do argumento será somada ao atributo lucro, e o método getBonusLucro(), que retornará o valor do atributo lucro. Além disso, reescreva o método getSalario que deverá retornar o valor do salário base
presente na superclasse Funcionario (utilize o comando Super), somado ao valor de 0.1*getBonusLucro().
*
*
*
Polimorfismo
_____________________
O que guarda uma variável do tipo Funcionario? 
Uma referência para um Funcionario.
Na herança, vimos que todo Gerente é um Funcionario, pois é uma extensão deste. 
Podemos nos referir a um Gerente como sendo um Funcionario. 
Se alguém precisa falar com um Funcionario do banco, pode falar com um Gerente! Porque? 
Pois Gerente é um Funcionario. Essa é a semântica da herança.
Gerente gerente = new Gerente();
Funcionario funcionario = gerente;
funcionario.setSalario(1000.0);
*
*
*
Polimorfismo
_____________________
Polimorfismo é a capacidade de um objeto poder ser referenciado de várias formas.
Cuidado, polimorfismo não quer dizer que o objeto fica se transformando, muito pelo contrário, um objeto nasce de um tipo e morre daquele tipo, o que pode mudar é a maneira como nos referimos a ele.
public class Funcionario {
 protected String nome; 
 protected double salario;
 
 public void setSalario (double sal) {
 this.salario= sal;
 } 
 public double getSalario () {
 return this.salario + this.salario*0.2;
 }
}
public class Gerente extends Funcionario{
 private String equipe; 
 
 public double getSalario () {
 return super.getSalario() + 2000.0;
 }
 public void setFormarEquipe (String nome) {
 this.equipe = this.equipe + nome + “\n”;
 }
}
*
*
*
Polimorfismo
_____________________
Até aqui tudo bem, mas e se eu tentar:
funcionario.getSalario();
Qual é o retorno desse método? 1200.0 ou 3200.0? 
No Java, a invocação de método sempre vai ser decidida em tempo de execução. 
O Java vai procurar o objeto na memória e, aí sim, decidir qual método deve ser chamado, sempre relacionando com sua classe de verdade, e não com a que estamos usando para referenciálo.
Apesar de estarmos nos referenciando a esse Gerente como sendo um Funcionario, o método executado é o do Gerente. O retorno é 3200.
*
*
*
Uso do polimorfismo
_____________________
Parece estranho criar um gerente e referenciá-lo como apenas um funcionário. Por que fariamos isso?
Na verdade, a situação que costuma aparecer é a que temos um método que recebe um argumento do tipo Funcionario:
class ControleDeSalarios {
 private double totalSalarios = 0;
 public void registra(Funcionario funcionario) {
 this.totalSalarios += funcionario.getSalario();
 }
 public double getTotalSalarios() {
 return this.totalSalarios;
 }
}
*
*
*
Exemplo
_____________________
ControleDeSalarios controle = new ControleDeSalarios();
Gerente funcionario1 = new Gerente();
funcionario1.setSalario(1000.0);
controle.registra(funcionario1);
Funcionario funcionario2 = new Funcionario();
funcionario2.setSalario(1000.0);
controle.registra(funcionario2);
System.out.println(controle. getTotalSalarios());
Qual o valor a ser exibido?
*
*
*
Acesso aos métodos
_____________________
Repare que conseguimos passar um Gerente para um método que recebe um Funcionario como argumento.
Pense como numa porta na agência bancária com o seguinte aviso: “Permitida a entrada apenas de Funcionários”. 
Um gerente pode passar nessa porta? Sim, pois Gerente é um Funcionario.
Qual será o valor resultante? Não importa que dentro do método registra da classe ControleDeSalarios receba Funcionario. Quando ele receber um objeto que realmente é um Gerente, o seu método reescrito será invocado.
Reafirmando: não importa como nos referenciamos a um objeto, o método que será invocado é sempre o que é dele.
*
*
*
Polimorfismo e reescrita (ou redefinição) 
de método
_____________________
No dia em que criarmos uma classe Secretaria, por exemplo, que é filha de Funcionario, precisaremos mudar a classe de ControleDeSalarios? Não. 
Basta a classe Secretaria reescrever os métodos que lhe parecerem necessários. É exatamente esse o poder do polimorfismo, juntamente com a herança e reescrita de método: diminuir acoplamento entre as classes, para evitar que novos códigos resultem em modificações em inúmeros lugares.
Repare que quem criou ControleDeSalarios pode nunca ter imaginado a criação da classe Secretaria
ou Engenheiro. Isso traz um reaproveitamento enorme de código.
*
*
*
Exercício 18
_____________________
Crie um pacote chamado colaboradoresCesupa, contendo Colaborador e Professor .
A classe Colaborador é uma super classe, que possui como atributo matricula (String), e os métodos void setMatricula (String m), que deverá realizar uma atribuição de m para matricula, e String getMatricula(), que retornará o valor de matricula;
A classe Professor é uma subclasse de Colaborador, possui como atributo numeroSindicato (int), realize a programação necessária para que seja implementado o método reescrito getMatricula(), que deverá retornar a soma do valor do atributo matricula herdado e mais o atributo numeroSindicato. Lembre-se de que a classe Professor deverá manipular o atributo numeroSindicato com setters e getters.
Crie um pacote chamado aplicacaoCesupa, contendo a classe ImplementaColaborador, que deverá ser a classe principal do sistema, capaz de instanciar objetos da classe professor. Realize a interação necessária com usuário capaz de obter dados de matricula e númeroSindicato, e também informar a atual matricula do professor.

Teste o Premium para desbloquear

Aproveite todos os benefícios por 3 dias sem pagar! 😉
Já tem cadastro?

Outros materiais