Refatoração com
Capitão Nascimento
    Eduardo Bregaida
Eduardo Bregaida
•   http://javawora.blogspot.com


•   @bregaida


•   eduardo.bregaida@gmail.com
O Conceito de
    Refatoração
Por Martin Fowler...
É o processo de
reescrever um programa
   de computador para
melhorar sua estrutura ou
legibilidade, preservando
         assim seu
   comportamento.
Teve sua origem nos anos
  80/90 com Smalltalk...
Desenvolveu-
   se formalmente na
Universidade de Illinois em
   Urbana-Champaign.
 Grupo do Prof. Ralph
       Johnson...
Capitão?
Pois não?
Seu 05 está dormindo
Seu 05
Senhor?
Tenha a bondade
 de segurar isto.
Seu 05 se o senhor dormir
o senhor vai se explodir, vai
me explodir...
E vai explodir a todos seus
  companheiros do FISL.
O senhor vai dormir seu 05?
Não Senhor!
Estamos todos confiando no
          senhor.
Vamos analisar a motivação
       de refatorar
Por que refatorar?
Mudará algo na geração dos
        bytecodes?
Mudará algo para o
  computador?
Mudará algo para o cliente?
Não!
Facilitará para o senhor
     no caso de uma
  manutenção lembrar
 facilmente do código
Facilitará para outros
desenvolvedores entender o
    que o código faz
Manutenibilidade
Um exemplo de código
      senhores
Dado um array de N elementos, calcular a seguinte operações:
Dado um array de N elementos, calcular a seguinte operações:


   * Primeiro item menos o último
Dado um array de N elementos, calcular a seguinte operações:


   * Primeiro item menos o último
   * O resultado dessa subtração elevado ao quadrado
   somado ao segundo item menos o penúltimo
Dado um array de N elementos, calcular a seguinte operações:


   * Primeiro item menos o último
   * O resultado dessa subtração elevado ao quadrado
   somado ao segundo item menos o penúltimo

   * O resultado dessa subtração elevado ao
   quadrado.
Dado um array de N elementos, calcular a seguinte operações:


   * Primeiro item menos o último
   * O resultado dessa subtração elevado ao quadrado
   somado ao segundo item menos o penúltimo

   * O resultado dessa subtração elevado ao
   quadrado.
   * uma seqüência com N elementos
Dado um array de N elementos, calcular a seguinte operações:


   * Primeiro item menos o último
   * O resultado dessa subtração elevado ao quadrado
   somado ao segundo item menos o penúltimo

   * O resultado dessa subtração elevado ao
   quadrado.
   * uma seqüência com N elementos

   * (1o - N) ^2 + (2o - (N-1)) ^2 + ... + (N/2 - N/2)^2
Dado um array de N elementos, calcular a seguinte operações:


   * Primeiro item menos o último
   * O resultado dessa subtração elevado ao quadrado
   somado ao segundo item menos o penúltimo

   * O resultado dessa subtração elevado ao
   quadrado.
   * uma seqüência com N elementos

   * (1o - N) ^2 + (2o - (N-1)) ^2 + ... + (N/2 - N/2)^2
   * Retornar essa soma ou "-1" caso a entrada seja
   inválida
public class Calcula
{
	   // contador do array
     static int count=-1;

       // método que faz o calculo solicitado
       public int somaSerie (int array [])
       {
           if(null == array)
               return -1;
           else if(++count>=array.length/2) // se chegou na metade, retorna
zero
               return 0;

           // aplica a fórmula com chamada recursiva
           return ((array[count]-array[array.length-(count+1)])*
                   (array[count]-array[array.length-(count+1)])) + somaSerie
(array);
    }

// Teste aqui?
    public static void main(String args[])
    {
        int ar[] = {1,2,3,4,5,6,7,8,9,10,11};
        System.out.println("Resultado=" + new Calcula().somaSerie(ar));
    }
}
Fácil de entender senhores?
Não...
Vamos ver como a
refatoração o melhora
public class CalculaRef {
	   //Constante do valor Metade
	   private static final int VALOR_METADE = 2;
	
	   // contador do array
	   static int count = -1;
	   private int[] array;
	   private int tamanho = 0;

	   // retorno para array nulo
	   final int ENTRADA_INVALIDA = -1;

	   public CalculaRef(int[] array) {
	   	    this.array = array;
	   	    tamanho = array.length;
	   }

	   // mesmo cálculo efetuado na classe Calcula
	   // mas as operações difíceis de ler
	   // foram transformadas em métodos
	   public int somaSerie() {
	   	    if (array.length == 0)
	   	    	    return ENTRADA_INVALIDA;
	   	    else if (isMaiorQueMetade(++count))
	   	    	    return 0;

	   	   int subtracaoValores = getPrimeiroItemArrayMenosUltimo();
	   	   return (getPotenciaQuadrado(subtracaoValores) + somaSerie());
	   }

	   public int getPotenciaQuadrado(int numeroOperacao) {
	   	    return numeroOperacao * numeroOperacao;
	   }

	   public int getPrimeiroItemArrayMenosUltimo() {
	   	    return array[count] - array[tamanho - (count + 1)];
	   }

	   public boolean isMaiorQueMetade(int indice) {
	   	    return indice >= tamanho / VALOR_METADE;
	   }

}
/**
  * Classe de Teste
  */
public class CalculaRefTeste {
	
	 public static void main(String args[]) {
	 	 int ar[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11 };
	 	 CalculaRef refTeste = new CalculaRef(ar);
	 	 System.out.println("Resultado=" + refTeste.somaSerie());
	 }
}
Podemos melhorar mais...
public class CalculaRefDois {

	   // contador deixou de ser estático e passou a ser um atributo
	   private int count = -1;

	   // array possui agora uma classe para efetuar suas operações específicas
	   private ArrayUtil arrayUtil;
	   final int ENTRADA_INVALIDA = -1;

	   public CalculaRefDois(int[] array) {
	   	   // inicializa classe do array
	   	   this.arrayUtil = new ArrayUtil(array);
	   }

	   // ficou quase igual ao método da classe CalculaRefHum a diferença
     // é que o IF e o ELSE IF chamam métodos da classe nova a ArrayUtils
	   public int somaSerie() {
	   	   if (arrayUtil.getTamanho() == 0)
	   	   	    return ENTRADA_INVALIDA;
	   	   else if (arrayUtil.isMaiorQueMetade(++count))
	   	   	    return 0;

	   	   int subtracaoValores = getPrimeiroItemArrayMenosUltimo();
	   	   return (getPotenciaQuadrado(subtracaoValores) + somaSerie());
	   }

	   public int getPotenciaQuadrado(int valorSubtracao) {
	   	   // Utilizando a lib Math do Java na qual já cálcula um valor passado e seu expoente
	   	   //Cast de int pois pow retorna double
	   	   return (int) Math.pow(valorSubtracao,EXPOENTE_QUADRADO);
	   }

	   public int getPrimeiroItemArrayMenosUltimo() {
	   	   return arrayUtil.getItem(count) - arrayUtil.getItemIndiceInverso(count);
	   }
}
public class ArrayUtil {

    	   private int array[];
    	   private int tamanho = 0;
	       private static final int VALOR_HUM = 1;
	       //Constante do valor Metade
	       private static final int VALOR_METADE = 2;

    	   public ArrayUtil(int array[]) {
    	   	   this.array = array;
    	   	   if (null == array)
    	   	   	   this.array = new int[] {};
    	   	   tamanho = this.array.length;
    	   }

