STATE_SAVING_METHOD - server ou client ?
Uma das dúvidas mais frequentes na lista de discussão do javasf está relacionado em como o JSF mantém o estado da árvore de componentes de uma view, qual a diferença e qual deve-se usar: client ou server? Acredito que a melhor maneira de responder a estas perguntas está em saber a diferença entre os dois métodos, e é exatamente isso que pretendo explanar mais abaixo..
A framework nos fornece dois métodos para mantermos o estado da árvore de componentes, os métodos são “client” e “server”, ou seja, é possível manter o estado dos componentes no cliente (página gerada) ou servidor (session).
Mas o que diacho é esse “estado da árvore de componentes”? Bem, no JSF os dados são armazenados em dois caminhos diferentes: dentro do escopo de beans e dentro da árvore de componentes. O escopo de beans(managed-beans) por si só é auto-explicativo. Já o escopo da árvore de componentes é onde estão armazenados os dados(estados) destes componentes, eles são armazenados no response e então são recuperados quando um request chega, com isso o JSF consegue manter o estado de cada componente inalterado até que se mude para uma nova página.
No método “client” o estado da árvore será persistido no cliente, para ser mais exacto ele será serializado e encodado na base64 e ficará em um campo do tipo HTML-input-hidden na página gerada para o cliente(browser). Já no método “server” o estado será persistido na sessão do usuário, ou seja, no servidor e não mais no cliente. Cada um possui suas vantagens e desvantagens, veremos isso mais a frente..
Você poderá facilmente definir qual método pretende usar inserindo no web.xml da sua aplicação o seguinte trecho de código:
<context-param>
<param-name>javax.faces.STATE_SAVING_METHOD</param-name>
<param-value>client</param-value>
</context-param>
No código acima o param-value poderá ser definido como server ou client, por default o método é o server. Bem, depois de definido qual método usar e re-startar a aplicação começaremos a sentir as diferenças.
Diferenças
Método server
- há um maior consumo de memória no lado servidor pois o estado esta sendo mantido na sessão por usuário;
- problemas com o back-button do browser (é necessário mais consumo de memória para resolver este problema);
- baixo consumo de banda na rede;
- baixo consumo de memória no cliente;
- uso de cpu no servidor é baixo;
- clustering - mover a sessão para um outro nó do cluster (muitos dados na sessão);
- melhora comunição via AJAX pois não é preciso reenviar o estado da view ao servidor;
- páginas carregam mais rapidamente (ou pode-se dizer normalmente?);
Método client
- menos uso da session e consequentemente menos consumo de memória no lado servidor;
- sem consumo de memória entre requests;
- corrige problemas de refresh (tecla F5) e botão voltar (back button) do browser;
- resolve problemas de concorrência;
- melhora escalabilidade da aplicação;
- alto consumo de banda da rede;
- maior overhead de cpu no servidor e cliente;
- maior consumo de memória no lado cliente;
- problemas de segurança;
- excelente na fase de desenvolvimento pois conseguimos manter o estado da view mesmo depois de restartar o servidor;
- páginas carregam ligeiramente mais lentas (isso depende muito da complexidade e quantidade de componentes na página);
* Os itens em vermelho indicam as desvantagens
Cada um traz consigo melhorias e problemas em determinados pontos, em alguns pontos há efeito contrário de acordo com o método escolhido, ou seja, se no método “server” algo funciona então pode ser que no método “client” pare de funcionar, e vice-versa. A maioria destes problemas podem ser resolvidos durante o desenvolvimento com algumas boas práticas, configurações, frameworks auxiliares e até truques, como por exemplo,
- pode-se diminuir o consumo de banda com a compactação do estado dos componentes antes de enviar ao cliente (método client);
- pode-se melhorar a segurança da aplicação encriptando o estado dos componentes antes de enviar ao cliente (método client);
- pode-se também diminuir o consumo de memória no lado servidor através de configurações mais detalhadas da framework, utilizando-se de frameworks como Spring-annotation, JBoss-Seam e até componentes como t:saveState do Myfaces-Tomahawk que nos fornecem “escopos” para mantermos os dados entre requests;
- pode-se corrigir o problema do back-button do browser aumentando o número de views por sessão (método server);
- etc
Estes são apenas alguns exemplos de como contornar tais problemas, algumas soluções o JSF já nos fornece com algumas configurações mais específicas, outras necessitaremos de frameworks auxiliares. No final das contas conseguimos contorna-los sem muitas dificuldades.
Então, qual devo usar?
Como podem ver há várias diferenças entre eles, mas dizer qual o melhor método é algo que vai depender muito do tipo de aplicação, quantidade de usuários e acessos, se é intranet ou não, entre outros detalhes do projeto. O ideal é conhecer a diferença entre eles e analisar qual método se encaixa melhor nos requisitos e necessidades do projeto.
Boa parte deste texto foi baseado nas discussões da lista javasf, leituras em blogs, fóruns, wikis, e principalmente nesta excelente apresentação, http://wiki.apache.org/myfaces-data/attachments/Performance/attachments/PerformanceBOFatJavaOne
Espero que esse texto sirva de alguma ajuda aos novatos na tecnologia e até alguns veteranos :))
Tags: back button, client, Java, JavaServer Faces, JBoss, JEE, JSF, method, performance, Seam, server, state saving, state saving method, STATE_SAVING_METHOD, t:saveState, web
October 15th, 2007 at 4:33 am
Meus parabéns Rafael, vc está escrevendo muito bem. As observações que tinha que fazer sobre o texto já te fiz, continue assim, sucesso.
October 15th, 2007 at 5:02 am
Boa Ponte….
Agora com um blog hein hehehe, muito bom….
Não deixe de nos avisar sempre que postar algo …..
Muito interessante este post, parabéns!
October 15th, 2007 at 6:16 am
Boa rafael ;D belo post, ta indo bem
October 16th, 2007 at 7:35 am
Não é querendo ser reclamão não, mas precisamos de artigos básicos que comecem explicando o início, tipo, o que é árvores de componentes antes de saber o que é estado das ditas
Faça um post ai sobre o ciclo de vida esse tipo de coisa.
October 19th, 2007 at 2:59 pm
Parabéns irmone. Muito foda.
Um dia chego lá.
kk
abraço
April 10th, 2008 at 4:41 pm
[...] coisa interessante é que por estarmos sempre trabalhando sob a mesma árvore de componentes (viewRoot) é necessário resetar o estado de alguns componentes ou bloco de componentes quando [...]
May 9th, 2008 at 1:53 pm
Só agora vi o post e achei muito bem explicado
Parabéns 
October 17th, 2008 at 6:51 am
Muito bom seu post!
Ressalvo o que foi dito em relação ao t:saveState, pois ele resolve problema de usar duas listas (selectManyListBox), na qual qualquer alteração que é feita em outra página, ao voltar para as listas, os dados não eram atualizados.
Quando eu tive esse problema fiquei dias procurando na internet até descobrir que o saveState (com o bean request) resolveria meu problema.
January 15th, 2009 at 5:40 am
Matou a pau!
March 16th, 2009 at 12:05 pm
Rafael, estou iniciando na área e tendo muito dos problemas que posta em seu blog…
um dia espero saber um pouco do que voc~e sabe….
parabéns pelo Blog.
abraço