Portée des variables
La portée d'une variable dépend du contexte
dans lequel la variable est définie. Pour la majorité des
variables, la portée concerne la totalité d'un script
PHP. Mais, lorsque vous définissez une fonction, la
portée d'une variable définie dans cette fonction
est locale à la fonction. Par exemple :
Exemple #1 Les variables sont locales à la fonction
<?php
$a = 1;
include 'b.inc';
?>
Ici, la variable $a sera accessible dans le script inclus
b.inc. Cependant, dans les fonctions
définies par l'utilisateur, une nouvelle définition
de cette variable sera donnée, limitée à la
fonction. Toute variable utilisée dans une fonction est,
par définition, locale. Par exemple :
Le script n'affichera rien à l'écran car
l'instruction echo utilise la variable locale $a,
et celle-ci n'a pas été assignée
préalablement dans la fonction. Vous pouvez noter que
ce concept diffère un petit peu du langage C dans
lequel une variable globale est automatiquement accessible dans
les fonctions, à moins d'être redéfinie
localement dans la fonction. Cela peut poser des problèmes
si vous redéfinissez des variables globales localement.
En PHP, une variable globale doit être
déclarée à l'intérieur de chaque
fonction afin de pouvoir être utilisée dans cette
fonction.
Le mot clé global
Commençons par un exemple avec global
:
Exemple #2 Exemple avec global
<?php
$a = 1;
$b = 2;
function somme() {
global $a, $b;
$b = $a + $b;
}
somme();
echo $b;
Le script ci-dessus va afficher la valeur 3
.
En déclarant globales les variables $a et
$b locales
de la fonction somme(), toutes les références à
ces variables concerneront les variables globales. Il n'y a
aucune limite au nombre de variables globales qui peuvent
être manipulées par une fonction.
Une deuxième méthode pour accéder aux
variables globales est d'utiliser le tableau associatif
pré-défini $GLOBALS. Le précédent
exemple peut être réécrit de la
manière suivante :
Exemple #3 Les variables globales et $GLOBALS
<?php
$a = 1;
$b = 2;
function somme() {
$GLOBALS['b'] = $GLOBALS['a'] + $GLOBALS['b'];
}
somme();
echo $b;
?>
Le tableau $GLOBALS est un tableau associatif avec le nom
des variables globales comme clé et les valeurs des éléments
du tableau comme valeur des variables. Notez que $GLOBALS
existe dans tous les contextes, car $GLOBALS est un
superglobal.
Voici un exemple des super globaux :
Exemple #4 Exemple montrant les superglobales et la portée
<?php
function test_superglobal()
{
echo $_POST['name'];
}
?>
Note:
L'utilisation du mot clé global
à l'extérieur
d'une fonction n'est pas une erreur. Il peut être utilisé si le fichier
est inclus depuis l'intérieur d'une fonction.
Utilisation des variables static
Une autre caractéristique importante de la portée des variables est
la notion de variable static. Une variable statique a
une portée locale uniquement, mais elle ne perd pas sa valeur lorsque le
script appelle la fonction. Prenons l'exemple suivant :
Exemple #5 Les variables statiques
<?php
function test()
{
$a = 0;
echo $a;
$a++;
}
?>
Cette fonction est un peu inutile car à chaque fois
qu'elle est appelée, elle initialise $a à 0
et
affiche "0
". L'incrémentation de la variable ($a++)
ne sert pas à grand chose, car dès que la
fonction est terminée, la variable $a disparaît.
Pour faire une fonction de comptage utile, c'est-à-dire qui
ne perdra pas la trace du compteur, la variable $a est
déclarée comme une variable statique :
Exemple #6 Les variables statiques (2)
<?php
function test()
{
static $a = 0;
echo $a;
$a++;
}
?>
Maintenant, la variable $a est initialisée uniquement
lors du premier appel à la fonction et, à chaque fois que la fonction
test()
est appelée, elle affichera une valeur de
$a incrémentée de 1.
Les variables statiques sont essentielles lorsque vous faites des
appels récursifs à une fonction. Une fonction
récursive est une fonction qui s'appelle elle-même.
Il faut faire attention lorsque vous écrivez une fonction
récursive car il est facile de faire une boucle infinie.
Vous devez vérifier que vous avez bien une condition qui
permet de terminer votre récursivité. La fonction
suivante compte récursivement jusqu'à 10, en utilisant la variable
$count pour savoir quand il faut s'arrêter :
Exemple #7 Les variables statiques et la récursivité
<?php
function test()
{
static $count = 0;
$count++;
echo $count;
if ($count < 10) {
test();
}
$count--;
}
?>
Les variables statiques peuvent être assignées des valeurs qui sont issue
d'expression constante, mais les expressions dynamique, tel que les appels
de fonctions, résulteront en une erreur d'analyse.
Exemple #8 Déclaration de variables statiques
<?php
function foo(){
static $int = 0; // correct
static $int = 1+2; // correct
static $int = sqrt(121); // faux (car c'est une fonction)
$int++;
echo $int;
}
?>
Note:
Les déclarations statiques sont résolues au moment de la
compilation.
Les références avec les variables global
et static
PHP implémente les modificateurs de variables
static
et global,
en terme de référence.
Par exemple, une vraie variable globale est importée dans un
contexte de fonction avec global
.
Cette commande crée en fait une référence sur la variable globale. Cela
peut vous mener à des comportements inattendus, par exemple :
L'exemple ci-dessus va afficher :
NULL
object(stdClass)#1 (0) {
}
Un comportement similaire s'applique à la commande static
.
Les références ne sont pas stockées dynamiquement :
L'exemple ci-dessus va afficher :
Static object: NULL
Static object: NULL
Static object: NULL
Static object: object(stdClass)#3 (1) {
["property"]=>
int(1)
}
Ces exemples illustrent les problèmes rencontrés lors de l'assignation
de référence à des variables statiques, qui sont
oubliées lorsque vous appelez
&get_instance_ref()
une seconde fois.