sábado, 16 de julho de 2016

Iterando sobre uma matriz com um único loop

Quando estamos nas aulas iniciais de programação, aprendemos a iterar sobre vetores e matrizes, sendo que no segundo caso é necessário criar um loop (for, while e etc) para cada dimensão de nossa matriz. Desta forma criamos muitas vezes um código com vários níveis de aninhamento no nosso código.

Uma outra forma de lidar com isto é criando uma forma de a partir de um único índice extrair o valores dos sub-índices, uma tarefa simples se utilizarmos módulo.

domingo, 18 de maio de 2014

Encurtando links em Java usando o Migre.me

Uma pequena forma de como usar o Migre.me dentro de sua aplicação para encurtar URLs. A requisição é feita utilizando alguns componentes da especificação JAX-RS, utilizando o Jersey como implementação.

Para fazer parser do JSON recebido, utilizamos o Jackson, o que deixa as coisas muito simples.

domingo, 16 de março de 2014

Dando um close no Java 7: AutoCloseable

Gerenciar recursos sempre foi algo muito chato, principalmente quando parte do ciclo de vida do objeto é fecha-lo, senão você poderá ter problemas na execução do seu programa. Vamos por exemplo analisar um código que abre um arquivo e itera sobre ele exibindo seu conteúdo linha a linha:
        FileReader readerAntigo = null;
        try {
            readerAntigo = new FileReader("arquivo.txt");
            Scanner scanner = new Scanner(readerAntigo);

        while(scanner.hasNext()) {
            System.out.println(scanner.next());
        }
        } catch (FileNotFoundException e ) {
            System.out.println("Arquivo não encontrado");
        } finally {
            try {
                readerAntigo.close();
            } catch (IOException e) {
                System.out.println("Erro ao fechar o arquivo");
            }
        }
Perceba como é chato gerenciar o arquivo, para que eu possa abrir e fechar são necessário duas capturas de sessão, sem contar que ainda é necessário que o programador lembre-se de fechar o arquivo, algo que não pode ser tão crítico, mas se imaginarmos conexões JDBC, logo o banco começará a recusar conexões pois esquecemos de fecha-las anteriormente.

Para ajudar a solucionar isso, no Java 7 foi introduzi o AutoCloseable, que permite que recursos existam apenas dentro de um bloco e ao seu final (ou caso seja lançado uma exceção) sejam fechados devidamente. Para isso, foi criada a interface AutoCloseable, e o a classe que a implementada de sobrescrever o método close(). Ao abrir um novo recurso, você deve declara-lo e instancia-lo dentro de parentes após o try, ficando assim:

quinta-feira, 6 de março de 2014

Mapeando e Reduzindo - Java 8

As vezes precisamos processar dados em um volume maior, onde é necessário filtrar a informação, selecionar ou alterar o que queremos para depois realizar o processamento. Como exemplo, vamos imaginar uma lista de alunos, onde cada instância tem seu nome e nota. Desta lista queremos a média geral, a média dos alunos aprovados (ou seja, quem tirou nota seis ou mais) e a média dos reprovados (nota menor que seis). Para isso, vamos primeiro visualizar nossa lista:

        List<Aluno>alunos = new ArrayList<>();

        Aluno aluno1 = new Aluno("Cebolinha", 8.0);
        Aluno aluno2 = new Aluno("Cascao", 6.5);
        Aluno aluno3 = new Aluno("Monica", 9.0);
        Aluno aluno4 = new Aluno("Magali", 3.0);

        alunos.addAll(Arrays.asList(aluno1, aluno2, aluno3, aluno4));

Apesar da simplicidade dos nossos dados, para poder separar a nossa lista de forma a realizar esta média, precisamos iterar sobre todos os seus elementos. Vamos fazer primeiro a média mais simples, sem nenhum critério:

        Double media = new Double("0");
        int quantidadeDeAlunos = 0;

        for (Aluno aluno : alunos) {
            media = Double.sum(media, aluno.getNota());
            quantidadeDeAlunos++;
        }

        System.out.println(media / quantidadeDeAlunos);

domingo, 2 de março de 2014

Filtrando resultados (Predicados) - Java 8

Muitas vezes precisamos filtrar listas, listas de todos os tipos, desde números básicos, passando por Strings, até tipos complexos. Abaixo temos um exemplo simples disto, onde temos uma lista com alguns nomes e queremos filtra-la e exibi-la para o usuário (no caso, todos os nomes que tiverem a letra "u" serão exibidos):

        List<string> nomes = new ArrayList<>();

        nomes.add("João da Silva");
        nomes.add("Joaquim da Padaria");
        nomes.add("Lucas Polo");
        nomes.add("Chaves");

        List<string> nomesFiltrados = new ArrayList<>();

        System.out.println("Iniciando filtragem");

        for (String nome : nomes) {
            System.out.println("Avaliando: " + nome);
            if (nome.contains("u")) nomesFiltrados.add(nome);
        }

        System.out.println("Iniciando impressão");

        for (String nome : nomesFiltrados) {
            System.out.println(nome);
        }
