Sobre (function(){ … }()) e callThis()

Pergunta :


O CoffeeScript compila seus arquivos dentro de um:

(function(){
  // código aqui
}()).callThis();

JSFiddle: http://jsfiddle.net/ZHrPp/

Algumas vezes encontramos na internet uma variação:

(function(){
  // código aqui
}());

JSFiddle: http://jsfiddle.net/pSK43/


Qual é a utilidade disso?

Autor da pergunta:

Resposta Sergio:

A principal utilidade destas IIFE (função imediatamente auto-invocada) é criar uma clausura para variáveis. Assim todas as variáveis defenidas dentro dessa clausura não são exportadas para o espaço global, e portanto não o poluíndo.

Essa funcionalidade tem a ver com o funcionamento normal das funções e como elas criam/alteram o escopo do código. O que é diferente aqui é que ele é imediatamente invocada/corrida e não fica à espera de ser chamada.

A outra razão importante é que as variaveis criadas ficam dentro daquele escopo e são libretadas da memoria depois da função correr, passando o seu valor para fora caso necessário (ver exemplo 2 em baixo).

Alguns exemplos práticos desta aplicação:

Exemplo 1:

(function(global){
  // código aqui
})(this);

Neste exemplo passamos para dentro da função o this do espaço global. No caso de javascript do lado do cliente isto será window mas no caso de Node.JS isto será o global.

Exemplo 2:

// #2479 - IE8 Cannot set HTML of style element
var canChangeStyleHtml = (function(){
    var div = document.createElement('style'),
        flag = false;
    try {
        div.innerHTML = '#justTesing{margin: 0px;}';
        flag = !!div.innerHTML;
    } catch(e){}
    return flag;
})();

Este exemplo em cima fiz para a biblioteca do MooTools para saber se o browser em que o código é corrido consegue mudar o HTML de um elemento <style>. Desta maneira a função corre (auto-invocada) retorna true ou false. Assim eu tenho essa informação guardada e posso usar no resto do meu código, e não preciso de correr mais essa função.

Exemplo 3:

var buttons = document.querySelectorAll('button');
for (var i = 0; i < buttons.length; i++) {
    (function () {
        var numero = Math.round(Math.random() * 20);
        buttons[i].addEventListener('click', function () {
            alert(numero);
        });
    })();
}

Este exemplo mostra um ciclo for no qual cada iteração corre uma IIFE. Essa função cria um escopo próprio que é passado e guardado. Assim a cada vez que clica num botão ele vai ter na memória o valor que lhe corresponde, sem ser re-escrito. Exemplo online: http://jsfiddle.net/dR8KG/, e já agora o mesmo exemplo sem a IIFE: http://jsfiddle.net/dR8KG/1/ – todos os botões têm o mesmo valor.

Related Posts:

Qual a diferença entre AppCompatActivity e Activity? – android android-activity
Pergunta: Qual a diferença da AppCompatActivity para Activity ? A partir de qual versão a AppCompatActivity foi adicionada ao Android? Autor da pergunta Luhhh A diferença reside ...
Como abreviar palavras em PHP? – php string
Pergunta: Possuo informações comuns como nome de pessoas e endereços, e preciso que elas contenham no máximo 30 caracteres sem cortar palavras. Exemplo: 'Avenida Natalino João Brescansin' ...
Qual é a finalidade de um parêntese vazio numa declaração Lambda? – c# expressões-lambda característica-linguagem
Pergunta: Criei um exemplo de uma declaração Lambda sem argumentos, entretanto, estou com duvidas referente a omissão do parêntese vazio () na declaração. Veja o exemplo: class ...
Boas práticas para URI em API RESTful – api rest restful
Pergunta: Estou com dúvida em relação às URIs de alguns recursos da api que estou desenvolvendo. Tenho os recursos projetos e atividades com relação 1-N, ...
Dúvidas sobre a integração do MySQL com Java – java mysql netbeans
Pergunta: Estou criando um sistema no NetBeans, utilizando a linguagem Java e o banco de dados MySQL. Escrevi o seguinte código para realizar a conexão ...
Qual é a finalidade da pasta Model do framework Inphinit? – php inphinit
Pergunta: No Inphinit micro-framework existe a pasta Model que fica dentro da pasta application, e nela é onde ficam as classes, mas eu estou muito ...
Uso do ‘@’ em variáveis – javascript typescript coffeescript
Pergunta: Vejo em algumas linguagens que compilam para javascript, como TypeScript e CoffeeScript, o uso do @ em variáveis, como também, casos em que o ...
Qual tamanho máximo um arquivo JSON pode ter? – json arquivo
Pergunta: Vou dar um exemplo para conseguir explicar minha duvida: Preciso recuperar informação de imagens vindas de uma API, esse banco de imagens me retorna JSON's ...
O que é Teste de Regressão? – terminologia engenharia-de-software testes
Pergunta: Na matéria de Teste de Software o professor abordou um termo chamado Teste de Regressão, isto dentro da disciplina de teste de software. Sendo ...
O que é um construtor da linguagem? – php característica-linguagem
Pergunta: Em PHP, já li e ouvi várias vezes a respeito dos Construtores da Linguagem. Os casos que sempre ouvi falar deles foi em casos ...
Função intrínseca para converter numérico para string – cobol
Pergunta: Estou a tentar saber se existe alguma função intrínseca do COBOL para converter um data numérico para string sem precisar usar a cláusula REDEFINES: ( ...
Porque usar implements? – java android
Pergunta: Qual a diferença entre usar btn.setOnClickListener(new OnClickListener() { e public class MainActivity extends Activity implements OnClickListener{ Estive fazendo um curso de Android e meu professor falou que ...
O que é XHTML e quando deve ser usado? – html xml xhtml
Pergunta: O que eu sei é que o XHTML precisa ser XML válido. Isso implica, por exemplo, que todas as tags precisam ser fechadas. Por ...
Uma placa aceleradora de vídeo pode melhorar o desempenho não-gráfico? [fechada] – desempenho
Pergunta: Para desenvolver em Ruby on Rails, eu utilizo aqui uma máquina virtual do VirtualBox com Ubuntu Server 14.04 sem interface gráfica instalada. Recentemente descobri uma ...
Concat() VS Union() – c# .net
Pergunta: Qual a diferença entre Concat() e Union() ? Quando usar Concat() e quando usar Union() ? Somente pode ser usado em list ? ...

Deixe uma resposta

O seu endereço de email não será publicado. Campos obrigatórios marcados com *