Herança (extends)

Herança

Em Java, podemos criar classes que herdem atributos e métodos de outras classes, evitando rescrita de código. Este tipo de relacionamento é chamado de Herança.

Para representarmos este tipo de relacionamento na linguagem, devemos utilizar a palavra reservada extends, de forma a apontar para qual classe a nossa nova classe deve herdar seus atributos e métodos.

No vídeo a seguir mostramos passo a passo como funciona a herança:

Vamos praticar mais um pouco, neste outro exemplo demonstraremos a vantagem do reaproveitamento de código utilizando a Herança. Temos as classes Funcionario e Coordenador que possuem o atributo nome e matricula em comum.

package material.heranca;

/**
 * Classe utilizada para representar o Funcionario.
 */
public class Funcionario {
    private String nome;
    private int matricula;
    private String departamento;
    
    public Funcionario(String nome, int matricula, String departamento) {
        this.nome = nome;
        this.matricula = matricula;
        this.departamento = departamento;
    }

    public int getMatricula() { return matricula; }
    public void setMatricula(int matricula) { this.matricula = matricula; }

    public String getNome() { return nome; }
    public void setNome(String nome) { this.nome = nome; }

    public String getDepartamento() { return departamento; }
    public void setDepartamento(String departamento) {
        this.departamento = departamento;
    }
}
package material.heranca;

/**
 * Classe utilizada para representar o Coordenador.
 */
public class Coordenador {
    private String nome;
    private int matricula;
    private String cursoCoordenado;
    
    public Coordenador(String nome, int matricula, String cursoCoordenado) {
        this.nome = nome;
        this.matricula = matricula;
        this.cursoCoordenado = cursoCoordenado;
    }

    public int getMatricula() { return matricula; }
    public void setMatricula(int matricula) { this.matricula = matricula; }

    public String getNome() { return nome; }
    public void setNome(String nome) { this.nome = nome; }

    public String getCursoCoordenado() { return cursoCoordenado; }
    public void setCursoCoordenado(String cursoCoordenado) {
        this.cursoCoordenado = cursoCoordenado;
    }
}

Como os atributos nome e matricula, são comuns para ambas as classes, e elas possuem algo a mais em comum que é seu propósito, ambas são utilizadas para representar Pessoas.

Podemos criar uma classe Pessoa que terá os atributos nome e matricula, e por meio da herança reaproveitaremos esses atributos nas classes Funcionario e Coordenador.

package material.heranca;

/**
 * Classe utilizada para representar a Pessoa.
 */
public class Pessoa {
    private String nome;
    private int matricula;
    
    /**
     * Construtor que recebe o nome da pessoa.
     *
     * @param nome
     */
    public Pessoa(String nome, int matricula) {
        this.nome = nome;
        this.matricula = matricula;
    }

    public int getMatricula() { return matricula; }
    public void setMatricula(int matricula) { this.matricula = matricula; }

    public String getNome() { return nome; }
    public void setNome(String nome) { this.nome = nome; }
}

Agora vamos alterar a classe Funcionario e Coordenador:

package material.heranca;

/**
 * Classe utilizada para representar um Funcionario que é uma Pessoa.
 */
public class Funcionario extends Pessoa {
    private String departamento;
    
    public Funcionario(String nome, int matricula, String departamento) {
        super(nome, matricula);
        this.departamento = departamento;
    }

    public String getDepartamento() { return departamento; }
    public void setDepartamento(String departamento) {
        this.departamento = departamento;
    }
}
package material.heranca;

/**
 * Classe utilizada para representar um Coordenador que é uma Pessoa.
 */
public class Coordenador extends Pessoa {
    private String cursoCoordenado;

    public Coordenador(String nome, int matricula, String cursoCoordenado) {
        super(nome, matricula);
        this.cursoCoordenado = cursoCoordenado;
    }

    public String getCursoCoordenado() { return cursoCoordenado; }
    public void setCursoCoordenado(String cursoCoordenado) {
        this.cursoCoordenado = cursoCoordenado;
    }
}

Com a declaração acima, temos as classes Funcionario e Coordenador como classes filha ou subclasses da classe pai Pessoa. Com isso podemos dizer que as subclasses Funcionario e Coordenador herdam todos os atributos e métodos da sua super classe Pessoa.

Por isso, lembre-se, o Funcionario É UMA Pessoa, pois é uma subclasse, logo, apenas possui algumas características a mais do que Pessoa, porém podemos sempre manuseá-lo como uma Pessoa, logo, também é possível se fazer o seguinte tipo de declaração:

package material.heranca;

/**
 * Classe utilizada para testar a Herança da classe
 * Funcionario.
 */
public class TesteFuncionario {
    public static void main(String[] args) {
        /* Declarações comuns. */
        Pessoa camilo = new Pessoa("Camilo", 123);
        Funcionario rafael = new Funcionario("Rafael", 111, "informatica");
        
        /* Todo Funcionario é uma Pessoa. */
        Pessoa sakurai = new Funcionario("Sakurai", 222, "telecomunicação");
        
        /* Erro de compilação, porque nem toda
           Pessoa é um Funcionario. */
        Funcionario cristiano = new Pessoa("Cristiano", 456);
    }
}

Porém note que na linha 18, temos um erro de compilação, pois uma Pessoa nem sempre é um Funcionario, afinal de contas, poderíamos ter a seguinte situação:

Estrutura de classes com herança modeladas com UML.

Neste exemplo, temos a super classe Pessoa, e três subclasses Funcionario, Aluno e Professor.

Uma classe pode herdar apenas de uma classe (super classe). Quando uma classe não define explicitamente que está herdando outra classe então, esta classe é filha de java.lang.Object, ou seja, a classe Object é a classe pai de todas as classes.

Por Object ser pai de todas as classes, todas as classes herdam os seguintes métodos dela:

Métodos herdados da classe Object.

Quando lidamos com classes que possuem a relação de herança, podemos fazer uso de duas palavras-chave que servem para identificar se estamos utilizando um método e ou atributo da classe atual ou de sua super classe. Estes comandos são:

this = Define que o recurso pertence à classe atual.

super = Define que o recurso pertence à super classe.

Podemos visualizar que a classe Coordenador utiliza ambas as palavras-chaves:

package material.heranca;

/**
 * Classe utilizada para representar um Coordenador que é uma Pessoa.
 */
public class Coordenador extends Pessoa {
    private String cursoCoordenado;

    public Coordenador(String nome, int matricula, String cursoCoordenado) {
        super(nome, matricula);
        this.cursoCoordenado = cursoCoordenado;
    }

    public String getCursoCoordenado() { return cursoCoordenado; }
    public void setCursoCoordenado(String cursoCoordenado) {
        this.cursoCoordenado = cursoCoordenado;
    }
}

O construtor que recebe o nome, matrícula e curso do coordenador. Note que neste construtor temos a chamada super(nome, matricula), que irá chamar o construtor da classe Pessoa que recebe um String e um inteiro como parâmetro.

Dentro deste mesmo construtor temos a seguinte chamada this.cursoCoordenado = cursoCoordenado. Utilizando a palavra-chave this, referenciaremos o atributo cursoCoordenador da própria classe Coordenador.