    	   public int getIndiceIntermediario() {
    	   	   return tamanho / VALOR_METADE;
    	   }

    	   public int getTamanho() {
    	   	   return tamanho;
    	   }

    	   public int getItem(int indice) {
    	   	   return array[indice];
    	   }

    	   public int getItemIndiceInverso(int indice) {
    	   	   return array[tamanho - (indice + VALOR_HUM)];
    	   }

    	   public boolean isMaiorQueMetade(int indice) {
    	   	   return indice >= getIndiceIntermediario();
    	   }
    }
Senhor, o que é preciso para
         refatorar?
Xerife, o senhor precisa
     testar o código
Xerife, o senhor precisa
sentir o “mau cheiro” do
         código!
O que seria sentir o “mau
cheiro” do código senhor?
Classes com muitas
 responsabilidades
Um código sem padrão
        algum
Sistemas sem alta
  coesão e baixo
   acoplamento
Ah, mas isso dá muito
    trabalho senhor
Xerife
O senhor é um fanfarrão
      seu Xerife
Os senhores têm
que refatorar sempre
public class PontoHandler {
	    private Ponto ponto = new Ponto();
	    private Session session;
     private DAO<Ponto> dao;
	    	
	    public void salvar() {
	    	    session = HibernateUtil.getCurrentSession();
	    	    dao = new DAO<Ponto>(session, Ponto.class);
	    	    if (ponto.getId() == 0) {
	    	    	     dao.salveOrUpdate(this.ponto);
	    	    } else {
	    	    	     session.merge(this.ponto);
	    	    }
	    	    this.ponto = new Ponto();

	    }

	    public void carregar(ActionEvent event) {
	    	    UIComponent comandLink = event.getComponent();
	    	    UIParameter parameter = (UIParameter) comandLink
	    	    	    	     .findComponent("editId");
	    	    Long id = (Long) parameter.getValue();
	    	    session = HibernateUtil.getCurrentSession();
	    	    dao = new DAO<Ponto>(session, Ponto.class);
	    	    this.ponto = dao.load(id);

	    }

	    public void excluir(ActionEvent event) {
	    	    UIComponent comandLink = event.getComponent();
	    	    UIParameter parameter = (UIParameter) comandLink
	    	    	    	     .findComponent("editId");
	    	    Long id = (Long) parameter.getValue();
	    	    session = HibernateUtil.getCurrentSession();
	    	    DAO<Ponto> dao = new DAO<Ponto>(session, Ponto.class);
	    	    this.ponto = dao.load(id);
	    	    dao.deleta(ponto);
	    	    this.ponto = new Ponto();

	    }

//Outros métodos

}
Eliminem a duplicidade
Dêem nomes decentes
   aos métodos e os
     parâmetros
public class PontoHandler {
	   private Ponto ponto = new Ponto();
	   private PontoBusiness negocio = new PontoBusiness();	
	   private Long id ;
	
	   public void salvar() {
	   	    if (ponto.getId() == 0) {
	   	    	    negocio.salvar(this.ponto);
	   	    } else {
	   	    	    negocio.merge(this.ponto);
	   	    }
	   	    this.ponto = new Ponto();

	   }

	   public void carregar(ActionEvent event) {
	   	    id = populaDados(event);
	   	    this.ponto = negocio.carregar(id);
	   }

	   public void excluir(ActionEvent event) {
	   	    id = populaDados(event);
	   	    this.ponto = negocio.carregar(id);
	   	    negocio.excluir(ponto);
	   	    this.ponto = new Ponto();
	   }
	
	   public List<Ponto> getAllPonto() {
	   	    return negocio.getAllPontos();
	   }

	   /**
	    * Popula os dados dos componentes JSF
	    * @param event
	    * @return id
	    */
	   private Long populaDados(ActionEvent event)
	   {
	   	    UIComponent comandLink = event.getComponent();
	   	    UIParameter parameter = (UIParameter) comandLink.findComponent("editId");
	   	    Long id = (Long) parameter.getValue();
	   	    return id;
	   }

    //Outros métodos
}
Os nomes dos
 métodos e parâmetros
tem que ser condizentes
 com o que eles fazem
public class PontoHandler {

	   private Ponto ponto = new Ponto();

	   private Session session;

    private DAO<Ponto> dao;
	   	
	   public void salvar() {
	     //Implementação
	   }



	   public void carregar(ActionEvent event) {
	     //Implementação
	   }



	   public void excluir(ActionEvent event) {
	     //Implementação
	   }



}
Os métodos e classes
 devem ser pequenos
public class PontoHandler {
	    private Ponto ponto = new Ponto();
	    private Session session;
     private DAO<Ponto> dao;
	    	
	    public void salvar() {
	    	    session = HibernateUtil.getCurrentSession();
	    	    dao = new DAO<Ponto>(session, Ponto.class);
	    	    if (ponto.getId() == 0) {
	    	    	     dao.salveOrUpdate(this.ponto);
	    	    } else {
	    	    	     session.merge(this.ponto);
	    	    }
	    	    this.ponto = new Ponto();

	    }

	    public void carregar(ActionEvent event) {
	    	    UIComponent comandLink = event.getComponent();
	    	    UIParameter parameter = (UIParameter) comandLink
	    	    	    	     .findComponent("editId");
	    	    Long id = (Long) parameter.getValue();
	    	    session = HibernateUtil.getCurrentSession();
	    	    dao = new DAO<Ponto>(session, Ponto.class);
	    	    this.ponto = dao.load(id);

	    }

	    public void excluir(ActionEvent event) {
	    	    UIComponent comandLink = event.getComponent();
	    	    UIParameter parameter = (UIParameter) comandLink
	    	    	    	     .findComponent("editId");
	    	    Long id = (Long) parameter.getValue();
	    	    session = HibernateUtil.getCurrentSession();
	    	    DAO<Ponto> dao = new DAO<Ponto>(session, Ponto.class);
	    	    this.ponto = dao.load(id);
	    	    dao.deleta(ponto);
	    	    this.ponto = new Ponto();

	    }

//Outros métodos

}
public class PontoHandler {
	   private Ponto ponto = new Ponto();
	   private PontoBusiness negocio = new PontoBusiness();	
	   private Long id ;
	
	   public void salvar() {
	   	   if (ponto.getId() == 0) {
	   	   	    negocio.salvar(this.ponto);
	   	   } else {
	   	   	    negocio.merge(this.ponto);
	   	   }
	   	   this.ponto = new Ponto();

	   }

	   public void carregar(ActionEvent event) {
	   	   id = populaDados(event);
	   	   this.ponto = negocio.carregar(id);
	   }

	   public void excluir(ActionEvent event) {
	   	   id = populaDados(event);
	   	   this.ponto = negocio.carregar(id);
	   	   negocio.excluir(ponto);
	   	   this.ponto = new Ponto();
	   }
	
	   /**
	    * Popula os dados dos componentes JSF
	    * @param event
	    * @return id
	    */
	   private Long populaDados(ActionEvent event)
	   {
	   	   UIComponent comandLink = event.getComponent();
	   	   UIParameter parameter = (UIParameter) comandLink.findComponent("editId");
	   	   Long id = (Long) parameter.getValue();
	   	   return id;
	   }

    //Outros métodos
}
Crie
comentários no
   código