A saida é:
Iniciando filtragem
Avaliando: João da Silva
Avaliando: Joaquim da Padaria
Avaliando: Lucas Polo
Avaliando: Chaves
Iniciando impressão
Joaquim da Padaria
Lucas Polo
Repare que primeiros os resultados são filtrados e depois exibidos, mas variando com o tamanho da lista, pode demorar muito tempo para filtrarmos uma lista inteira, e por isso, muitas vezes, é melhor avaliar se o elemento passa no filtro e trata-lo devidamente, ganhando assim mais velocidade. Na programação funcional, temos um meio de lidar com isso, chamado avaliação preguiçosa.

Lembrando que métodos também pode ser passados como parâmetros, por isso, podemos armazena-los e utilizar suas funções posteriormente. Métodos que realizam a avaliação de elementos, são conhecidos como predicados, que no Java é representado pela classe Predicate e contem o método filter.

segunda-feira, 23 de setembro de 2013

Inversão de Controle e Injeção de Dependências

Quando comecei a programar na escola (curso técnico que fiz na época), comecei em uma das linguagens mais legais e, de certo modo, estranha que existe, a linguagem C. Alguns conceitos dela não são claros logo de inicio, pois não fazem nenhum paralelo com o mundo real. Um deles é o conceito de ponteiros e alocação de memória, onde você dinamicamente pode alocar memória para seu programa utilizar. Porém, como dizia o Tio Ben: "Grandes poderes trazem grandes responsabilidades", logo você se via obrigado a "lembrar" de liberar toda a memória alocada quando não fizesse mais uso dela, senão seu programa logo travaria junto com o seu sistema operacional.

O mundo mudou, os tempos agora são outros, temos a internet, o relógio digital e o Garbage Collector, que faz toda a gerência da coleta de memória que não é mais utilizada por nossos objetos. Porém agora temos que gerênciar outros recursos, como objetos que abrem arquivos e devem fecha-los ou conexões com o banco de dados. Caso você não os feche, seu arquivo não poderá ser aberto novamente ou seu banco de dados começará a recusar conexões, e sinceramente, isso é muito chato.

domingo, 22 de setembro de 2013

Métodos como parâmetros - Java 8

O Java 8 que será lançado em breve e já possui versões de teste disponíveis para download tem uma novidade interessante, que é tratar métodos como objetos (criando assim o suporte a funções Lambda). Uma vez que métodos podem ser tratados como objetos, você pode passa-los como parâmetro ou cria-los em tempo de execução.

Vamos dissecar mais sobre essas novidades no decorrer do tempo, mas por enquanto vamos ver um exemplo do que podemos fazer. Antigamente, para ler um arquivo e imprimir o seu conteúdo linha a linha poderíamos fazer algo assim:

public class LeitorDeArquivo {
    public static void main(String[] args) {
    
        try(Scanner scanner = 
                new Scanner(new FileInputStream("/Users/lucaspolo/teste"))) {
                
            while(scanner.hasNext()) {
                System.out.println(scanner.next());
            }
            
        } catch (FileNotFoundException ex) {
            System.err.println("Problemas ao abrir e ler arquivo: " + ex.getMessage());
        }
}
Basicamente estamos dizendo o seguinte enquanto houver um próximo elemento, imprima este próximo elemento. No Java 8 podemos deixar menor ainda este trecho de código, invertendo a maneira como lidamos com este tipo de instrução, agora podemos dizer para os elementos seguintes, faça a seguinte operação, conforme o código abaixo, que possui o mesmo funcionamento:

public class LeitorDeArquivo {
    public static void main(String[] args) {
    
        try(Scanner scanner = 
                new Scanner(new FileInputStream("/Users/lucaspolo/teste"))) {
                
            scanner.forEachRemaining(System.out::println);
            
        } catch (FileNotFoundException ex) {
            System.err.println("Problemas ao abrir e ler arquivo: " + ex.getMessage());
        }
}
Repare que estamos passando o método println do objeto out da classe estática System como parâmetro. Perceba que o operador :: é responsável por esse passagem da referência do (a partir do Java 8) objeto println. 

O segredo ocorre no tratamento desse parâmetro recebido, ele converte em uma interface e injeta o seu objeto que no caso é uma String. No futuro veremos qual interface é essa e seu funcionamento por dentro.