Closures в PHP: подробности
Внимательно прочитав документ по замыканиям в PHP, я вынес для себя следующее. Патч работает очень просто и довольно изящно. Замыкания — это не какой-то новый базовый тип, а экземпляр класса Closure. Причём, для того, чтобы объект вёл себя как функция, вводится новый «магический» метод __invoke:
class Example {
public function __invoke () {
echo "Hello World!\n";
}
}
$foo = new Example;
$foo ();
Слово «use», которое появилось в полной версии создания функции — это, конечно, уже полностью новый синтаксис, а не просто косметические изменения. Для тех, кто не прочитал документ, поясняю, что анонимная функция может быть создана, например, так:
function getAdder($x) {
return function ($y) use ($x) {
// or: lexical $x;
return $x + $y;
}
}
«Use» вытекает из идеологии PHP, где глобальные переменные не видны сразу внутри функции, а определяются программистом (тоже хаком, кстати). «Use» именно «доставляет» переменную. Причём не из глобального уровня, а с родительского уровня. Что (ура) реализует мою давнюю мечту об изоляции переменных.
Замыкания в PHP, эээх.
Я думаю, в один прекрасный день, ты откроешь для себя либо C++, либо какой-нибудь FPL. Чур на копрофагус не посылать.
А что там за новое слово “lexical”?
Комментарий для david-m.livejournal.com:
lexical отменили
Комментарий для zorgg.blogspot.com:
C++? Я его для себя закрыл. Лучше ты открой для себя что-то более гармоничное. Objective C, например.
Комментарий для jimidini.ya.ru:
Нет его :) Это из примеров кода копипастом попало :)
Гм. return function? В PHP 5.3 функции — это обьекты? :)
Только что проверил — да, так и есть:
Input:
$lambda = function () { echo «Hello World!\n»; };
var_dump($lambda);
Output:
object(Closure)#1 (0) {
}
Комментарий для jimidini.ya.ru:
Там в статье как-то очень тщательно выдерживается coding style… поэтому я лучше тут спрошу: а можно будет делать сразу
preg_replace_callback (’/( +) /’, function($a) {…}, $text)
?
Комментарий для david-m.livejournal.com:
Да.
Input:
$array = array(1, 2, 3);
$array = array_map(function($v) { return $v * $v; }, $array);
var_dump($array);
Output:
array(3) {
[0]=>
int(1)
[1]=>
int(4)
[2]=>
int(9)
}
Комментарий для blog.fxposter.org:
Спасибо! Это радует.
Комментарий для blog.fxposter.org:
Чего проверять-то? Из статьи это следует :)
Комментарий для Евгения Степанищева:
Я как-то привык перепроверять «очевидное» в php:)
Комментарий для david-m.livejournal.com:
Это не очевидное, всё это прямо упоминается в статье )