Utilize o JavaDoc
/**
  * @author eduardobregaida
  * PontoHandler
  * ManagedBean para fazer a ponte entre a View e o Controller
  */
public class PontoHandler {
	
	    // Classe de Ponto do ônibus
	    private Ponto ponto = new Ponto();

	   // Classe da Linha do ônibus
	   private Linha linha = new Linha();
	
	   // Método para Salvar um Ponto
	   public void salvar() {
	   	   if (ponto.getId() == 0) {
	   	   	   negocio.salvar(this.ponto);
	   	   } else {
	   	   	   negocio.merge(this.ponto);
	   	   }
	   	   this.ponto = new Ponto();

	   }

// Método para excluir um ponto recebe um evento do componente
	   	
	   public void excluir(ActionEvent event) {
	   	   id = populaDados(event);
	   	   this.ponto = negocio.carregar(id);
	   	   negocio.excluir(ponto);
	   	   this.ponto = new Ponto();
	   }

}
Tire as responsabilidades
     a mais nas classes
public class PontoHandler {
	    private Ponto ponto = new Ponto();
	    private Session session;
     private DAO<Ponto> dao;
	    	
	    public void salvar() {
	    	    session = HibernateUtil.getCurrentSession();
	    	    dao = new DAO<Ponto>(session, Ponto.class);
	    	    if (ponto.getId() == 0) {
	    	    	     dao.salveOrUpdate(this.ponto);
	    	    } else {
	    	    	     session.merge(this.ponto);
	    	    }
	    	    this.ponto = new Ponto();

	    }

	    public void carregar(ActionEvent event) {
	    	    UIComponent comandLink = event.getComponent();
	    	    UIParameter parameter = (UIParameter) comandLink
	    	    	    	     .findComponent("editId");
	    	    Long id = (Long) parameter.getValue();
	    	    session = HibernateUtil.getCurrentSession();
	    	    dao = new DAO<Ponto>(session, Ponto.class);
	    	    this.ponto = dao.load(id);

	    }

	    public void excluir(ActionEvent event) {
	    	    UIComponent comandLink = event.getComponent();
	    	    UIParameter parameter = (UIParameter) comandLink
	    	    	    	     .findComponent("editId");
	    	    Long id = (Long) parameter.getValue();
	    	    session = HibernateUtil.getCurrentSession();
	    	    DAO<Ponto> dao = new DAO<Ponto>(session, Ponto.class);
	    	    this.ponto = dao.load(id);
	    	    dao.deleta(ponto);
	    	    this.ponto = new Ponto();

	    }

//Outros métodos

}
public class PontoBusiness
{
	   private Session session;
    private DAO<Ponto> dao;
    private Ponto ponto;
	
	   public void salvar(Ponto ponto) {
	   	   session = HibernateUtil.getCurrentSession();
	   	   dao = new DAO<Ponto>(session, Ponto.class);
	   	   dao.salveOrUpdate(ponto);
	   }
	
	   public void merge(Ponto ponto) {
	   	   session = HibernateUtil.getCurrentSession();
	   	   dao = new DAO<Ponto>(session, Ponto.class);
	   	   session.merge(ponto);
	   }

	   public Ponto carregar(Long id) {
	   	   ponto = new Ponto();
	   	   session = HibernateUtil.getCurrentSession();
	   	   dao = new DAO<Ponto>(session, Ponto.class);
	   	   ponto = dao.load(id);
	   	   return ponto;
	   }

	   public void excluir(Ponto ponto) {
	   	   session = HibernateUtil.getCurrentSession();
	   	   DAO<Ponto> dao = new DAO<Ponto>(session, Ponto.class);
	   	   dao.deleta(ponto);
	   }
	
	   //Outros métodos
}
Não exponha o
  interior dos seus
objetos, encapsulem
   seus métodos
Esta classe apenas deve saber que o método soma recebe dois valores



public class CalculadoraTeste {

    int valorA = 1;
	 int valorB = 2;
	
//	Exemplo de DRY (Don't repeat yourself)
	 public static void main(String[] args) {
	 	
	 	 Calculadora calculadora = new Calculadora();
	 	 int resultado = calculadora.soma(valorA, valorB);
	 	 System.out.println("Resultado: " + resultado);
      }

}
O Método está em outra classe deixando invisível para classe acima



public class Calculadora {

	 public int soma(int valorA, int valorB) {
	 	 return valorA + valorB;
	 }
}
Quando usar números
   crie constantes
public class Calcula {
	
	 // retorno para array nulo
	 final int ENTRADA_INVALIDA = -1;
	 final int VALOR_ZERO = 0;



        public int somaSerie() {
	   	   if (array.length == VALOR_ZERO)
	   	   	 return ENTRADA_INVALIDA;
	   	   else if (isMaiorQueMetade(++count))
	   	   	 return VALOR_ZERO;
        }

	 // Outros métodos da classe

}
Utiliza a Herança
quando necessário
public class Funcionario {

	   private String nome;
	   private double salario;
	   private int idade;
	   private int tempoRegistro;
	   protected String cpf;

	 // Getters e Setters

}
public class Gerente extends Funcionario {
	 private int senha;
	 private int numeroDeFuncionariosGerenciados;

	 // Getters e Setters

	
}
public class GerenteTeste {

	   public static void main(String[] args) {
	   	 Funcionario funcionario = new Funcionario();
	   	 funcionario.setNome("Carlos Bergamasco");
	   	 funcionario.setSalario(5000.0);
	   	    System.out.println(funcionario.getNome()+" "+funcionario.getBonificacao());
	   	
	   	    Gerente gerente = new Gerente();
	   	    // podemos chamar metodos do Funcionario:
	   	    gerente.setNome("Eduardo Bregaida");
	   	    // e tambem metodos do Gerente!
	   	    gerente.setSenha(4231);
	   	    gerente.autentica(gerente.getSenha());
	   	    gerente.setSalario(5000.0);
	   	    System.out.println(gerente.getNome()+" "+gerente.getBonificacao());

	   	
	   }	
}
Utilize e prefira
 Polimorfismo
public abstract class Funcionario {

	   private String nome;
	   private double salario;
	   private int idade;
	   private int tempoRegistro;
	   protected String cpf;
	   private int senha;

	
	 // Getters e Setters

}
public class Gerente extends Funcionario {
	
	 private int numeroDeFuncionariosGerenciados;

	
	 // Getters e Setters

}
public class Vendedor extends Funcionario {
	 private long quantidadesVenda;

	
	 // Getters e Setters
	
}
public class PolimorfismoTeste {

	   public static void main(String[] args) {

	   	   Gerente gerente = new Gerente();
	   	   gerente.setNome("Eduardo Bregaida");
	   	   gerente.setSenha(4231);
	   	   gerente.autentica(gerente.getSenha());
	   	   gerente.setSalario(5000.0);
	   	   System.out.println("Gerente "+gerente.getNome()+" "+gerente.getBonificacao());
	   	
	   	   Vendedor vendedor = new Vendedor();
	   	   vendedor.setNome("Consani");
	   	   vendedor.setSalario(52);
	   	   System.out.println("Vendedor "+vendedor.getNome()+" "+vendedor.getSalario());


	   	   Funcionario funcionario = new Gerente();
	   	   funcionario.setNome("Carlos Bergamasco");
	   	   funcionario.getBonificacao();
	   	   funcionario.setSalario(44.0);
	   	   System.out.println("Funcionario Gerente "+funcionario.getNome()+" "+funcionario.getSalario());

	   	
	   	   funcionario = new Vendedor();
	   	   funcionario.setNome("Carlos Bergamasco");
	   	   funcionario.getBonificacao();
	   	   funcionario.setSalario(44.0);
	   	   System.out.println("Funcionario Vendedor "+funcionario.getNome()+" "+funcionario.getSalario());

	   	
	   	
	   }
}
Utilize a Interface
public abstract class Funcionario {

