+2

PHP 5.3 parte 3: Lambda e Closures

criado por Douglas V. Pasqua em 05/01/2010 2:14pm
Closures

O recurso de Lambda não adiciona nenhuma funcionalidade em especial em relação a versão 4, onde podemo reproduzir o uso de funções anônimas através da função create_function. É ai que entra em cena, no php 5.3, o recurso chamado Closures. Closures funcionam como Lambdas, porém de forma mais inteligente. Permitem a interação com variáveis fora do ambiente de onde são definidos.

Veja abaixo um exemplo bem simples da utilização de Closures:

<?php
$string
= "Olá teste!";
$closure
= function() use ($string) { echo $string; };

$closure
();

A saída do código anterior:

Olá teste!

Variáveis de fora do ambiente do closure podem ser importadas através da palavra-chave use. No exemplo anterior a variável $string foi importada para dentro do closure. Por padrão as variáveis importadas são passadas por valor e não por referência. Dessa forma se quisermos que a variável importada seja modificada dentro do closure, e que a mudança continue aplicada após a chamada da função, devemos utilizar o operador & para especificar que a variável esta sendo passada por referência e não por valor.

Segue abaixo mais um exemplo da utilização de closures importando variáveis por referência:

<?php
$a
= 10;
$closure
= function() use (&$a) { $a += 50; };

$closure
();
var_dump
($a);

$closure
();
var_dump
($a);

$closure
();
var_dump
($a);

A saída do script anterior:

int(60)
int(110)
int(160)

Comentários:

Mostrando 1 - 3 de 3 comentários
Respondendo a questão: Afinal, onde este método seria útil ?

O método __invoke pode ser usado para que uma classe se comporte como um closure. Imagino que a principal utilidade do closure seria em funções que aceitam callback como parâmetro. Funções como array_map, usort, etc.

Vamos supor que você queira criar diversas funções de comparação para utilizar em uma rotina de ordenação, como usort. Você pode criar diversas classes implementando o método __invoke e passar as instâncias dessas classes como parâmetro para a função usort.

Veja o exemplo abaixo para entender melhor:

<?php

class testeInvoke {
       
public function __invoke($a, $b) {
               
return(strcmp($a, $b));
       
}
}

$o
= new testeInvoke();
$array
= array ("Ziraldo", "Luiz", "Joao", "Rodrigo", "Alberto");

usort
($array, $o);
var_dump
($array);

Criamos uma classe testeInvoke implementando o método mágico __invoke e passamos a sua instância para a função usort ordernar o array.

O resultado do script anterior:

# php invoke.php
array
(5) {
 
[0]=>
 
string(7) "Alberto"
 
[1]=>
 
string(4) "Joao"
 
[2]=>
 
string(4) "Luiz"
 
[3]=>
 
string(7) "Rodrigo"
 
[4]=>
 
string(7) "Ziraldo"
}
06/01/2010 6:05pm (~15 anos atrás)

Aliás, o __invoke só é chamado quando a instância é invocada com os parênteses né? Acho que o primeiro exemplo do "$validacao = validacao()" nem funcionaria...
05/01/2010 9:40pm (~15 anos atrás)

Estive pensando: será que seria uma boa usar o __invoke para retornar a instância de uma classe singleton?

Algo assim:
$validacao = validacao::getInstance();

Ficaria apenas assim:
$validacao = validacao();

Não sei se isso tornaria o código menos intuitivo, já que "getInstance" é consagrado.

Mas se formos parar pra pensar: sempre que se usar o __invoke perderemos um pouco da legibilidade, afinal, estaremos omitindo o nome de um método. Afinal, onde este método seria útil?

Quem sabe para métodos que retornam objetos...
// Obtendo um usuario
$usuario
= $alguma_coisa->usuario;

// Obtendo um usuario com codigo 1
$usuario
= $alguma_coisa->usuario(3);

Enfim, as vezes o PHP tras umas coisas que nos faz pensar nas novas possibilidades...
05/01/2010 9:35pm (~15 anos atrás)

Novo Comentário:

(Você pode usar tags como <b>, <i> ou <code>. URLs serão convertidas para links automaticamente.)