Category Archives: Linguagens

ExtJS e programação funcional – 2

[disclaimer]
Os códigos desse post estão no Gist do Github, se não aparece no seu leitor de Feeds vai ter que entrar no site ou ir direto para o github
[/disclaimer]

Continuando a falar sobre programação funcional com o Framework ExtJS, vou avançar sobre a API que fornece funções úteis para trabalhar sob esse paradigma e quando a abordagem complica a leitura, principalmente para quem não tem tanta intimidade com essa forma de pensar.

Imagine o seguinte Widget abaixo que tem a responsabilidade de plugar uma função para observar o evento busca de outro componente.

Ext.define("Sorteio", {
extend: "Ext.panel.Panel",
alias: 'widget.sorteio',
layout: "border",
/* ignora demais confs */
reiniciar: function(values) {
this.concorrentes.listar(values);
},
initComponent: function() {
this.callParent();
this.sorteioform.on("busca", this.reiniciar, this);
}
});
view raw a.js hosted with ❤ by GitHub

Se voce observar atentamente, o mapeamento é feito um-para-um com uma função que já existe no componente final, inclusive com a mesma quantidade de parâmetros, vimos no artigo passado que bastaria plugar a função diretamente e controlar o escopo this dessa função.

Ext.define("Sorteio", {
/* ignora confs */
initComponent: function() {
this.callParent();
this.sorteioform.on("busca",
this.concorrentes.listar,
this.concorrentes);
}
});
view raw b.js hosted with ❤ by GitHub

Agora imagine que o componente Sorteio tem mais uma responsabilidade no momento que o botão de outro componente for acionado, ele precisaria limpar a área de um terceiro componente, teríamos que voltar o código do primeiro exemplo e fazer aquele mapeamento em um método do próprio componente Sorteio:

Ext.define("Sorteio", {
reiniciar: function(values) {
this.concorrentes.listar(values);
this.ganhadores.update("");
},
initComponent: function() {
this.callParent();
this.sorteioform.on("busca", this.reiniciar, this);
}
});
view raw c.js hosted with ❤ by GitHub

Bem, com as funções encontradas no objeto Ext.Function podemos mapear diretamente os métodos dos responsáveis principais sem a necessidade de uma terceira função no objeto Sorteio numa abordagem mais FP aproveitando funções úteis que encadeam execuções e retornam outra funções com a sequência desejada.

A assinatura de colocar um listener escutando um determinado evento é:

this.sorteioform.on("busca", fn, escopo);

Desejamos executar duas funções de objetos distintos em uma sequência lógica, mas só podemos plugar uma única função por vez. Existe um método createSequence que fornece esse comportamento desejado, observe:

var fn = Ext.Function.createSequence(fn1, fn2, escopo);

Esse método gerará uma terceira função com o this dentro dela referente ao escopo passado no terceiro argumento e executará as duas funções – fn1 e fn2 – na sequência indicada.

Poderíamos simplesmente encadear as duas funções dos dois objetos na assinatura

var fn = Ext.Function.createSequence(this.concorrentes.listar, this.ganhadores.update, this.concorrentes);

Mas tem somente um problema, a função this.ganhadores.update precisa receber um parâmetro – no mínimo uma string vazia “” – para ter o comportamento adequado.

Existe uma outra função chamada pass que gera uma outra função com essa característica, voce pode definir uma função com valores previamente definidos caso não haja passagem de parâmetros.

var fn2 = Ext.Function.pass(this.ganhadores.update, "");

Dessa forma basta substituir agora a segunda função da sequência por uma gerada com valores predefinidos.


var fn2 = Ext.Function.pass(this.ganhadores.update, "");
var fn = Ext.Function.createSequence(this.concorrentes.listar, fn2, this.concorrentes);

Para garantir que o update executará no escopo de ganhadores, teríamos que definir o terceiro argumento para ganhadores

var fn = Ext.Function.createSequence(this.concorrentes.listar, fn2, this.ganhadores);

E para garantir que o this no listener execute cada função da sequência nos seus contextos corretos voce define o escopo do on para concorrentes
this.sorteioform.on("busca", fn, this.concorrentes);

O resultado final seria um encadeamento das chamadas como podemos ver logo em seguida:

this.sorteioform.on("busca",
Ext.Function.createSequence(
this.concorrentes.listar,
Ext.Function.pass(this.ganhadores.update, ""),
this.ganhadores),
this.concorrentes);
view raw d.js hosted with ❤ by GitHub

Se voce comparar com uma abordagem mais tradicional verá que nem sempre é mais fácil de ler, portanto é salutar dosar o uso desse tipo de solução.

ExtJS e programação funcional

[disclaimer]
Os códigos desse post estão no Gist do Github, se não aparece no seu leitor de Feeds vai ter que entrar no site ou ir direto para o github
[/disclaimer]

Javascript possui funções como tipos de primeira classe na linguagem e implementa vários conceitos de programação funcional, mas essa forma de programar sempre é relegada quando escrevemos código com ExtJS.

Observe no código abaixo um trecho usando ExtJS para expandir as linhas de uma Grid:

//a partir de uma grid, obtém um plugin que tem a
// capacidade de expandir as linhas com mais informações
var plugin = grid.getPlugin('expandplugin');
// uma grid possui uma entidade do tipo Store que
// representa uma coleção de entidades de negócio, no Framework chamadas de Model
var store = grid.getStore();
//percorre cada Model nesse repositório
for(i = 0; i < store.getCount(); i++)
//aciona um método que expande a linha passando o index de cada linha;
plugin.toggleRow(i);
view raw a.js hosted with ❤ by GitHub
Código imperativo comum encontrado nos projetos com ExtJS, o mesmo código conhecendo um pouco a API pode ser feito como se vê abaixo:
var plugin = grid.getPlugin('expandplugin');
var store = grid.getStore();
//método que percorre cada um dos Models contidos no repositório
// e passa como parâmetro para a função configurada como argumento desse método.
store.each( function(model, index, id) {
plugin.toggleRow(index);
})
view raw b.js hosted with ❤ by GitHub