	   private String nome;
	   private double salario;
	   private int idade;
	   private int tempoRegistro;
	   protected String cpf;

	
	 // Getters e Setters

}
public interface Autenticavel {
	
	 public boolean autentica(int senha);
	
}
public class Gerente extends Funcionario
                         implements Autenticavel {
	   private int senha;

	   // assinatura do contrato pela interface
	   public boolean autentica(int senha) {
	   	 if (this.senha != senha) {
	   	 	 System.out.println("Acesso Permitido!");
	   	 	 return true;
	   	 } else {
	   	 	 System.out.println("Acesso Negado!");
	   	 	 return false;
	   	 }

	   }
}
Façam testes unitários
public class StringUtil {
	   public static String recuperaNomeAtributo(String nomeMetodo){
	   	   if(nomeMetodo==null) throw new IllegalArgumentException("Agurmento não pode ser nulo");
	   	   return nomeMetodo;
	   }
	
	   public static String recuperaNomeAtributoNaoPodeSerVazio(String nomeMetodo){
	   	   if(nomeMetodo.equals("")) throw new IllegalArgumentException("Agurmento não pode         ser vazio");
	   	   return nomeMetodo;
	   }
	
	   public static String recuperaNomeAtributoNaoPodeSerMenorQue4(String nomeMetodo){
	   	   if(nomeMetodo.length()<4)
             throw new IllegalArgumentException("Agurmento não pode ser menor do que 4 caracteres");
	   	   return nomeMetodo;
	   }
	
	   public static String recuperaNomeAtributoNaoPodeTerLetraMaiuscula(String nomeMetodo){
	   	   String maiuscula="ABCDEFGHIJKLMNOPQRSTUVWXYZ";
	   	   	       for(int i=0; i<nomeMetodo.length(); i++){
	   	   	          if (maiuscula.indexOf(nomeMetodo.charAt(i),0)!=-1){
	   	   	        	   throw new IllegalArgumentException("Agurmento não pode ser nulo");
	   	   	          }
	   	   	       }
	         return nomeMetodo;
	   }

}
import junit.framework.TestCase;
import br.com.cb.jUnitTes2.StringUtil;


public class StringUtilTest extends TestCase{


	   public void testRecuperaNomeAtributoNaoPodeSerNulo() throws Exception{
	   	   assertEquals("nome", StringUtil.recuperaNomeAtributo("nome"));
	   }
	
	   public void testRecuperaNomeAtributoNaoPodeSerVazio()throws Exception{
	   	   assertEquals("XPTO", StringUtil.recuperaNomeAtributoNaoPodeSerVazio("XPTO"));
	   	
	   }
	
	   public void testRecuperaNomeAtributoNaoPodeSerMenorQue4()throws Exception{
	   	   assertEquals("Abcd", StringUtil.recuperaNomeAtributoNaoPodeSerMenorQue4("Abcd"));
	   }
	
	   public void testRecuperaNomeAtributoNaoPodeTerLetraMaiuscula()throws Exception{
	   	   assertEquals("teste funfando", StringUtil.recuperaNomeAtributoNaoPodeTerLetraMaiuscula("teste funfando"));

	   }
}
TDD - Test Driven
  Development
import static org.junit.Assert.assertEquals;
import org.junit.Test;
import br.com.cb.tdd.junit.Calculadora;

public class CalculadoraTeste {
	 Calculadora calculadora = new Calculadora();

	   @Test
	   public void deveriaSomarDoisValoresPassados() throws Exception {
	   	 assertEquals(3, calculadora.soma(1, 2));
	   }

}
public class Calculadora {

	   public int soma(int valorA, int valorB) {
	   	 return valorA + valorB;
	   }

}
import static org.junit.Assert.assertEquals;
import org.junit.Test;
import br.com.cb.tdd.junit.Calculadora;

public class CalculadoraTeste {
	 Calculadora calculadora = new Calculadora();

	   @Test
	   public void deveriaSomarDoisValoresPassados() throws Exception {
	   	 assertEquals(3, calculadora.soma(1, 2));
	   }

	   @Test
	   public void deveriaSubtrairDoisValoresPassados() throws Exception {
	   	 assertEquals(2, calculadora.subtracao(5, 3));
	   }

	
}
public class Calculadora {

	   public int soma(int valorA, int valorB) {
	   	 return valorA + valorB;
	   }

	   public int subtracao(int valorA, int valorB) {
	   	 return valorA - valorB;
	   }

}
import static org.junit.Assert.assertEquals;
import org.junit.Test;
import br.com.cb.tdd.junit.Calculadora;

public class CalculadoraTeste {
	 Calculadora calculadora = new Calculadora();

	   @Test
	   public void deveriaSomarDoisValoresPassados() throws Exception {
	   	 assertEquals(3, calculadora.soma(1, 2));
	   }

	   @Test
	   public void deveriaSubtrairDoisValoresPassados() throws Exception {
	   	 assertEquals(2, calculadora.subtracao(5, 3));
	   }

	   @Test
	   public void deveriaMultiplicarDoisValoresPassados() throws Exception
{
	   	   assertEquals(15, calculadora.multiplicacao(5, 3));
	   }
	
}
public class Calculadora {

	   public int soma(int valorA, int valorB) {
	   	 return valorA + valorB;
	   }

	   public int subtracao(int valorA, int valorB) {
	   	 return valorA - valorB;
	   }

	   public int multiplicacao(int valorA, int valorB) {
	   	 return valorA * valorB;
	   }

}
import static org.junit.Assert.assertEquals;
import org.junit.Test;
import br.com.cb.tdd.junit.Calculadora;

public class CalculadoraTeste {
	 Calculadora calculadora = new Calculadora();

	   @Test
	   public void deveriaSomarDoisValoresPassados() throws Exception {
	   	 assertEquals(3, calculadora.soma(1, 2));
	   }

	   @Test
	   public void deveriaSubtrairDoisValoresPassados() throws Exception {
	   	 assertEquals(2, calculadora.subtracao(5, 3));
	   }

	   @Test
	   public void deveriaMultiplicarDoisValoresPassados() throws Exception
{
	   	   assertEquals(15, calculadora.multiplicacao(5, 3));
	   }
	
	   @Test
	   public void deveriaDividirDoisValoresPassados() throws Exception {
	   	 assertEquals(3, calculadora.divisao(9, 3));
	   }
}
public class Calculadora {

	   public int soma(int valorA, int valorB) {
	   	 return valorA + valorB;
	   }

	   public int subtracao(int valorA, int valorB) {
	   	 return valorA - valorB;
	   }

	   public int multiplicacao(int valorA, int valorB) {
	   	 return valorA * valorB;
	   }

	   public int divisao(int valorA, int valorB) {
	   	 return valorA / valorB;
	   }
}
Esse negócio aí de
   testar é chato
 demais e demora
O senhor seu Xerife
     é um fraco
Seu lugar não é
    aqui com
 profissionais
