NOM
perldata - Types de données de Perl
DESCRIPTION
Noms des variables
Perl a trois types de données intégrés : les scalaires, les tableaux de scalaires, et les tableaux associatifs de scalaires, appelés « hachages ». Les tableaux normaux sont des listes ordonnées de scalaires indexées par des nombres, en commençant par 0 et où les indices négatifs sont comptés depuis la fin. Les tables de hachages sont des collections non ordonnées de valeurs scalaires indexées par des chaînes qui sont leurs clés associées.
On fait habituellement référence aux valeurs par leur nom, ou par une référence nommée. Le premier caractère du nom vous indique à quel type de structure de données il correspond. Le reste du nom vous dit à quelle valeur particulière il fait référence. Habituellement, ce nom est un simple identifiant, c'est-à-dire une chaîne commençant par une lettre ou un caractère souligné, et contenant des lettres, des soulignés, et des chiffres. Dans certains cas, il peut être une chaîne d'identifiants, séparés par ::
(ou par le légèrement archaïque '
) ; tous sauf le dernier sont interprétés comme des noms de paquetages, pour localiser l'espace de nommage dans lequel l'identifiant final doit être recherché (voir "Paquetages" in perlmod pour plus de détails). Il est possible de substituer à un simple identifiant une expression qui produit une référence à la valeur lors de l'exécution. Ceci est décrit plus en détails plus bas, et dans perlref.
Perl a aussi ses propres variables intégrées dont les noms ne suivent pas ces règles. Elles ont des noms étranges pour qu'elles ne rentrent pas accidentellement en collision avec l'une de vos variables normales. Les chaînes qui correspondent aux parties entre parenthèses d'une expression rationnelle sont sauvées sous des noms qui ne contiennent que des chiffres après le $
(voir perlop et perlre). De plus, plusieurs variables spéciales qui ouvrent des fenêtres dans le fonctionnement interne de Perl ont des noms contenant des signes de ponctuation et des caractères de contrôle. Elles sont décrites dans perlvar.
Les valeurs scalaires sont toujours désignées par un '$', même si l'on se réfère à un scalaire qui fait partie d'un tableau ou d'un hachage. Le symbole '$' fonctionne d'un point de vue sémantique comme les mots « le », « la ». Ainsi, nous avons :
$days # la simple valeur scalaire "days"
$days[28] # le 29ème élément du tableau @days
$days{'Feb'} # la veleur 'Feb' dans le hachage %days
$#days # le dernier indice du tableau @days
Les tableaux complets (et les tranches de tableaux ou de hachage sont dénotés par '@', qui fonctionne plutôt comme le mot « ces , en ce sens qu'il indique que des valeurs multiples sont attendues :
@days # ($days[0], $days[1],... $days[n])
@days[3,4,5] # identique à ($days[3],$days[4],$days[5])
@days{'a','c'} # identique à ($days{'a'},$days{'c'})
Les hachages complets sont dénotés par '%' :
%days # (clé1, valeur1, clé2, valeur2 ...)
De plus, les sous-programmes sont nommés avec un '&' initial, bien que ce soit optionnel lorsqu'il n'y a pas d'ambiguïté, tout comme « faire » est souvent redondant en français. Les entrées des tables de symboles peuvent être nommées avec un '*' initial, mais vous ne vous souciez pas vraiment de cela pour le moment (si jamais :-).
Chaque type de variable a son propre espace de nommage, tout comme les identifiants de plusieurs types autres que les variables. Ceci signifie que vous pouvez, sans craindre un conflit, utiliser le même nom pour une variable scalaire, un tableau, ou un hachage -- ou, pour cette affaire, un handle de fichier, un handle de répertoire, un nom de sous-programme, ou un label. Ceci veut dire que $foo et @foo sont deux variables différentes. Ceci veut aussi dire que $foo[1]
fait partie de @foo, et pas de $foo. Cela peut sembler un peu étrange, mais c'est normal, puisque c'est étrange.
Puisque les références de variables commencent toujours par '$', '@', ou '%', les mots « réservés » ne sont en fait pas réservés en ce qui concerne les noms de variables (Ils SONT toutefois réservés en ce qui concerne les labels et les handles de fichiers, qui n'ont pas de caractère spécial initial. Vous ne pouvez pas avoir un handle de fichier nommé « log », par exemple. Indice : vous pourriez dire open(LOG,'logfile')
plutôt que open(log,'logfile')
. Utiliser des handles de fichiers en lettres majuscules améliore aussi la lisibilité et vous protège de conflits avec de futurs mots réservés. La casse est significative -- « FOO », « Foo », et « foo » sont tous des noms différents. Les noms qui commencent par une lettre ou un caractère souligné peuvent aussi contenir des chiffres et des soulignés.
Il est possible de remplacer un tel nom alphanumérique par une expression qui retourne une référence au type approprié. Pour une description de ceci, voir perlref.
Les noms qui commencent par un chiffre ne peuvent contenir que des chiffres. Les noms qui ne commencent pas par une lettre, un souligné ou un chiffre sont limités à un caractère, e.g., $%
or $$
(La plupart de ces noms d'un seul caractère ont une signification prédéfinie pour Perl. Par exemple, $$
est l'id. du processus courant).
Contexte
L'interprétation des opérations et des valeurs en Perl dépend parfois des exigences du contexte de l'opération ou de la valeur. Il existe deux contextes majeurs : le contexte de liste et le contexte scalaire. Certaines opérations retournent des valeurs de liste dans les contextes qui réclament une liste, et des valeurs scalaires autrement Ssi ceci est vrai pour une opération alors cela sera mentionné dans la documentation pour cette opération. En d'autres termes, Perl surcharge certaines opérations selon que la valeur de retour attendue est singulière ou plurielle. Certains mots en français fonctionnent aussi de cette façon, comme « lys » et « dos ».
Réciproquement, une opération fournit un contexte scalaire ou de liste à chacun de ses arguments. Par exemple, si vous dites :
int( <STDIN> )
L'opération int fournit un contexte scalaire pour l'opérateur <STDIN>, qui répond en lisant une ligne depuis STDIN et en la passant à l'opération int, qui trouvera alors la valeur entière de cette ligne et retournera cela. Si, au contraire, vous dites :
sort( <STDIN> )
alors l'opération sort fournit un contexte de liste pour <STDIN>, qui se mettra à lire toutes les lignes disponibles jusqu'à la fin du fichier, et passera cette liste de lignes à la routine de tri, qui triera alors ces lignes et les retournera en tant que liste à ce qui est le contexte de sort, quel qu'il soit.
L'affectation est un petit peu spéciale en ce sens qu'elle utilise son argument gauche pour déterminer le contexte de l'argument droit. L'affectation à un scalaire évalue la partie droite dans un contexte scalaire, tandis que l'affectation à un tableau ou à un hachage évalue la partie droite dans un contexte de liste. L'affectation à une liste (ou à une tranche, qui est juste une liste de toute façon) évalue aussi la partie droite dans un contexte de liste.
Lorsque vous utilisez le pragma use warnings
ou l'option de ligne de commande -w de Perl, il arrive que vous voyiez des avertissements sur un usage inutile de constantes ou de fonctions dans un « contexte vide » (« void context », NDT). Le contexte vide signifie juste que la valeur a été abandonnée, comme pour une instruction ne contenant que "fred";
ou getpwuid(0);
. Il compte toujours pour un contexte scalaire pour les fonctions qui se soucient de savoir si elles sont ou non appelées dans un contexte scalaire.
Les sous-programmes définis par l'utilisateur peuvent se soucier d'avoir été appelés dans un contexte vide, scalaire ou de liste. La plupart des sous-programmes n'en ont toutefois pas besoin. C'est parce que les scalaires et les listes sont automatiquement interpolés en listes. Voir wantarray()
dans perlfunc pour une façon dont vous pourriez discerner dynamiquement le contexte d'appel de votre fonction.
Valeurs scalaires
Toute donnée en Perl est un scalaire, un tableau de scalaires ou un hachage de scalaires. Les variables scalaires peuvent contenir des une seule valeur de trois formes différentes : un nombre, une chaîne ou une référence. En général, la conversion d'une forme à une autre est transparente. Bien qu'un scalaire ne puisse pas contenir des valeurs multiples, il peut contenir une référence à un tableau ou à un hachage qui à son tour contient des valeurs multiples.
Les scalaires ne sont pas nécessairement une chose ou une autre. Il n'y a pas d'endroit où déclarer qu'une variable scalaire doit être de type « chaîne », de type « nombre », de type « référence », ou n'importe quoi d'autre. Du fait de la conversion automatique des scalaires, les opérations qui en retournent n'ont pas besoin de se soucier (et en fait ne le peuvent pas) de savoir si leur appelant attend une chaîne, un nombre ou une référence. Perl est un langage contextuellement polymorphe dont les scalaires peuvent être des chaînes, des nombres, ou des références (ce qui inclut les objets). Tandis que les chaînes et les nombres sont considérés comme presque la même chose pour pratiquement tous les usages, les références sont des pointeurs au typage fort et impossible à forcer, avec comptage de référence intégré et invocation de destructeur.
Une valeur scalaire est interprétée comme TRUE (VRAIE, NDT) au sens booléen si ce n'est pas une chaîne vide ou le nombre 0 (ou son équivalent sous forme de chaîne, « 0 »). Le contexte booléen est juste un genre spécial de contexte scalaire, où aucune conversion vers une chaîne ou un nombre n'est jamais effectuée.
Il existe en fait deux variétés de chaînes nulles (parfois appelées des chaînes « vides »), l'une définie et l'autre non. La version définie est juste une chaîne de longueur zéro, telle que ""
. La version non définie est la valeur qui indique qu'il n'existe pas de vraie valeur pour quelque chose, comme lorsqu'il s'est produit une erreur, ou à la fin d'un fichier, ou lorsque vous vous référez à une variable ou à un élément de tableau ou de hachage non initialisé. Bien que dans les anciennes versions de Perl, un scalaire indéfini ait pu devenir défini lorsqu'il était utilisé pour la première fois dans un endroit où une valeur définie était attendue, cela ne se produit plus, sauf dans de rares cas d'autovivification tels qu'expliqués dans perlref. Vous pouvez utiliser l'opérateur defined() pour déterminer si une valeur scalaire est définie (cela n'a pas de sens pour les tableaux ou les hachages), et l'opérateur undef() pour produire une valeur indéfinie.
Pour trouver si une chaîne donnée est un nombre différent de zéro valide, il suffit parfois de la tester à la fois avec le 0 numérique et le « 0 » lexical (bien que ceci provoquera du bruit en provenance de -w). C'est parce que les chaînes qui ne sont pas des nombres comptent comme 0, tout comme en awk :
if ($str == 0 && $str ne "0") {
warn "That doesn't look like a number";
}
Cette méthode est peut-être meilleure parce qu'autrement vous ne traiteriez pas correctement les notations IEEE comme NaN
ou Infinity
. À d'autres moments, vous pourriez préférer déterminer si une donnée chaîne peut être utilisée numériquement en appelant la fonction POSIX::strtod() ou en inspectant votre chaîne avec une expression rationnelle (tel que documenté dans perlre).
warn "has nondigits" if /\D/;
warn "not a whole number" unless /^\d+$/;
warn "not an integer" unless /^[+-]?\d+$/
warn "not a decimal number" unless /^[+-]?\d+\.?\d*$/
warn "not a C float"
unless /^([+-]?)(?=\d|\.\d)\d*(\.\d*)?([Ee]([+-]?\d+))?$/;
La longueur d'un tableau est une valeur scalaire. Vous pourriez trouver la longueur du tableau @days en évaluant $#days
, comme en csh. D'un point de vue technique, ce n'est pas la longueur du tableau ; c'est l'indice de son dernier élément, parce qu'il y a ordinairement un élément numéro 0. Une affectation à $#days
change véritablement la longueur du tableau. Le raccourcissement d'un tableau par cette méthode détruit les valeurs intermédiaires. L'agrandissement d'un tableau ayant précédemment été raccourci ne récupère pas les valeurs qui étaient stockées dans ces éléments (c'était le cas en Perl 4, mais nous avons dû supprimer cela pour nous assurer que les destructeurs sont bien appelés quand on s'y attend).
Vous pouvez aussi gagner en efficacité en pré-étendant un tableau qui va devenir gros (vous pouvez aussi étendre un tableau en affectant des données à un élément qui est au-delà de la fin du tableau). Vous pouvez tronquer totalement un tableau en y affectant la liste vide (). Les expressions suivantes sont équivalentes :
@whatever = ();
$#whatever = -1;
Si vous évaluez un tableau dans un contexte scalaire, cela renvoie la longueur du tableau (notez que ceci n'est pas vrai pour les listes, qui renvoient leur dernière valeur, comme l'opérateur virgule en C, et contrairement aux fonctions intégrées, qui renvoient ce qu'elles ont envie de renvoyer). Ce qui suit est toujours vrai :
scalar(@whatever) == $#whatever - $[ + 1;
La version 5 de Perl a changé la sémantique de $[
: les fichiers qui ne fixent pas la valeur de $[
n'ont plus besoin de s'inquiéter de savoir si un autre fichier a changé sa valeur (en d'autres termes, l'usage de $[
est désapprouvé). Donc de façon générale, vous pouvez présumer que
scalar(@whatever) == $#whatever + 1;
Certains programmeurs choisissent d'utiliser une conversion explicite pour ne rien laisser au hasard :
$element_count = scalar(@whatever);
Si vous évaluez un hachage dans un contexte scalaire, vous obtenez faux si le hachage est vide. S'il contient une paire clé/valeur quelconque, il renvoie vrai ; plus précisément, la valeur retournée est une chaîne constituée du nombre de buckets utilisés et du nombre de buckets alloués, séparés par un signe de division. Ceci n'a tendance à être très utile que pour déterminer si les algorithmes de hachage (compilés) en Perl ont des performances médiocres sur vos données. Par exemple, vous mettez 10 000 trucs dans un hachage, mais l'évaluation de %HASH dans un contexte scalaire révèle « 1/16 », ce qui signifie qu'un seul des seize buckets a été touché, et contient probablement tous vos 10 000 éléments. Cela ne devrait pas se produire).
Constructeurs de valeurs scalaires
Les littéraux numériques sont spécifiés dans un quelconque des formats suivants de nombres entiers ou à virgule flottante :
12345
12345.67
.23E-10 # un très petit nombre
4_294_967_296 # souligné pour la lisibilité
0xff # hexa
0377 # octal
0b011011 # binaire
Les littéraux de chaîne sont habituellement délimités soit par des apostrophes, soit par des guillemets. Ils fonctionnent beaucoup comme dans un shell Unix standard : les littéraux de chaîne entre guillemets sont sujets aux substitutions de variables et au préfixage par barre oblique inverse ; les chaînes entre apostrophes ne le sont pas (sauf pour \'
et \\
). Les règles habituelles d'utilisation de la barre oblique inverse en C s'appliquent aussi bien pour créer des caractères comme la nouvelle ligne, la tabulation, etc., que sous des formes plus exotiques. Voir "Opérateurs apostrophe et type apostrophe" in perlop pour une liste.
Les représentations hexadécimales, octales ou binaires sous forme de chaînes (e.g. '0xff') ne sont pas automatiquement converties sous leur représentation entière. Les fonctions hex() et oct() font ces conversions pour vous. Voir hex() et oct() pour plus de détails.
Vous pouvez aussi inclure des « nouvelles lignes » directement dans vos chaînes, i.e., elles peuvent se terminer sur une ligne différente de celles où elles ont commencé. C'est bien joli, mais si vous oubliez votre apostrophe de fin (ou votre guillemet - NDT), l'erreur ne sera pas rapportée avant que Perl n'ait trouvé une autre ligne comportant une apostrophe, qui peut se trouver bien plus loin dans le script. La substitution de variable à l'intérieur des chaînes est limitée aux variables scalaires, aux tableaux et aux tranches de tableau ou de hachage (en d'autres termes, des noms commençant par $ ou @, suivi d'une expression optionnelle entre crochets comme indice). Le segment de code qui suit affiche « The price is $100. »
$Price = '$100'; # pas interprété
print "The price is $Price.\n"; # interprété
Comme dans certains shells, vous pouvez mettre des accolades autour d'un nom pour le séparer des caractères alphanumériques qui le suivent. Vous devez aussi faire cela lorsque vous interpolez une variable dans une chaîne pour séparer son nom d'un deux-points ou d'une apostrophe, puisqu'ils seraient autrement traités comme un séparateur de paquetage :
$who = "Larry";
print PASSWD "${who}::0:0:Superuser:/:/bin/perl\n";
print "We use ${who}speak when ${who}'s here.\n";
Sans les accolades, Perl aurait cherché un $whospeak, un $who::0
, et une variable who's
. Les deux dernières auraient été les variables $0 et $s dans le paquetage who
(probablement) inexistant.
En fait, un identifiant situé entre de telles accolades est forcé d'être une chaîne, tout comme l'est tout identificateur isolé à l'intérieur d'un indice d'un hachage. Aucun des deux n'a besoin d'apostrophes. Notre exemple précédent, $days{'Feb'}
peut être écrit sous la forme $days{Feb}
et les apostrophes seront présumées automatiquement. Mais tout ce qui est plus compliqué dans l'indice sera interprété comme étant une expression.
Un littéral de la forme v1.20.300.4000
est analysé comme une chaîne composée de caractères correspondants aux ordinaux spécifiés. Ceci fournit une façon alternative plus lisible pour construire des chaînes, au lieu d'utiliser l'interpolation quelque peu moins lisible "\x{1}\x{14}\x{12c}\x{fa0}"
. C'est utile pour représenter des chaînes Unicode, et pour comparer des numéros de version en utilisant les opérateurs de comparaison de chaînes, cmp
, gt
, lt
etc. Si le littéral contient plusieurs points, le premier v
peut être omis.
print v9786; # affiche le SMILEY codé en UTF-8,
# "\x{263a}"
print v102.111.111; # affiche "foo"
print 102.111.111; # idem
De tels littéraux sont acceptés à la fois par require
et use
pour réaliser une vérification de numéro de version. La variable spéciale $^V
contient aussi le numéro de version sous cette forme de l'interpréteur Perl en cours d'utilisation. Voir "$^V" in perlvar.
Les littéraux spéciaux __FILE__, __LINE__, et __PACKAGE__ représentent le nom de fichier courant, le numéro de la ligne, et le nom du paquetage à ce point de votre programme. Ils ne peuvent être utilisés que comme des mots-clé isolés ; ils ne seront pas interpolés dans les chaînes. S'il n'existe pas de paquetage courant (à cause d'une directive package;
), __PACKAGE__ est la valeur indéfinie.
Les deux caractères de contrôle ^D et ^Z, et les mots-clé __END__ et __DATA__ peuvent être utilisés pour indique la fin logique d'un script avant la fin effective du fichier. Tout texte les suivant est ignoré.
Le texte qui suit __DATA__ peut être lu via le handle de fichier PACKNAME::DATA
, où PACKNAME
est le paquetage qui était courant lorsque le mot-clé __DATA__ a été rencontré. Le handle de fichier est laissé ouvert, pointant vers le contenu après __DATA__. Il est de la responsabilité du programme d'effectuer un close DATA
lorsqu'il a fini d'y lire. Pour la compatibilité avec d'anciens scripts écrits avant que __DATA__ ne soit introduit, __END__ se comporte comme __DATA__ dans le script principal (mais pas dans les fichiers chargés par require
ou do
) et laisse le contenu restant du fichier accessible via main::DATA
.
Voir SelfLoader pour une plus longue description de __DATA__, et un exemple de son utilisation. Notez que vous ne pouvez pas lire depuis le handle de fichier DATA dans un bloc BEGIN : ce bloc est exécuté dès qu'il est vu (pendant la compilation), à un moment où le mot-clé __DATA__ (ou __END__) correspondant n'a pas encore été rencontré.
Un mot qui n'a aucune autre interprétation dans la grammaire sera traité comme s'il était une chaîne entre apostrophes. Ces mots sont connus sous le nom de « barewords ». Comme pour les handles de fichier et les labels, un bareword constitué entièrement de lettres minuscules risque d'entrer en conflit avec de futurs mots réservés, et si vous utilisez le pragma use warnings
ou l'option -w, Perl vous avertira pour chacun d'entre eux. Certaines personnes pourraient vouloir rendre les barewords totalement hors-la-loi. Si vous dites
use strict 'subs';
alors tout bareword qui ne serait PAS interprété comme un appel à un sous-programme produit à la place une erreur au moment de la compilation. La restriction continue jusqu'à la fin du bloc qui le contient. Un bloc interne pourrait annuler ceci en disant no strict 'subs'
.
Les tableaux et les tranches sont interpolés en chaînes entre guillemets en joignant tous les éléments avec le délimiteur spécifié dans la variable $"
($LIST_SEPARATOR
dans le paquetage English.pm), une espace par défaut. Les expressions suivantes sont équivalentes :
$temp = join($", @ARGV);
system "echo $temp";
system "echo @ARGV";
À l'intérieur d'un motif de recherche (qui subit aussi la substitution entre guillemets) il y a une malheureuse ambiguïté : est-ce que /$foo[bar]/
doit être interprété comme /${foo}[bar]/
(où [bar]
est une classe de caractères pour l'expression régulière) ou comme /${foo[bar]}/
(où [bar]
est un indice du tableau @foo) ? Si @foo n'existe pas par ailleurs, alors c'est évidemment une classe de caractères. Si @foo existe, Perl choisit de deviner la valeur de [bar]
, et il a presque toujours raison. S'il se trompe, ou si vous êtes simplement complètement paranoïaque, vous pouvez forcer l'interprétation correcte avec des accolades comme ci-dessus.
Une forme de citation orientée ligne est basée sur la syntaxe « here-document » du shell. Après un <<
vous spécifiez une chaîne pour terminer le matériel cité, et toutes les lignes qui suivent la ligne courante jusqu'à la chaîne de terminaison forment la valeur de l'élément. La chaîne de terminaison peut être soit un identificateur (un mot), soit du texte cité. S'il est entre guillemets, le type de guillemets que vous utilisez détermine le traitement du texte, tout comme dans une citation normale. Un identificateur sans guillemets fonctionne comme des guillemets normaux. Il ne doit pas y avoir d'espace entre le <<
et l'identificateur (si vous mettez une espace, elle sera traitée comme un identificateur nul, ce qui est valide, et correspond à la première ligne vide). La chaîne de terminaison doit apparaître toute seule (sans guillemets et sans espaces l'entourant) sur la ligne de terminaison.
print <<EOF;
The price is $Price.
EOF
print <<"EOF"; # comme ci-dessus
The price is $Price.
EOF
print <<`EOC`; # exécute les commandes
echo hi there
echo lo there
EOC
print <<"foo", <<"bar"; # vous pouvez les empiler
I said foo.
foo
I said bar.
bar
myfunc(<<"THIS", 23, <<'THAT');
Here's a line
or two.
THIS
and here's another.
THAT
N'oubliez simplement pas que vous devez mettre un point-virgule à la fin pour terminer la déclaration, car Perl ne sait pas que vous n'allez pas essayer de faire ceci :
print <<ABC
179231
ABC
+ 20;
Si vous désirez que vos here-documents soient indentés avec le reste du code, vous devrez retirer manuellement la première espace de chaque ligne :
($quote = <<'FINIS') =~ s/^\s+//gm;
The Road goes ever on and on,
down from the door where it began.
FINIS
Constructeurs de listes de valeurs
Les valeurs de liste sont dénotées en séparant les valeurs individuelles par des virgules (et en enfermant la liste entre parenthèses lorsque la précédence le requiert) :
(LIST)
Dans un contexte qui ne requiert pas une valeur de liste, la valeur de ce qui apparaît être un littéral de liste est simplement la valeur de l'élément final, comme avec l'opérateur virgule en C. Par exemple,
@foo = ('cc', '-E', $bar);
affecte la totalité de la valeur de liste au tableau @foo, mais
$foo = ('cc', '-E', $bar);
affecte la valeur de la variable $bar à la variable $foo. Notez que la valeur d'un véritable tableau dans un contexte scalaire est la longueur du tableau ; ce qui suit affecte la valeur 3 à $foo :
@foo = ('cc', '-E', $bar);
$foo = @foo; # $foo prend la valeur 3
Vous pouvez avoir une virgule optionnelle avant la parenthèse fermante d'un littéral de liste, vous pouvez donc dire :
@foo = (
1,
2,
3,
);
Pour utiliser un here-document afin d'affecter un tableau, une ligne par élément, vous pourriez utiliser l'approche suivante :
@sauces = <<End_Lines =~ m/(\S.*\S)/g;
normal tomato
spicy tomato
green chile
pesto
white wine
End_Lines
Les LIST font une interpolation automatique des sous-listes. C'est-à-dire que lorsqu'une LIST est évaluée, chaque élément de la liste est évalué dans un contexte de liste, et la valeur de liste résultante est interpolée en LIST tout comme si chaque élément était un membre de LIST. Ainsi, les tableaux perdent leur identité dans une LIST - la liste
(@foo,@bar,&SomeSub)
contient tous les éléments de @foo suivis par tous les éléments de @bar, suivis par tous les éléments retournés par le sous-programme appelé SomeSub quand il est appelé dans un contexte de liste. Pour faire une référence à une liste qui NE soit PAS interpolée, voir perlref.
La liste vide est représentée par (). L'interpoler dans une liste n'a aucun effet. Ainsi, ((),(),()) est équivalent à (). De façon similaire, interpoler un tableau qui ne contient pas d'élément revient à ce qu'aucun tableau n'ait été interpolé à ce moment-là.
Une valeur de liste peut aussi être indicée comme un tableau normal. Vous devez mettre la liste entre parenthèses pour éviter les ambiguïtés. Par exemple :
# Stat renvoit une valeur de liste.
$time = (stat($file))[8];
# ICI, ERREUR DE SYNTAXE.
$time = stat($file)[8]; # OOPS, OUBLI DES PARENTHESES
# Trouver un chiffre hexadécimal.
$hexdigit = ('a','b','c','d','e','f')[$digit-10];
# Un "opérateur virgule inversé".
return (pop(@foo),pop(@foo))[0];
Les listes ne peuvent être affectées que si chaque élément de la liste peut l'être lui aussi :
($a, $b, $c) = (1, 2, 3);
($map{'red'}, $map{'blue'}, $map{'green'}) = (0x00f, 0x0f0, 0xf00);
Une exception à ceci est que vous pouvez affecter undef
dans une liste. C'est pratique pour balancer certaines valeurs de retour d'une fonction :
($dev, $ino, undef, undef, $uid, $gid) = stat($file);
L'affectation de liste dans un contexte scalaire renvoie le nombre d'éléments produits par l'expression du côté droit de l'affectation :
$x = (($foo,$bar) = (3,2,1)); # met 3 dans $x, pas 2
$x = (($foo,$bar) = f()); # met le nombre de valeurs
# de retour de f() dans $x
Ceci est pratique lorsque vous voulez faire une affectation de liste dans un contexte booléen, parce que la plupart des fonctions de liste renvoient une liste vide quand elle se terminent, ce qui donne un 0 quand on l'affecte, 0 qui est interprété comme FALSE (FAUX - NDT).
L'élément final peut être un tableau ou un hachage :
($a, $b, @rest) = split;
local($a, $b, %rest) = @_;
Vous pouvez en vérité mettre un tableau ou un hachage n'importe où dans la liste, mais le premier situé dans la liste va aspirer toutes les valeurs, et tout ce qui le suivra deviendra indéfini. Cela peut être pratique dans un local() ou un my().
Un hachage peut être initialisé en utilisant une liste de littéraux contenant des paires d'éléments qui doivent être interprétées comme des couples clé/valeur :
# identique à l'affectation de map ci-dessus
%map = ('red',0x00f,'blue',0x0f0,'green',0xf00);
Tandis que les littéraux de liste et les tableaux nommés sont souvent interchangeables, ce n'est pas le cas pour les hachages. Le simple fait que vous puissiez indicer une valeur de liste comme un tableau normal ne veut pas dire que vous pouvez indicer une valeur de liste comme un hachage. De la même manière, les hachages inclus comme parties d'autres listes (y compris les listes de paramètres et les listes de retour de fonctions) s'aplatissent toujours en paires clé/valeur. C'est pourquoi il est parfois bon d'utiliser des références.
Il est parfois plus lisible d'utiliser l'opérateur =>
dans les paires clé/valeur. L'opérateur =>
est principalement juste un synonyme plus clair visuellement qu'une virgule, mais il permet aussi à son opérande de gauche d'être interprété comme une chaîne, si c'est un bareword qui serait un identifiant légal. Cela rend plus jolie l'initialisation des hachages :
%map = (
red => 0x00f,
blue => 0x0f0,
green => 0xf00,
);
ou pour initialiser les références de hachage devant être utilisées en tant qu'enregistrements :
$rec = {
witch => 'Mable the Merciless',
cat => 'Fluffy the Ferocious',
date => '10/31/1776',
};
ou pour utiliser l'appel par variables pour les fonctions compliquées :
$field = $query->radio_group(
name => 'group_name',
values => ['eenie','meenie','minie'],
default => 'meenie',
linebreak => 'true',
labels => \%labels
);
Notez que ce n'est pas parce qu'un hachage est initialisé dans un certain ordre qu'il ressortira dans cet ordre. Voir sort()
pour des exemples sur la façon de s'arranger pour obtenir des sorties ordonnées.
Tranches
Une façon commune d'accéder à un tableau ou à un hachage est d'en prendre un élément à la fois. Vous pouvez aussi indicer une liste pour en obtenir un seul élément.
$whoami = $ENV{"USER"}; # un élément du hachage
$parent = $ISA[0]; # un élément du tableau
$dir = (getpwnam("daemon"))[7]; # idem, mais avec une liste
Une tranche accède à plusieurs éléments d'une liste, d'un tableau ou d'un hachage simultanément en utilisant une liste d'indices. C'est plus pratique que d'écrire les éléments individuellement sous la forme d'une liste de valeurs scalaires séparées.
($him, $her) = @folks[0,-1]; # tranche de tableau
@them = @folks[0 .. 3]; # tranche de tableau
($who, $home) = @ENV{"USER", "HOME"}; # tranche de hachage
($uid, $dir) = (getpwnam("daemon"))[2,7]; # tranche de liste
Puisque vous pouvez affecter à une liste de variables, vous pouvez aussi affecter à une tranche de tableau ou de hachage.
@days[3..5] = qw/Wed Thu Fri/;
@colors{'red','blue','green'}
= (0xff0000, 0x0000ff, 0x00ff00);
@folks[0, -1] = @folks[-1, 0];
Les affectations précédentes sont exactement équivalents à
($days[3], $days[4], $days[5]) = qw/Wed Thu Fri/;
($colors{'red'}, $colors{'blue'}, $colors{'green'})
= (0xff0000, 0x0000ff, 0x00ff00);
($folks[0], $folks[-1]) = ($folks[0], $folks[-1]);
Puisque changer une tranche change le tableau ou le hachage original dont la tranche est issue, une structure foreach
altèrera certaines -- ou même toutes les -- valeurs du tableau ou du hachage.
foreach (@array[ 4 .. 10 ]) { s/peter/paul/ }
foreach (@hash{keys %hash}) {
s/^\s+//; # supprime les espaces au début des éléments
s/\s+$//; # supprime les espaces à la fin des éléments
s/(\w+)/\u\L$1/g; # met une majuscule aux mots
}
Une tranche d'une liste vide est encore une liste vide. Ainsi :
@a = ()[1,0]; # @a n'a pas d'éléments
@b = (@a)[0,1]; # @b n'a pas d'éléments
@c = (0,1)[2,3]; # @c n'a pas d'éléments
Mais :
@a = (1)[1,0]; # @a a deux éléments
@b = (1,undef)[1,0,2]; # @b a trois éléments
Ceci rend aisée l'écriture de boucles qui se terminent lorsqu'une liste nulle est renvoyée :
while ( ($home, $user) = (getpwent)[7,0]) {
printf "%-8s %s\n", $user, $home;
}
Comme noté précédemment dans ce document, le sens scalaire de l'affectation de liste est le nombre d'éléments de la partie droite de l'affectation. La liste nulle ne contient pas d'éléments, donc lorsque le fichier de mots de passe est vidé, le résultat est 0 et non pas 2.
Si vous êtes troublé par le pourquoi de l'usage d'un '@' ici sur une tranche de hachage au lieu d'un '%', pensez-y ainsi. Le type de parenthésage (avec des crochets ou des accolades) décide si c'est un tableau ou un hachage qui est examiné. D'un autre côté, le symbole en préfixe ('$' ou '@') du tableau ou du hachage indique si vous récupérez une valeur simple (un scalaire) ou une valeur multiple (une liste).
Typeglobs et Handles de Fichiers
Perl utilise un type interne appelé un typeglob pour contenir une entrée complète de table de symbole. Le préfixe de type d'un typeglob est une *
, parce qu'il représente tous les types. Ceci fut la manière favorite de passer par référence à une fonction des tableaux et des hachages, mais maintenant que nous avons de vraies références, c'est rarement nécessaire.
Le principal usage des typeglobs dans le Perl moderne est de créer des alias de table de symbole. Cette affectation :
*this = *that;
fait de $this un alias de $that, @this un alias de @that, %this un alias de %that, &this un alias de &that, etc. Il est bien plus sûr d'utiliser une référence. Ceci :
local *Here::blue = \$There::green;
fait temporairement de $Here::blue un alias de $There::green, mais ne fait pas de @Hzere::blue un alias de @There::green, ou de %Here::blue un alias de %There::green, etc. Voir "Tables de Symboles" in perlmod pour plus d'exemples de ceci. Aussi étrange que cela puisse paraître, c'est la base de tout le système d'import/export de module.
Un autre usage des typeglobs est le passage de handles de fichiers à une fonction, ou la création de nouveaux handles de fichiers. Si vous avez besoin d'utiliser un typeglob pour sauvegarder un handle de fichier, faites-le de cette façon :
$fh = *STDOUT;
ou peut-être comme une vraie référence, comme ceci :
$fh = \*STDOUT;
Voir perlsub pour des exemples d'usages de ceci comme handles de fichiers indirects dans des fonctions.
Les typeglobs sont aussi une façon de créer un handle de fichier local en utilisant l'opérateur local(). Ceux-ci ne durent que jusqu'à ce que leur bloc soit terminé, mais peuvent être passés en retour. Par exemple :
sub newopen {
my $path = shift;
local *FH; # not my!
open (FH, $path) || return undef;
return *FH;
}
$fh = newopen('/etc/passwd');
Maintenant que nous avons la notation *foo{THING}
, les typeglobs ne sont plus autant utilisés pour les manipulations de handles de fichiers, même s'ils sont toujours nécessaires pour passer des fichiers tout neufs et des handles de répertoire dans les fonctions. C'est parce que *HANDLE{IO}
ne fonctionne que si HANDLE a déjà été utilisé en tant que handle. En d'autres termes, *FH
doit être utilisé pour créer de nouvelles entrées de table de symboles ; *foo{THING}
ne le peut pas. En cas de doute, utilisez *FH
.
Toutes les fonctions qui sont capables de créer des handles de fichiers (open(), opendir(), pipe(), socketpair(), sysopen(), socket(), et accept()) créent automatiquement un handle de fichier anonyme si le handle qui leur est passé est une variable scalaire non initialisée. Ceci permet aux constructions telles que open(my $fh, ...)
et open(local $fh,...)
d'être utilisées pour créer des handles de fichiers qui seront convenablement et automatiquement fermés lorsque la portée se termine, pourvu qu'il n'existe aucune autre référence vers eux. Ceci élimine largement le besoin pour les typeglobs lors de l'ouverture des handles de fichiers qui doivent être passés à droite et à gauche, comme dans l'exemple suivant :
sub myopen {
open my $fh, "@_"
or die "Can't open '@_': $!";
return $fh;
}
{
my $f = myopen("</etc/motd");
print <$f>;
# $f implicitly closed here
}
Une autre façon de créer des handles de fichiers anonymes est d'utiliser le module Symbol ou le module IO::Handle et ceux de son acabit. Ces modules ont l'avantage de ne pas cacher les différents types du même nom pendant le local(). Voir la fin de la documentation de la fonction open()
pour un exemple.
VOIR AUSSI
Voir perlvar pour une description des variables intégrées de Perl et une discussion des noms de variable légaux. Voir perlref, perlsub, et "Tables de Symboles" in perlmod pour plus de détails sur les typeglobs et la syntaxe *foo{THING}
.
TRADUCTION
Version
Cette traduction française correspond à la version anglaise distribuée avec perl 5.6.0. Pour en savoir plus concernant ces traductions, consultez http://perl.enstimac.fr/.
Traducteur
Roland Trique <roland.trique@free.fr>
Relecture
Jim Fox <fox@sugar-land.dowell.slb.com>, Etienne Gauthier <egauthie@capgemini.fr>, Gérard Delafond