Você percebe que utilizando uma abordagem só um pouco mais funcional (como passar função como argumento de outra função) nem sempre vai ter menos código e pode até ser bem maior, mas observando a API com mais atenção você detecta que o método toggleRow pode receber tanto um index quanto o próprio Model, então você abusa mais um pouquinho e passa a própria função como argumento do método each (como podemos ver abaixo).

var plugin = grid.getPlugin('expandplugin');
//o método recebe dois argumentos, o segundo é
//justamente de quem será o this na função passada
//como o primeiro parâmetro
grid.getStore().each( plugin.toggleRow, plugin)
view raw c.js hosted with ❤ by GitHub
Comparando os dois códigos você pode até reclamar que a sintaxe imperativa vai ser mais fácil de ler, aí será questão de conhecimento em programação e experiência com essas outras abordagens, reconheço que programação funcional não é comum principalmente para quem programa com ExtJS no cotidiano.
var plugin = grid.getPlugin('expandplugin');
//comparando as duas abordagens
//assim
for(i = 0; i < grid.getStore().getCount(); i++) plugin.toggleRow(i);
//ou assim
grid.getStore().each( plugin.toggleRow, plugin)
view raw d.js hosted with ❤ by GitHub

Apesar de tudo vale a pena se esforçar um pouco e começar a escrever um código mais funcional.

Desenvolver em Java em pleno 2012, mesmos erros de 2005

Post já nasce datado, mas só faz sentido para agora mesmo. Passei uns 10 anos da minha vida programando na linguagem Java e nos últimos 3 anos eu peguei poucos projetos, mas o que me impressiona nesses poucos projetos é que as coisas não mudam, inclusive a tara por patterns desnecessários e antipatterns.

Comecemos por Nomenclatura

Se voce chama sua classes de WhateverController, WhateverService e ou WhateverDAO, voce está usando notação hungara desnecessária e complicando a modelagem do seu negócio. Se o seu framework te obriga a nomear as classes com sufixos ou prefixos, ele está errado e é melhor procurar uma solução.

Se voce chama classes como WhateverModel ou WhateverEntity aí voce está estragando a amizade, se mate.

Se voce tem uma classe chamada Whatever e tem propriedades como nameWhatever, leia urgente Clean Code.

 

BOLOVO

As pessoas criavam entidades chamadas WhateverManager por não saberem orientação a objetos, se existe isso no seu projeto na maioria das vezes não tem muito o que fazer, mude de emprego ou de projeto. Mas… se for corajoso comece a refatorar isso guiado por testes, o livro “Growing Object-Oriented Software, Guided by Tests” vai te ajudar bastante. O Paulo Silveira e o Phillip Calçado nomearam esse anti-pattern de BOLOVO.

DAO

DAO é o pattern inútil quando falamos de negócios, principalmente CRUD. A não ser que você esteja codando Framework ou comittando em projetos como o Hibernate, voce não precisa escrever DAO. Voce usa Hibernate, a Session é seu DAO.

Se voce precisa de uma entidade para agrupar alguma lógica de ORM mais complexa no seu negócio – como algumas transações com rollback lógicos, uma alternativa é Repository. Mas por favor, leia o artigo do Phillip primeiro e não faça WhateverRepository. Não há problema nenhum voce ter Criteria dentro de um controller por exemplo, afinal isso é um pattern bem estabelecido e o mapeamento um-pra-um com outra entidade só vai complicar e não traz ganho algum.

Só uma dica aproveitando o tema ORM, o Hibernate trabalha e sempre trabalhou com convenções usando anotações, então não precisa mapear tudo. Basta um @Entity na maioria das vezes. Nos relacionamentos observe se ele já não mapeia tranquilo apenas com o @ManyToOne e diminua o ruído.

Em termos de Patterns, por mais caduco que já esteja, o PoEAA do Fowler ainda reina.

 

Interface e Implementação

Existe uma boa prática como guia que é desenvolver orientado a interface, só que isso não é lei e deve ser usado o bom senso como sempre. A maioria dos desenvolvedores criam a Interface Whatever e uma – e apenas uma – implementação WhateverImpl. Isso é desnecessário e muita gente nem sabe que o Spring sempre funcionou injetar em classes concretas e não apenas em Interface. Deixe a Interface gritar na sua cara para refatorar.

Service e Domain Driven Design

Aqui que mora o perigo, sempre quando eu vejo WhateverService a implementação dessa classe é o mesmo código do antigo WhateverManager. Depois que Domain Driven Design fez sucesso todo mundo finje que modela o domínio.

As classes do seu domínio devem e podem ter métodos de negócios, se ela apenas tem propriedades é sinal do BOLOVO e representa uma tabela do banco de dados vitaminada. Um Service não é a parte de negócios do seu domain, a grosso modo de explicar o código dele explodiu na sua cara por manipular duas ou mais entidades e não ser responsabilidade de nenhuma delas.

ANUNCIO EM LETRAS GARRAFAIS

Cuidado com os livros que eu indiquei, quando foram escritos o Hibernate e o Spring estavam nascendo ou ainda não tinham nascidos, portanto leia com moderação. Várias coisas já forma implementadas pelos Frameworks e não vá fazer uma roda por cima de outra roda.

TL;DR

Cuidado com o código que voce escreve, faça-o guiado por testes, leia bons livros de Orientação a objetos e Patterns. Não escreva código igual aos outros porque é assim que todo mundo faz.