Eu desisto senhor!
Os senhores estão
   deixando seu
 capitão muito feliz
Testes mostram
    qualidade
Qualidade mostra que
     irá funcionar
Teste é o que
    diferencia
programadores de
    crianças
Estude Mocks
Há diversos
 frameworks que
facilitam o uso de
      mocks
Estude teste de
   Integração
Quer subir seu
código no controle de
      versão?
Sem teste?
Não vai subir
  ninguém
Vai todo mundo
ficar quietinho aí e
       testar
Se não testar...
Já avisei
Isso vai dar merda
Se os senhores não
   fizerem isso...
Bota ele no saco
Se persistir no erro...
Pede pra sair
Pede pra sair
Tira essa roupa
preta, porque você
não merece usar
Se não sair e não mudar
Bota na conta do papa
Dúvidas?
•   http://www.guj.com.br/


•

•
    http://blog.caelum.com.br/


    http://www.caelum.com.br/apostilas/
                                                               Para Estudar
•   http://www.klauslaube.com.br/wp-content/uploads/2011/01/
    TestDrivenGameDevelopment.png


•   http://improveit.com.br/xp/praticas/tdd


•   http://www.testdriven.com/


•   http://javafree.uol.com.br/noticia/3772/
•   http://www.caelum.com.br/apostilas/


•   http://martinfowler.com/books.html


•   http://www.slideshare.net/jeveaux/testes-e-refatorao      Para Estudar
•   http://www.amazon.com/Clean-Code-Handbook-Software-Craftsmanship/dp/0132350882


•   http://www.slideshare.net/cassiomarques/refatorao-design-patterns-em-rubyhttp://compare.buscape.com.br/
    refatoracao-aperfeicoando-o-projeto-de-codigo-existente-martin-fowler-8536303956.html


•   http://www.ime.usp.br/~kon/presentations/


•   http://ccsl.ime.usp.br/pt-br/palestras


•   http://www.slideshare.net/guestd37c23/refactory-worshop
• Agradecimentos
 • Carlos Daniel Bergamasco
 • Braulio Consani
 • Christian Reichel
 • Marcelo L. Z. Ingarano
 • Adriana A. Gutierre
 • Rogério Ap. Bregaida Junior
Fim




                               Obrigado!

eduardo.bregaida@gmail.com
http://javawora.blogspot.com
 http://www.slideshare.net/
       eduardo.bregaida
          @bregaida

Refatoração de código com Capitão Nascimento versão completa

  • 1.
  • 2.
    Eduardo Bregaida • http://javawora.blogspot.com • @bregaida • eduardo.bregaida@gmail.com
  • 5.
    O Conceito de Refatoração Por Martin Fowler...
  • 8.
    É o processode reescrever um programa de computador para melhorar sua estrutura ou legibilidade, preservando assim seu comportamento.
  • 11.
    Teve sua origemnos anos 80/90 com Smalltalk...
  • 14.
    Desenvolveu- se formalmente na Universidade de Illinois em Urbana-Champaign. Grupo do Prof. Ralph Johnson...
  • 17.
  • 19.
  • 21.
    Seu 05 estádormindo
  • 24.
  • 26.
  • 29.
    Tenha a bondade de segurar isto.
  • 33.
    Seu 05 seo senhor dormir o senhor vai se explodir, vai me explodir...
  • 35.
    E vai explodira todos seus companheiros do FISL.
  • 37.
    O senhor vaidormir seu 05?
  • 39.
  • 41.
  • 44.
    Vamos analisar amotivação de refatorar
  • 47.
  • 49.
    Mudará algo nageração dos bytecodes?
  • 51.
    Mudará algo parao computador?
  • 53.
  • 55.
  • 57.
    Facilitará para osenhor no caso de uma manutenção lembrar facilmente do código
  • 59.
    Facilitará para outros desenvolvedoresentender o que o código faz
  • 61.
  • 63.
    Um exemplo decódigo senhores
  • 66.
    Dado um arrayde N elementos, calcular a seguinte operações:
  • 67.
    Dado um arrayde N elementos, calcular a seguinte operações: * Primeiro item menos o último
  • 68.
    Dado um arrayde N elementos, calcular a seguinte operações: * Primeiro item menos o último * O resultado dessa subtração elevado ao quadrado somado ao segundo item menos o penúltimo
  • 69.
    Dado um arrayde N elementos, calcular a seguinte operações: * Primeiro item menos o último * O resultado dessa subtração elevado ao quadrado somado ao segundo item menos o penúltimo * O resultado dessa subtração elevado ao quadrado.
  • 70.
    Dado um arrayde N elementos, calcular a seguinte operações: * Primeiro item menos o último * O resultado dessa subtração elevado ao quadrado somado ao segundo item menos o penúltimo * O resultado dessa subtração elevado ao quadrado. * uma seqüência com N elementos
  • 71.
    Dado um arrayde N elementos, calcular a seguinte operações: * Primeiro item menos o último * O resultado dessa subtração elevado ao quadrado somado ao segundo item menos o penúltimo * O resultado dessa subtração elevado ao quadrado. * uma seqüência com N elementos * (1o - N) ^2 + (2o - (N-1)) ^2 + ... + (N/2 - N/2)^2
  • 72.
    Dado um arrayde N elementos, calcular a seguinte operações: * Primeiro item menos o último * O resultado dessa subtração elevado ao quadrado somado ao segundo item menos o penúltimo * O resultado dessa subtração elevado ao quadrado. * uma seqüência com N elementos * (1o - N) ^2 + (2o - (N-1)) ^2 + ... + (N/2 - N/2)^2 * Retornar essa soma ou "-1" caso a entrada seja inválida
  • 73.
    public class Calcula { // contador do array static int count=-1; // método que faz o calculo solicitado public int somaSerie (int array []) { if(null == array) return -1; else if(++count>=array.length/2) // se chegou na metade, retorna zero return 0; // aplica a fórmula com chamada recursiva return ((array[count]-array[array.length-(count+1)])* (array[count]-array[array.length-(count+1)])) + somaSerie (array); } // Teste aqui? public static void main(String args[]) { int ar[] = {1,2,3,4,5,6,7,8,9,10,11}; System.out.println("Resultado=" + new Calcula().somaSerie(ar)); } }
  • 75.
  • 77.
  • 79.
    Vamos ver comoa refatoração o melhora
  • 81.
    public class CalculaRef{ //Constante do valor Metade private static final int VALOR_METADE = 2; // contador do array static int count = -1; private int[] array; private int tamanho = 0; // retorno para array nulo final int ENTRADA_INVALIDA = -1; public CalculaRef(int[] array) { this.array = array; tamanho = array.length; } // mesmo cálculo efetuado na classe Calcula // mas as operações difíceis de ler // foram transformadas em métodos public int somaSerie() { if (array.length == 0) return ENTRADA_INVALIDA; else if (isMaiorQueMetade(++count)) return 0; int subtracaoValores = getPrimeiroItemArrayMenosUltimo(); return (getPotenciaQuadrado(subtracaoValores) + somaSerie()); } public int getPotenciaQuadrado(int numeroOperacao) { return numeroOperacao * numeroOperacao; } public int getPrimeiroItemArrayMenosUltimo() { return array[count] - array[tamanho - (count + 1)]; } public boolean isMaiorQueMetade(int indice) { return indice >= tamanho / VALOR_METADE; } }
  • 82.
    /** *Classe de Teste */ public class CalculaRefTeste { public static void main(String args[]) { int ar[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11 }; CalculaRef refTeste = new CalculaRef(ar); System.out.println("Resultado=" + refTeste.somaSerie()); } }
  • 84.
  • 86.
    public class CalculaRefDois{ // contador deixou de ser estático e passou a ser um atributo private int count = -1; // array possui agora uma classe para efetuar suas operações específicas private ArrayUtil arrayUtil; final int ENTRADA_INVALIDA = -1; public CalculaRefDois(int[] array) { // inicializa classe do array this.arrayUtil = new ArrayUtil(array); } // ficou quase igual ao método da classe CalculaRefHum a diferença // é que o IF e o ELSE IF chamam métodos da classe nova a ArrayUtils public int somaSerie() { if (arrayUtil.getTamanho() == 0) return ENTRADA_INVALIDA; else if (arrayUtil.isMaiorQueMetade(++count)) return 0; int subtracaoValores = getPrimeiroItemArrayMenosUltimo(); return (getPotenciaQuadrado(subtracaoValores) + somaSerie()); } public int getPotenciaQuadrado(int valorSubtracao) { // Utilizando a lib Math do Java na qual já cálcula um valor passado e seu expoente //Cast de int pois pow retorna double return (int) Math.pow(valorSubtracao,EXPOENTE_QUADRADO); } public int getPrimeiroItemArrayMenosUltimo() { return arrayUtil.getItem(count) - arrayUtil.getItemIndiceInverso(count); } }
  • 87.
    public class ArrayUtil{ private int array[]; private int tamanho = 0; private static final int VALOR_HUM = 1; //Constante do valor Metade private static final int VALOR_METADE = 2; public ArrayUtil(int array[]) { this.array = array; if (null == array) this.array = new int[] {}; tamanho = this.array.length; } public int getIndiceIntermediario() { return tamanho / VALOR_METADE; } public int getTamanho() { return tamanho; } public int getItem(int indice) { return array[indice]; } public int getItemIndiceInverso(int indice) { return array[tamanho - (indice + VALOR_HUM)]; } public boolean isMaiorQueMetade(int indice) { return indice >= getIndiceIntermediario(); } }
  • 89.
    Senhor, o queé preciso para refatorar?
  • 92.
    Xerife, o senhorprecisa testar o código
  • 94.
    Xerife, o senhorprecisa sentir o “mau cheiro” do código!
  • 97.
    O que seriasentir o “mau cheiro” do código senhor?
  • 100.
    Classes com muitas responsabilidades
  • 102.
    Um código sempadrão algum
  • 105.
    Sistemas sem alta coesão e baixo acoplamento
  • 108.
    Ah, mas issodá muito trabalho senhor
  • 111.
  • 113.
    O senhor éum fanfarrão seu Xerife
  • 116.
    Os senhores têm querefatorar sempre
  • 118.
    public class PontoHandler{ private Ponto ponto = new Ponto(); private Session session; private DAO<Ponto> dao; public void salvar() { session = HibernateUtil.getCurrentSession(); dao = new DAO<Ponto>(session, Ponto.class); if (ponto.getId() == 0) { dao.salveOrUpdate(this.ponto); } else { session.merge(this.ponto); } this.ponto = new Ponto(); } public void carregar(ActionEvent event) { UIComponent comandLink = event.getComponent(); UIParameter parameter = (UIParameter) comandLink .findComponent("editId"); Long id = (Long) parameter.getValue(); session = HibernateUtil.getCurrentSession(); dao = new DAO<Ponto>(session, Ponto.class); this.ponto = dao.load(id); } public void excluir(ActionEvent event) { UIComponent comandLink = event.getComponent(); UIParameter parameter = (UIParameter) comandLink .findComponent("editId"); Long id = (Long) parameter.getValue(); session = HibernateUtil.getCurrentSession(); DAO<Ponto> dao = new DAO<Ponto>(session, Ponto.class); this.ponto = dao.load(id); dao.deleta(ponto); this.ponto = new Ponto(); } //Outros métodos }
  • 120.
  • 122.
    Dêem nomes decentes aos métodos e os parâmetros
  • 124.
    public class PontoHandler{ private Ponto ponto = new Ponto(); private PontoBusiness negocio = new PontoBusiness(); private Long id ; public void salvar() { if (ponto.getId() == 0) { negocio.salvar(this.ponto); } else { negocio.merge(this.ponto); } this.ponto = new Ponto(); } public void carregar(ActionEvent event) { id = populaDados(event); this.ponto = negocio.carregar(id); } public void excluir(ActionEvent event) { id = populaDados(event); this.ponto = negocio.carregar(id); negocio.excluir(ponto); this.ponto = new Ponto(); } public List<Ponto> getAllPonto() { return negocio.getAllPontos(); } /** * Popula os dados dos componentes JSF * @param event * @return id */ private Long populaDados(ActionEvent event) { UIComponent comandLink = event.getComponent(); UIParameter parameter = (UIParameter) comandLink.findComponent("editId"); Long id = (Long) parameter.getValue(); return id; } //Outros métodos }
  • 126.
    Os nomes dos métodos e parâmetros tem que ser condizentes com o que eles fazem
  • 128.
    public class PontoHandler{ private Ponto ponto = new Ponto(); private Session session; private DAO<Ponto> dao; public void salvar() { //Implementação } public void carregar(ActionEvent event) { //Implementação } public void excluir(ActionEvent event) { //Implementação } }
  • 130.
    Os métodos eclasses devem ser pequenos
  • 132.
    public class PontoHandler{ private Ponto ponto = new Ponto(); private Session session; private DAO<Ponto> dao; public void salvar() { session = HibernateUtil.getCurrentSession(); dao = new DAO<Ponto>(session, Ponto.class); if (ponto.getId() == 0) { dao.salveOrUpdate(this.ponto); } else { session.merge(this.ponto); } this.ponto = new Ponto(); } public void carregar(ActionEvent event) { UIComponent comandLink = event.getComponent(); UIParameter parameter = (UIParameter) comandLink .findComponent("editId"); Long id = (Long) parameter.getValue(); session = HibernateUtil.getCurrentSession(); dao = new DAO<Ponto>(session, Ponto.class); this.ponto = dao.load(id); } public void excluir(ActionEvent event) { UIComponent comandLink = event.getComponent(); UIParameter parameter = (UIParameter) comandLink .findComponent("editId"); Long id = (Long) parameter.getValue(); session = HibernateUtil.getCurrentSession(); DAO<Ponto> dao = new DAO<Ponto>(session, Ponto.class); this.ponto = dao.load(id); dao.deleta(ponto); this.ponto = new Ponto(); } //Outros métodos }
  • 133.
    public class PontoHandler{ private Ponto ponto = new Ponto(); private PontoBusiness negocio = new PontoBusiness(); private Long id ; public void salvar() { if (ponto.getId() == 0) { negocio.salvar(this.ponto); } else { negocio.merge(this.ponto); } this.ponto = new Ponto(); } public void carregar(ActionEvent event) { id = populaDados(event); this.ponto = negocio.carregar(id); } public void excluir(ActionEvent event) { id = populaDados(event); this.ponto = negocio.carregar(id); negocio.excluir(ponto); this.ponto = new Ponto(); } /** * Popula os dados dos componentes JSF * @param event * @return id */ private Long populaDados(ActionEvent event) { UIComponent comandLink = event.getComponent(); UIParameter parameter = (UIParameter) comandLink.findComponent("editId"); Long id = (Long) parameter.getValue(); return id; } //Outros métodos }
  • 135.
  • 137.
  • 139.
    /** *@author eduardobregaida * PontoHandler * ManagedBean para fazer a ponte entre a View e o Controller */ public class PontoHandler { // Classe de Ponto do ônibus private Ponto ponto = new Ponto(); // Classe da Linha do ônibus private Linha linha = new Linha(); // Método para Salvar um Ponto public void salvar() { if (ponto.getId() == 0) { negocio.salvar(this.ponto); } else { negocio.merge(this.ponto); } this.ponto = new Ponto(); } // Método para excluir um ponto recebe um evento do componente public void excluir(ActionEvent event) { id = populaDados(event); this.ponto = negocio.carregar(id); negocio.excluir(ponto); this.ponto = new Ponto(); } }
  • 141.
    Tire as responsabilidades a mais nas classes
  • 143.
    public class PontoHandler{ private Ponto ponto = new Ponto(); private Session session; private DAO<Ponto> dao; public void salvar() { session = HibernateUtil.getCurrentSession(); dao = new DAO<Ponto>(session, Ponto.class); if (ponto.getId() == 0) { dao.salveOrUpdate(this.ponto); } else { session.merge(this.ponto); } this.ponto = new Ponto(); } public void carregar(ActionEvent event) { UIComponent comandLink = event.getComponent(); UIParameter parameter = (UIParameter) comandLink .findComponent("editId"); Long id = (Long) parameter.getValue(); session = HibernateUtil.getCurrentSession(); dao = new DAO<Ponto>(session, Ponto.class); this.ponto = dao.load(id); } public void excluir(ActionEvent event) { UIComponent comandLink = event.getComponent(); UIParameter parameter = (UIParameter) comandLink .findComponent("editId"); Long id = (Long) parameter.getValue(); session = HibernateUtil.getCurrentSession(); DAO<Ponto> dao = new DAO<Ponto>(session, Ponto.class); this.ponto = dao.load(id); dao.deleta(ponto); this.ponto = new Ponto(); } //Outros métodos }
  • 144.
    public class PontoBusiness { private Session session; private DAO<Ponto> dao; private Ponto ponto; public void salvar(Ponto ponto) { session = HibernateUtil.getCurrentSession(); dao = new DAO<Ponto>(session, Ponto.class); dao.salveOrUpdate(ponto); } public void merge(Ponto ponto) { session = HibernateUtil.getCurrentSession(); dao = new DAO<Ponto>(session, Ponto.class); session.merge(ponto); } public Ponto carregar(Long id) { ponto = new Ponto(); session = HibernateUtil.getCurrentSession(); dao = new DAO<Ponto>(session, Ponto.class); ponto = dao.load(id); return ponto; } public void excluir(Ponto ponto) { session = HibernateUtil.getCurrentSession(); DAO<Ponto> dao = new DAO<Ponto>(session, Ponto.class); dao.deleta(ponto); } //Outros métodos }
  • 146.
    Não exponha o interior dos seus objetos, encapsulem seus métodos
  • 148.
    Esta classe apenasdeve saber que o método soma recebe dois valores public class CalculadoraTeste { int valorA = 1; int valorB = 2; // Exemplo de DRY (Don't repeat yourself) public static void main(String[] args) { Calculadora calculadora = new Calculadora(); int resultado = calculadora.soma(valorA, valorB); System.out.println("Resultado: " + resultado); } }
  • 149.
    O Método estáem outra classe deixando invisível para classe acima public class Calculadora { public int soma(int valorA, int valorB) { return valorA + valorB; } }
  • 151.
    Quando usar números crie constantes
  • 153.
    public class Calcula{ // retorno para array nulo final int ENTRADA_INVALIDA = -1; final int VALOR_ZERO = 0; public int somaSerie() { if (array.length == VALOR_ZERO) return ENTRADA_INVALIDA; else if (isMaiorQueMetade(++count)) return VALOR_ZERO; } // Outros métodos da classe }
  • 155.
  • 157.
    public class Funcionario{ private String nome; private double salario; private int idade; private int tempoRegistro; protected String cpf; // Getters e Setters }
  • 158.
    public class Gerenteextends Funcionario { private int senha; private int numeroDeFuncionariosGerenciados; // Getters e Setters }
  • 159.
    public class GerenteTeste{ public static void main(String[] args) { Funcionario funcionario = new Funcionario(); funcionario.setNome("Carlos Bergamasco"); funcionario.setSalario(5000.0); System.out.println(funcionario.getNome()+" "+funcionario.getBonificacao()); Gerente gerente = new Gerente(); // podemos chamar metodos do Funcionario: gerente.setNome("Eduardo Bregaida"); // e tambem metodos do Gerente! gerente.setSenha(4231); gerente.autentica(gerente.getSenha()); gerente.setSalario(5000.0); System.out.println(gerente.getNome()+" "+gerente.getBonificacao()); } }
  • 161.
    Utilize e prefira Polimorfismo
  • 163.
    public abstract classFuncionario { private String nome; private double salario; private int idade; private int tempoRegistro; protected String cpf; private int senha; // Getters e Setters }
  • 164.
    public class Gerenteextends Funcionario { private int numeroDeFuncionariosGerenciados; // Getters e Setters }
  • 165.
    public class Vendedorextends Funcionario { private long quantidadesVenda; // Getters e Setters }
  • 166.
    public class PolimorfismoTeste{ public static void main(String[] args) { Gerente gerente = new Gerente(); gerente.setNome("Eduardo Bregaida"); gerente.setSenha(4231); gerente.autentica(gerente.getSenha()); gerente.setSalario(5000.0); System.out.println("Gerente "+gerente.getNome()+" "+gerente.getBonificacao()); Vendedor vendedor = new Vendedor(); vendedor.setNome("Consani"); vendedor.setSalario(52); System.out.println("Vendedor "+vendedor.getNome()+" "+vendedor.getSalario()); Funcionario funcionario = new Gerente(); funcionario.setNome("Carlos Bergamasco"); funcionario.getBonificacao(); funcionario.setSalario(44.0); System.out.println("Funcionario Gerente "+funcionario.getNome()+" "+funcionario.getSalario()); funcionario = new Vendedor(); funcionario.setNome("Carlos Bergamasco"); funcionario.getBonificacao(); funcionario.setSalario(44.0); System.out.println("Funcionario Vendedor "+funcionario.getNome()+" "+funcionario.getSalario()); } }
  • 168.
  • 170.
    public abstract classFuncionario { private String nome; private double salario; private int idade; private int tempoRegistro; protected String cpf; // Getters e Setters }
  • 171.
    public interface Autenticavel{ public boolean autentica(int senha); }
  • 172.
    public class Gerenteextends Funcionario implements Autenticavel { private int senha; // assinatura do contrato pela interface public boolean autentica(int senha) { if (this.senha != senha) { System.out.println("Acesso Permitido!"); return true; } else { System.out.println("Acesso Negado!"); return false; } } }
  • 174.
  • 176.
    public class StringUtil{ public static String recuperaNomeAtributo(String nomeMetodo){ if(nomeMetodo==null) throw new IllegalArgumentException("Agurmento não pode ser nulo"); return nomeMetodo; } public static String recuperaNomeAtributoNaoPodeSerVazio(String nomeMetodo){ if(nomeMetodo.equals("")) throw new IllegalArgumentException("Agurmento não pode ser vazio"); return nomeMetodo; } public static String recuperaNomeAtributoNaoPodeSerMenorQue4(String nomeMetodo){ if(nomeMetodo.length()<4) throw new IllegalArgumentException("Agurmento não pode ser menor do que 4 caracteres"); return nomeMetodo; } public static String recuperaNomeAtributoNaoPodeTerLetraMaiuscula(String nomeMetodo){ String maiuscula="ABCDEFGHIJKLMNOPQRSTUVWXYZ"; for(int i=0; i<nomeMetodo.length(); i++){ if (maiuscula.indexOf(nomeMetodo.charAt(i),0)!=-1){ throw new IllegalArgumentException("Agurmento não pode ser nulo"); } } return nomeMetodo; } }
  • 177.
    import junit.framework.TestCase; import br.com.cb.jUnitTes2.StringUtil; publicclass StringUtilTest extends TestCase{ public void testRecuperaNomeAtributoNaoPodeSerNulo() throws Exception{ assertEquals("nome", StringUtil.recuperaNomeAtributo("nome")); } public void testRecuperaNomeAtributoNaoPodeSerVazio()throws Exception{ assertEquals("XPTO", StringUtil.recuperaNomeAtributoNaoPodeSerVazio("XPTO")); } public void testRecuperaNomeAtributoNaoPodeSerMenorQue4()throws Exception{ assertEquals("Abcd", StringUtil.recuperaNomeAtributoNaoPodeSerMenorQue4("Abcd")); } public void testRecuperaNomeAtributoNaoPodeTerLetraMaiuscula()throws Exception{ assertEquals("teste funfando", StringUtil.recuperaNomeAtributoNaoPodeTerLetraMaiuscula("teste funfando")); } }
  • 179.
    TDD - TestDriven Development
  • 182.
    import static org.junit.Assert.assertEquals; importorg.junit.Test; import br.com.cb.tdd.junit.Calculadora; public class CalculadoraTeste { Calculadora calculadora = new Calculadora(); @Test public void deveriaSomarDoisValoresPassados() throws Exception { assertEquals(3, calculadora.soma(1, 2)); } }
  • 183.
    public class Calculadora{ public int soma(int valorA, int valorB) { return valorA + valorB; } }
  • 185.
    import static org.junit.Assert.assertEquals; importorg.junit.Test; import br.com.cb.tdd.junit.Calculadora; public class CalculadoraTeste { Calculadora calculadora = new Calculadora(); @Test public void deveriaSomarDoisValoresPassados() throws Exception { assertEquals(3, calculadora.soma(1, 2)); } @Test public void deveriaSubtrairDoisValoresPassados() throws Exception { assertEquals(2, calculadora.subtracao(5, 3)); } }
  • 186.
    public class Calculadora{ public int soma(int valorA, int valorB) { return valorA + valorB; } public int subtracao(int valorA, int valorB) { return valorA - valorB; } }
  • 188.
    import static org.junit.Assert.assertEquals; importorg.junit.Test; import br.com.cb.tdd.junit.Calculadora; public class CalculadoraTeste { Calculadora calculadora = new Calculadora(); @Test public void deveriaSomarDoisValoresPassados() throws Exception { assertEquals(3, calculadora.soma(1, 2)); } @Test public void deveriaSubtrairDoisValoresPassados() throws Exception { assertEquals(2, calculadora.subtracao(5, 3)); } @Test public void deveriaMultiplicarDoisValoresPassados() throws Exception { assertEquals(15, calculadora.multiplicacao(5, 3)); } }
  • 189.
    public class Calculadora{ public int soma(int valorA, int valorB) { return valorA + valorB; } public int subtracao(int valorA, int valorB) { return valorA - valorB; } public int multiplicacao(int valorA, int valorB) { return valorA * valorB; } }
  • 191.
    import static org.junit.Assert.assertEquals; importorg.junit.Test; import br.com.cb.tdd.junit.Calculadora; public class CalculadoraTeste { Calculadora calculadora = new Calculadora(); @Test public void deveriaSomarDoisValoresPassados() throws Exception { assertEquals(3, calculadora.soma(1, 2)); } @Test public void deveriaSubtrairDoisValoresPassados() throws Exception { assertEquals(2, calculadora.subtracao(5, 3)); } @Test public void deveriaMultiplicarDoisValoresPassados() throws Exception { assertEquals(15, calculadora.multiplicacao(5, 3)); } @Test public void deveriaDividirDoisValoresPassados() throws Exception { assertEquals(3, calculadora.divisao(9, 3)); } }
  • 192.
    public class Calculadora{ public int soma(int valorA, int valorB) { return valorA + valorB; } public int subtracao(int valorA, int valorB) { return valorA - valorB; } public int multiplicacao(int valorA, int valorB) { return valorA * valorB; } public int divisao(int valorA, int valorB) { return valorA / valorB; } }
  • 195.
    Esse negócio aíde testar é chato demais e demora
  • 198.
    O senhor seuXerife é um fraco
  • 200.
    Seu lugar nãoé aqui com profissionais
  • 202.
  • 205.
    Os senhores estão deixando seu capitão muito feliz
  • 209.
    Testes mostram qualidade
  • 211.
    Qualidade mostra que irá funcionar
  • 213.
    Teste é oque diferencia programadores de crianças
  • 215.
  • 217.
    Há diversos frameworksque facilitam o uso de mocks
  • 219.
    Estude teste de Integração
  • 222.
    Quer subir seu códigono controle de versão?
  • 224.
  • 227.
  • 229.
    Vai todo mundo ficarquietinho aí e testar
  • 232.
  • 234.
  • 236.
  • 239.
    Se os senhoresnão fizerem isso...
  • 241.
  • 245.
  • 248.
  • 250.
  • 253.
    Tira essa roupa preta,porque você não merece usar
  • 256.
    Se não saire não mudar
  • 258.
  • 263.
  • 264.
    http://www.guj.com.br/ • • http://blog.caelum.com.br/ http://www.caelum.com.br/apostilas/ Para Estudar • http://www.klauslaube.com.br/wp-content/uploads/2011/01/ TestDrivenGameDevelopment.png • http://improveit.com.br/xp/praticas/tdd • http://www.testdriven.com/ • http://javafree.uol.com.br/noticia/3772/
  • 265.
    http://www.caelum.com.br/apostilas/ • http://martinfowler.com/books.html • http://www.slideshare.net/jeveaux/testes-e-refatorao Para Estudar • http://www.amazon.com/Clean-Code-Handbook-Software-Craftsmanship/dp/0132350882 • http://www.slideshare.net/cassiomarques/refatorao-design-patterns-em-rubyhttp://compare.buscape.com.br/ refatoracao-aperfeicoando-o-projeto-de-codigo-existente-martin-fowler-8536303956.html • http://www.ime.usp.br/~kon/presentations/ • http://ccsl.ime.usp.br/pt-br/palestras • http://www.slideshare.net/guestd37c23/refactory-worshop
  • 266.
    • Agradecimentos •Carlos Daniel Bergamasco • Braulio Consani • Christian Reichel • Marcelo L. Z. Ingarano • Adriana A. Gutierre • Rogério Ap. Bregaida Junior
  • 267.
    Fim Obrigado! eduardo.bregaida@gmail.com http://javawora.blogspot.com http://www.slideshare.net/ eduardo.bregaida @bregaida