Elixir - Guide rapide

Elixir est un langage dynamique et fonctionnel conçu pour créer des applications évolutives et maintenables. Il exploite la machine virtuelle Erlang, connue pour exécuter des systèmes à faible latence, distribués et tolérants aux pannes, tout en étant également utilisée avec succès dans le développement Web et le domaine des logiciels embarqués.

Elixir est un langage fonctionnel et dynamique construit sur Erlang et la VM Erlang. Erlang est un langage qui a été écrit à l'origine en 1986 par Ericsson pour aider à résoudre des problèmes de téléphonie tels que la distribution, la tolérance aux pannes et la concurrence. Elixir, écrit par José Valim, étend Erlang et fournit une syntaxe plus conviviale dans la VM Erlang. Il le fait tout en gardant les performances du même niveau qu'Erlang.

Caractéristiques d'Elixir

Parlons maintenant de quelques caractéristiques importantes d'Elixir -

  • Scalability - Tout le code Elixir s'exécute dans des processus légers qui sont isolés et échangent des informations via des messages.

  • Fault Tolerance- Elixir fournit des superviseurs qui décrivent comment redémarrer des parties de votre système lorsque les choses tournent mal, en revenant à un état initial connu qui est garanti de fonctionner. Cela garantit que votre application / plate-forme n'est jamais en panne.

  • Functional Programming - La programmation fonctionnelle favorise un style de codage qui aide les développeurs à écrire un code court, rapide et maintenable.

  • Build tools- Elixir est livré avec un ensemble d'outils de développement. Mix est l'un de ces outils qui facilite la création de projets, la gestion des tâches, l'exécution de tests, etc. Il possède également son propre gestionnaire de packages - Hex.

  • Erlang Compatibility - Elixir fonctionne sur la VM Erlang, donnant aux développeurs un accès complet à l'écosystème d'Erlang.

Pour exécuter Elixir, vous devez le configurer localement sur votre système.

Pour installer Elixir, vous aurez d'abord besoin d'Erlang. Sur certaines plates-formes, les packages Elixir sont livrés avec Erlang.

Installer Elixir

Voyons maintenant l'installation d'Elixir dans différents systèmes d'exploitation.

l'installation de Windows

Pour installer Elixir sous Windows, téléchargez le programme d'installation depuis https://repo.hex.pm/elixirwebsetup.exe et cliquez simplement Nextpour suivre toutes les étapes. Vous l'aurez sur votre système local.

Si vous rencontrez des problèmes lors de son installation, vous pouvez consulter cette page pour plus d'informations.

Configuration Mac

Si Homebrew est installé, assurez-vous qu'il s'agit de la dernière version. Pour la mise à jour, utilisez la commande suivante -

brew update

Maintenant, installez Elixir en utilisant la commande donnée ci-dessous -

brew install elixir

Configuration Ubuntu / Debian

Les étapes pour installer Elixir dans une configuration Ubuntu / Debian sont les suivantes -

Ajouter le repo Erlang Solutions -

wget https://packages.erlang-solutions.com/erlang-solutions_1.0_all.deb && sudo 
dpkg -i erlang-solutions_1.0_all.deb 
sudo apt-get update

Installez la plateforme Erlang / OTP et toutes ses applications -

sudo apt-get install esl-erlang

Installer Elixir -

sudo apt-get install elixir

Autres distributions Linux

Si vous avez une autre distribution Linux, veuillez visiter cette page pour configurer elixir sur votre système local.

Test de la configuration

Pour tester la configuration d'Elixir sur votre système, ouvrez votre terminal et entrez iex dedans. Cela ouvrira la coquille d'élixir interactive comme suit -

Erlang/OTP 19 [erts-8.0] [source-6dc93c1] [64-bit] 
[smp:4:4] [async-threads:10] [hipe] [kernel-poll:false]  

Interactive Elixir (1.3.1) - press Ctrl+C to exit (type h() ENTER for help) 
iex(1)>

Elixir est maintenant configuré avec succès sur votre système.

Nous commencerons par le programme habituel «Hello World».

Pour démarrer le shell interactif Elixir, entrez la commande suivante.

iex

Une fois le shell démarré, utilisez le IO.putspour "mettre" la chaîne sur la sortie de la console. Entrez ce qui suit dans votre coquille Elixir -

IO.puts "Hello world"

Dans ce tutoriel, nous utiliserons le mode script Elixir où nous conserverons le code Elixir dans un fichier avec l'extension .ex. Gardons maintenant le code ci-dessus dans letest.exfichier. Dans l'étape suivante, nous l'exécuterons en utilisantelixirc-

IO.puts "Hello world"

Essayons maintenant d'exécuter le programme ci-dessus comme suit -

$elixirc test.ex

Le programme ci-dessus génère le résultat suivant -

Hello World

Ici, nous appelons une fonction IO.putspour générer une chaîne vers notre console en sortie. Cette fonction peut également être appelée comme nous le faisons en C, C ++, Java, etc., en fournissant des arguments entre parenthèses après le nom de la fonction -

IO.puts("Hello world")

commentaires

Les commentaires sur une seule ligne commencent par un symbole «#». Il n'y a pas de commentaire sur plusieurs lignes, mais vous pouvez empiler plusieurs commentaires. Par exemple -

#This is a comment in Elixir

Fin de ligne

Il n'y a pas de fin de ligne obligatoire comme ";" dans Elixir. Cependant, nous pouvons avoir plusieurs instructions sur la même ligne, en utilisant ';'. Par exemple,

IO.puts("Hello"); IO.puts("World!")

Le programme ci-dessus génère le résultat suivant -

Hello 
World!

Identifiants

Des identifiants tels que des variables, des noms de fonction sont utilisés pour identifier une variable, une fonction, etc. Dans Elixir, vous pouvez nommer vos identifiants en commençant par un alphabet minuscule avec des chiffres, des traits de soulignement et des lettres majuscules par la suite. Cette convention de dénomination est communément appelée snake_case. Par exemple, voici quelques identifiants valides dans Elixir -

var1       variable_2      one_M0r3_variable

Veuillez noter que les variables peuvent également être nommées avec un trait de soulignement en tête. Une valeur qui n'est pas destinée à être utilisée doit être affectée à _ ou à une variable commençant par un trait de soulignement -

_some_random_value = 42

Elixir s'appuie également sur des traits de soulignement pour rendre les fonctions privées aux modules. Si vous nommez une fonction avec un trait de soulignement dans un module et importez ce module, cette fonction ne sera pas importée.

Il existe de nombreuses autres subtilités liées à la dénomination des fonctions dans Elixir dont nous parlerons dans les chapitres à venir.

Mots réservés

Les mots suivants sont réservés et ne peuvent pas être utilisés comme noms de variables, de modules ou de fonctions.

after     and     catch     do     inbits     inlist     nil     else     end 
not     or     false     fn     in     rescue     true     when     xor 
__MODULE__    __FILE__    __DIR__    __ENV__    __CALLER__

Pour utiliser n'importe quelle langue, vous devez comprendre les types de données de base pris en charge par la langue. Dans ce chapitre, nous aborderons 7 types de données de base supportés par le langage elixir: entiers, flottants, booléens, atomes, chaînes, listes et tuples.

Types numériques

Elixir, comme tout autre langage de programmation, prend en charge à la fois les entiers et les flottants. Si vous ouvrez votre shell elixir et entrez un entier ou un flottant en entrée, il retournera sa valeur. Par exemple,

42

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

42

Vous pouvez également définir des nombres en bases octales, hexadécimales et binaires.

Octal

Pour définir un nombre en base octale, préfixez-le avec «0o». Par exemple, 0o52 en octal équivaut à 42 en décimal.

Hexadécimal

Pour définir un nombre en base décimale, préfixez-le avec «0x». Par exemple, 0xF1 en hexadécimal équivaut à 241 en décimal.

Binaire

Pour définir un nombre en base binaire, préfixez-le avec «0b». Par exemple, 0b1101 en binaire équivaut à 13 en décimal.

Elixir prend en charge la double précision 64 bits pour les nombres à virgule flottante. Et ils peuvent également être définis à l'aide d'un style d'exponentiation. Par exemple, 10145230000 peut être écrit sous la forme 1.014523e10

Atomes

Les atomes sont des constantes dont le nom est leur valeur. Ils peuvent être créés à l'aide du symbole de couleur (:). Par exemple,

:hello

Booléens

Elixir prend en charge true et falsecomme booléens. Ces deux valeurs sont en fait attachées aux atomes: true et: false respectivement.

Cordes

Les chaînes d'Elixir sont insérées entre des guillemets doubles et sont encodées en UTF-8. Ils peuvent s'étendre sur plusieurs lignes et contenir des interpolations. Pour définir une chaîne, entrez-la simplement entre guillemets -

"Hello world"

Pour définir des chaînes multilignes, nous utilisons une syntaxe similaire à python avec des guillemets doubles triples -

"""
Hello
World!
"""

Nous en apprendrons davantage sur les chaînes, les binaires et les listes de caractères (similaires aux chaînes) dans le chapitre sur les chaînes.

Binaires

Les binaires sont des séquences d'octets entre << >> séparés par une virgule. Par exemple,

<< 65, 68, 75>>

Les binaires sont principalement utilisés pour gérer les bits et les octets de données liées, si vous en avez. Ils peuvent, par défaut, stocker de 0 à 255 dans chaque valeur. Cette limite de taille peut être augmentée en utilisant la fonction de taille qui indique combien de bits il faut pour stocker cette valeur. Par exemple,

<<65, 255, 289::size(15)>>

Listes

Elixir utilise des crochets pour spécifier une liste de valeurs. Les valeurs peuvent être de n'importe quel type. Par exemple,

[1, "Hello", :an_atom, true]

Les listes sont livrées avec des fonctions intégrées pour la tête et la queue de la liste nommées hd et tl qui renvoient respectivement la tête et la queue de la liste. Parfois, lorsque vous créez une liste, elle renvoie une liste de caractères. En effet, lorsque elixir voit une liste de caractères ASCII imprimables, il l'imprime sous forme de liste de caractères. Veuillez noter que les chaînes et les listes de caractères ne sont pas égales. Nous discuterons plus en détail des listes dans les chapitres suivants.

Tuples

Elixir utilise des accolades pour définir les tuples. Comme les listes, les tuples peuvent contenir n'importe quelle valeur.

{ 1, "Hello", :an_atom, true

Une question se pose ici, pourquoi fournir les deux lists et tuplesquand ils travaillent tous les deux de la même manière? Eh bien, ils ont des implémentations différentes.

  • Les listes sont en fait stockées sous forme de listes liées, donc les insertions, les suppressions sont très rapides dans les listes.

  • Les tuples, d'autre part, sont stockés dans un bloc de mémoire contigu, ce qui accélère leur accès mais ajoute un coût supplémentaire sur les insertions et les suppressions.

Une variable nous fournit un stockage nommé que nos programmes peuvent manipuler. Chaque variable dans Elixir a un type spécifique, qui détermine la taille et la disposition de la mémoire de la variable; la plage de valeurs pouvant être stockées dans cette mémoire; et l'ensemble des opérations qui peuvent être appliquées à la variable.

Types de variables

Elixir prend en charge les types de variables de base suivants.

Entier

Ceux-ci sont utilisés pour les nombres entiers. Ils sont de taille 32 bits sur une architecture 32 bits et 64 bits sur une architecture 64 bits. Les entiers sont toujours signés dans elixir. Si un entier commence à augmenter en taille au-dessus de sa limite, élixir le convertit en un gros entier qui prend de la mémoire dans la plage de 3 à n mots, selon ce qui peut le tenir en mémoire.

Flotteurs

Les flotteurs ont une précision de 64 bits en élixir. Ils sont également comme des entiers en termes de mémoire. Lors de la définition d'un flottant, la notation exponentielle peut être utilisée.

Booléen

Ils peuvent prendre 2 valeurs qui sont vraies ou fausses.

Cordes

Les chaînes sont encodées en utf-8 en elixir. Ils ont un module de chaînes qui fournit beaucoup de fonctionnalités au programmeur pour manipuler les chaînes.

Fonctions anonymes / Lambdas

Ce sont des fonctions qui peuvent être définies et affectées à une variable, qui peuvent ensuite être utilisées pour appeler cette fonction.

Les collections

Il existe de nombreux types de collections disponibles dans Elixir. Certains d'entre eux sont des listes, des tuples, des cartes, des binaires, etc. Ceux-ci seront discutés dans les chapitres suivants.

Déclaration de variable

Une déclaration de variable indique à l'interpréteur où et combien créer le stockage pour la variable. Elixir ne nous permet pas de déclarer simplement une variable. Une variable doit être déclarée et affectée d'une valeur en même temps. Par exemple, pour créer une variable nommée life et lui attribuer une valeur 42, nous procédons comme suit -

life = 42

Cela liera la durée de vie de la variable à la valeur 42. Si nous voulons réaffecter à cette variable une nouvelle valeur, nous pouvons le faire en utilisant la même syntaxe que ci-dessus, c'est-à-dire,

life = "Hello world"

Dénomination des variables

La dénomination des variables suit un snake_caseconvention dans Elixir, c'est-à-dire que toutes les variables doivent commencer par une lettre minuscule, suivie de 0 ou plusieurs lettres (majuscules et minuscules), suivies à la fin d'un '?' optionnel OU '!'.

Les noms de variables peuvent également être démarrés par un trait de soulignement, mais cela ne doit être utilisé que si vous ignorez la variable, c'est-à-dire que cette variable ne sera plus utilisée mais doit être affectée à quelque chose.

Variables d'impression

Dans le shell interactif, les variables s'imprimeront si vous entrez simplement le nom de la variable. Par exemple, si vous créez une variable -

life = 42

Et entrez `` vie '' dans votre shell, vous obtiendrez le résultat comme -

42

Mais si vous souhaitez générer une variable vers la console (lors de l'exécution d'un script externe à partir d'un fichier), vous devez fournir la variable en entrée pour IO.puts fonction -

life = 42  
IO.puts life

ou

life = 42 
IO.puts(life)

Cela vous donnera la sortie suivante -

42

Un opérateur est un symbole qui indique au compilateur d'effectuer des manipulations mathématiques ou logiques spécifiques. Il existe BEAUCOUP d'opérateurs fournis par elixir. Ils sont répartis dans les catégories suivantes -

  • Opérateurs arithmétiques
  • Opérateurs de comparaison
  • opérateurs booléens
  • Divers opérateurs

Opérateurs arithmétiques

Le tableau suivant présente tous les opérateurs arithmétiques pris en charge par le langage Elixir. Supposons une variableA détient 10 et variable B détient 20, alors -

Afficher des exemples

Opérateur La description Exemple
+ Ajoute 2 nombres. A + B donnera 30
- Soustrait le deuxième nombre du premier. AB donnera -10
* Multiplie deux nombres. A * B donnera 200
/ Divise le premier nombre du second. Cela jette les nombres en flottants et donne un résultat flottant A / B donnera 0,5.
div Cette fonction est utilisée pour obtenir le quotient sur division. div (10,20) donnera 0
rem Cette fonction est utilisée pour obtenir le reste en division. rem (A, B) donnera 10

Opérateurs de comparaison

Les opérateurs de comparaison d'Elixir sont pour la plupart communs à ceux fournis dans la plupart des autres langues. Le tableau suivant résume les opérateurs de comparaison dans Elixir. Supposons une variableA détient 10 et variable B détient 20, alors -

Afficher des exemples

Opérateur La description Exemple
== Vérifie si la valeur à gauche est égale à la valeur à droite (Type caste les valeurs si elles ne sont pas du même type). A == B donnera faux
! = Vérifie si la valeur à gauche n'est pas égale à la valeur à droite. A! = B donnera vrai
=== Vérifie si le type de valeur à gauche est égal au type de valeur à droite, si oui, vérifiez la même chose pour la valeur. A === B donnera faux
! == Idem que ci-dessus mais vérifie l'inégalité au lieu de l'égalité. A! == B donnera vrai
> Vérifie si la valeur de l'opérande gauche est supérieure à la valeur de l'opérande droit; si oui, alors la condition devient vraie. A> B donnera faux
< Vérifie si la valeur de l'opérande gauche est inférieure à la valeur de l'opérande droit; si oui, alors la condition devient vraie. A <B donnera vrai
> = Vérifie si la valeur de l'opérande gauche est supérieure ou égale à la valeur de l'opérande droit; si oui, alors la condition devient vraie. A> = B donnera faux
<= Vérifie si la valeur de l'opérande gauche est inférieure ou égale à la valeur de l'opérande droit; si oui, alors la condition devient vraie. A <= B donnera vrai

Opérateurs logiques

Elixir fournit 6 opérateurs logiques: and, or, not, &&, || et !. Les trois premiers,and or notsont des opérateurs booléens stricts, ce qui signifie qu'ils s'attendent à ce que leur premier argument soit un booléen. Un argument non booléen déclenchera une erreur. Alors que les trois suivants,&&, || and !ne sont pas strictes, ne nous obligent pas à avoir la première valeur strictement sous forme de booléen. Ils fonctionnent de la même manière que leurs homologues stricts. Supposons une variableA est vrai et variable B détient 20, alors -

Afficher des exemples

Opérateur La description Exemple
et Vérifie si les deux valeurs fournies sont véridiques, si oui, renvoie la valeur de la deuxième variable. (Logique et). A et B donneront 20
ou Vérifie si l'une des valeurs fournies est véridique. Renvoie la valeur la plus vraie. Sinon, renvoie false. (Logique ou). A ou B donnera vrai
ne pas Opérateur unaire qui inverse la valeur d'une entrée donnée. pas A donnera faux
&& Non strict and. Fonctionne de la même manière queand mais ne s'attend pas à ce que le premier argument soit un booléen. B && A donnera 20
|| Non strict or. Fonctionne de la même manière queor mais ne s'attend pas à ce que le premier argument soit un booléen. B || A donnera vrai
! Non strict not. Fonctionne de la même manière quenot mais ne s'attend pas à ce que l'argument soit un booléen. ! A donnera faux

NOTE −et , ou , && et || || sont des opérateurs de court-circuit. Cela signifie que si le premier argument deandest faux, alors il ne recherchera plus le second. Et si le premier argument deorest vrai, alors il ne recherchera pas le second. Par exemple,

false and raise("An error")  
#This won't raise an error as raise function wont get executed because of short
#circuiting nature of and operator

Opérateurs au niveau du bit

Les opérateurs au niveau du bit travaillent sur des bits et exécutent des opérations bit par bit. Elixir fournit des modules au niveau du bit dans le cadre du packageBitwise, donc pour les utiliser, vous devez utiliser le module bitwise. Pour l'utiliser, entrez la commande suivante dans votre shell -

use Bitwise

Supposons que A soit 5 et B soit 6 pour les exemples suivants -

Afficher des exemples

Opérateur La description Exemple
&&& L'opérateur et au niveau du bit copie un peu le résultat s'il existe dans les deux opérandes. A &&& B donnera 4
||| L'opérateur ou au niveau du bit copie un peu le résultat s'il existe dans l'un ou l'autre des opérandes. A ||| B donnera 7
>>> L'opérateur de décalage vers la droite au niveau du bit décale les premiers bits d'opérande vers la droite du nombre spécifié dans le deuxième opérande. A >>> B donnera 0
<<< L'opérateur de décalage gauche au niveau du bit décale les premiers bits d'opérande vers la gauche du nombre spécifié dans le deuxième opérande. A <<< B donnera 320
^^^ L'opérateur XOR au niveau du bit copie un bit dans le résultat uniquement s'il est différent sur les deux opérandes. A ^^^ B donnera 3
~~~ Unaire au niveau du bit n'inverse pas les bits sur le nombre donné. ~~~ A donnera -6

Opérateurs divers

Outre les opérateurs ci-dessus, Elixir fournit également une gamme d'autres opérateurs tels que Concatenation Operator, Match Operator, Pin Operator, Pipe Operator, String Match Operator, Code Point Operator, Capture Operator, Ternary Operator cela en fait un langage assez puissant.

Afficher des exemples

Le pattern matching est une technique dont Elixir hérite d'Erlang. C'est une technique très puissante qui nous permet d'extraire des sous-structures plus simples à partir de structures de données complexes telles que des listes, des tuples, des cartes, etc.

Un match comporte 2 parties principales, une left et un rightcôté. Le côté droit est une structure de données de toute nature. Le côté gauche tente de faire correspondre la structure de données sur le côté droit et de lier toutes les variables sur la gauche à la sous-structure respective sur la droite. Si aucune correspondance n'est trouvée, l'opérateur génère une erreur.

La correspondance la plus simple est une variable isolée à gauche et toute structure de données à droite. This variable will match anything. Par exemple,

x = 12
x = "Hello"
IO.puts(x)

Vous pouvez placer des variables à l'intérieur d'une structure afin de pouvoir capturer une sous-structure. Par exemple,

[var_1, _unused_var, var_2] = [{"First variable"}, 25, "Second variable" ]
IO.puts(var_1)
IO.puts(var_2)

Cela stockera les valeurs, {"First variable"}dans var_1 et"Second variable"dans var_2 . Il y a aussi un spécial_ variable (ou variables préfixées par '_') qui fonctionne exactement comme les autres variables mais indique à elixir, "Make sure something is here, but I don't care exactly what it is.". Dans l'exemple précédent, _unused_var était l'une de ces variables.

Nous pouvons faire correspondre des modèles plus compliqués en utilisant cette technique. Pourexample si vous souhaitez dérouler et obtenir un nombre dans un tuple qui se trouve dans une liste qui est elle-même dans une liste, vous pouvez utiliser la commande suivante -

[_, [_, {a}]] = ["Random string", [:an_atom, {24}]]
IO.puts(a)

Le programme ci-dessus génère le résultat suivant -

24

Cela liera a à 24. Les autres valeurs sont ignorées car nous utilisons «_».

Dans le pattern matching, si nous utilisons une variable sur le right, sa valeur est utilisée. Si vous souhaitez utiliser la valeur d'une variable sur la gauche, vous devrez utiliser l'opérateur pin.

Par exemple, si vous avez une variable «a» ayant la valeur 25 et que vous voulez la faire correspondre avec une autre variable «b» ayant la valeur 25, alors vous devez entrer -

a = 25
b = 25
^a = b

La dernière ligne correspond à la valeur actuelle de a, au lieu de l'attribuer, à la valeur de b. Si nous avons un ensemble non correspondant de côtés gauche et droit, l'opérateur de correspondance génère une erreur. Par exemple, si nous essayons de faire correspondre un tuple avec une liste ou une liste de taille 2 avec une liste de taille 3, une erreur sera affichée.

Les structures de prise de décision exigent que le programmeur spécifie une ou plusieurs conditions à évaluer ou à tester par le programme, ainsi qu'une ou plusieurs instructions à exécuter si la condition est déterminée comme étant true, et éventuellement d'autres instructions à exécuter si la condition est déterminée false.

Voici le général d'une structure de prise de décision typique trouvée dans la plupart des langages de programmation -

Elixir fournit des constructions conditionnelles if / else comme beaucoup d'autres langages de programmation. Il a également uncondinstruction qui appelle la première valeur vraie trouvée. Case est une autre instruction de flux de contrôle qui utilise la correspondance de modèles pour contrôler le flux du programme. Jetons un coup d'œil à eux.

Elixir fournit les types suivants de déclarations de prise de décision. Cliquez sur les liens suivants pour vérifier leurs détails.

Sr.No. Déclaration et description
1 si déclaration

Une instruction if consiste en une expression booléenne suivie de do, une ou plusieurs instructions exécutables et enfin un endmot-clé. Le code dans l'instruction if s'exécute uniquement si la condition booléenne est évaluée à true.

2 instruction if..else

Une instruction if peut être suivie d'une instruction else facultative (dans le bloc do..end), qui s'exécute lorsque l'expression booléenne est fausse.

3 sauf déclaration

Une instruction sauf a le même corps qu'une instruction if. Le code à l'intérieur de l'instruction sauf si la condition spécifiée est fausse.

4 instruction sauf..else

Une instruction sauf..else a le même corps qu'une instruction if..else. Le code à l'intérieur de l'instruction sauf si la condition spécifiée est fausse.

5 cond

Une instruction cond est utilisée lorsque nous voulons exécuter du code sur la base de plusieurs conditions. Cela fonctionne un peu comme une construction if ... else if ... .else dans plusieurs autres langages de programmation.

6 Cas

L'instruction Case peut être considérée comme un remplacement de l'instruction switch dans les langages impératifs. Case prend une variable / littéral et lui applique une correspondance de modèle avec différents cas. Si un cas correspond, Elixir exécute le code associé à ce cas et quitte l'instruction case.

Les chaînes d'Elixir sont insérées entre des guillemets doubles et sont encodées en UTF-8. Contrairement à C et C ++ où les chaînes par défaut sont encodées en ASCII et seuls 256 caractères différents sont possibles, UTF-8 se compose de 1 112 064 points de code. Cela signifie que le codage UTF-8 se compose de ces nombreux caractères possibles. Puisque les chaînes utilisent utf-8, nous pouvons également utiliser des symboles tels que: ö, ł, etc.

Créer une chaîne

Pour créer une variable chaîne, affectez simplement une chaîne à une variable -

str = "Hello world"

Pour l'imprimer sur votre console, appelez simplement le IO.puts function et passez-lui la variable str -

str = str = "Hello world" 
IO.puts(str)

Le programme ci-dessus génère le résultat suivant -

Hello World

Chaînes vides

Vous pouvez créer une chaîne vide en utilisant le littéral de chaîne, "". Par exemple,

a = ""
if String.length(a) === 0 do
   IO.puts("a is an empty string")
end

Le programme ci-dessus génère le résultat suivant.

a is an empty string

Interpolation de chaîne

L'interpolation de chaîne est un moyen de construire une nouvelle valeur String à partir d'un mélange de constantes, de variables, de littéraux et d'expressions en incluant leurs valeurs dans un littéral de chaîne. Elixir prend en charge l'interpolation de chaîne, pour utiliser une variable dans une chaîne, lors de son écriture, enveloppez-la avec des accolades et ajoutez les accolades avec un'#' signe.

Par exemple,

x = "Apocalypse" 
y = "X-men #{x}"
IO.puts(y)

Cela prendra la valeur de x et la remplacera par y. Le code ci-dessus générera le résultat suivant -

X-men Apocalypse

Concaténation de chaînes

Nous avons déjà vu l'utilisation de la concaténation de chaînes dans les chapitres précédents. L'opérateur '<>' est utilisé pour concaténer des chaînes dans Elixir. Pour concaténer 2 chaînes,

x = "Dark"
y = "Knight"
z = x <> " " <> y
IO.puts(z)

Le code ci-dessus génère le résultat suivant -

Dark Knight

Longueur de chaine

Pour obtenir la longueur de la chaîne, nous utilisons le String.lengthfonction. Passez la chaîne comme paramètre et il vous montrera sa taille. Par exemple,

IO.puts(String.length("Hello"))

Lors de l'exécution du programme ci-dessus, il produit le résultat suivant -

5

Inverser une chaîne

Pour inverser une chaîne, transmettez-la à la fonction String.reverse. Par exemple,

IO.puts(String.reverse("Elixir"))

Le programme ci-dessus génère le résultat suivant -

rixilE

Comparaison des chaînes

Pour comparer 2 chaînes, nous pouvons utiliser les opérateurs == ou ===. Par exemple,

var_1 = "Hello world"
var_2 = "Hello Elixir"
if var_1 === var_2 do
   IO.puts("#{var_1} and #{var_2} are the same")
else
   IO.puts("#{var_1} and #{var_2} are not the same")
end

Le programme ci-dessus génère le résultat suivant -

Hello world and Hello elixir are not the same.

Correspondance de chaîne

Nous avons déjà vu l'utilisation de l'opérateur de correspondance = ~ string. Pour vérifier si une chaîne correspond à une expression régulière, nous pouvons également utiliser l'opérateur de correspondance de chaîne ou String.match? fonction. Par exemple,

IO.puts(String.match?("foo", ~r/foo/))
IO.puts(String.match?("bar", ~r/foo/))

Le programme ci-dessus génère le résultat suivant -

true 
false

Cela peut également être réalisé en utilisant l'opérateur = ~. Par exemple,

IO.puts("foo" =~ ~r/foo/)

Le programme ci-dessus génère le résultat suivant -

true

Fonctions de chaîne

Elixir prend en charge un grand nombre de fonctions liées aux chaînes, certaines des plus utilisées sont répertoriées dans le tableau suivant.

Sr.No. Fonction et son objectif
1

at(string, position)

Renvoie le graphème à la position de la chaîne utf8 donnée. Si la position est supérieure à la longueur de la chaîne, elle renvoie nil

2

capitalize(string)

Convertit le premier caractère de la chaîne donnée en majuscules et le reste en minuscules

3

contains?(string, contents)

Vérifie si la chaîne contient l'un des contenus donnés

4

downcase(string)

Convertit tous les caractères de la chaîne donnée en minuscules

5

ends_with?(string, suffixes)

Renvoie true si la chaîne se termine par l'un des suffixes donnés

6

first(string)

Renvoie le premier graphème d'une chaîne utf8, nil si la chaîne est vide

sept

last(string)

Renvoie le dernier graphème d'une chaîne utf8, nil si la chaîne est vide

8

replace(subject, pattern, replacement, options \\ [])

Renvoie une nouvelle chaîne créée en remplaçant les occurrences de motif dans le sujet par remplacement

9

slice(string, start, len)

Renvoie une sous-chaîne commençant au début du décalage et de longueur len

dix

split(string)

Divise une chaîne en sous-chaînes à chaque occurrence d'espaces Unicode avec les espaces de début et de fin ignorés. Les groupes d'espaces sont traités comme une seule occurrence. Les divisions ne se produisent pas sur les espaces blancs insécables

11

upcase(string)

Convertit tous les caractères de la chaîne donnée en majuscules

Binaires

Un binaire n'est qu'une séquence d'octets. Les binaires sont définis en utilisant<< >>. Par exemple:

<< 0, 1, 2, 3 >>

Bien entendu, ces octets peuvent être organisés de n'importe quelle manière, même dans une séquence qui n'en fait pas une chaîne valide. Par exemple,

<< 239, 191, 191 >>

Les chaînes sont également des binaires. Et l'opérateur de concaténation de chaîne<> est en fait un opérateur de concaténation binaire:

IO.puts(<< 0, 1 >> <> << 2, 3 >>)

Le code ci-dessus génère le résultat suivant -

<< 0, 1, 2, 3 >>

Notez le caractère ł. Comme il est encodé en utf-8, cette représentation de caractère occupe 2 octets.

Étant donné que chaque nombre représenté dans un binaire est censé être un octet, lorsque cette valeur passe de 255, elle est tronquée. Pour éviter cela, nous utilisons le modificateur de taille pour spécifier le nombre de bits que nous voulons que ce nombre prenne. Par exemple -

IO.puts(<< 256 >>) # truncated, it'll print << 0 >>
IO.puts(<< 256 :: size(16) >>) #Takes 16 bits/2 bytes, will print << 1, 0 >>

Le programme ci-dessus générera le résultat suivant -

<< 0 >>
<< 1, 0 >>

Nous pouvons également utiliser le modificateur utf8, si un caractère est un point de code alors, il sera produit dans la sortie; sinon les octets -

IO.puts(<< 256 :: utf8 >>)

Le programme ci-dessus génère le résultat suivant -

Ā

Nous avons également une fonction appelée is_binaryqui vérifie si une variable donnée est un binaire. Notez que seules les variables stockées sous forme de multiples de 8 bits sont des binaires.

Bitstrings

Si nous définissons un binaire à l'aide du modificateur de taille et lui passons une valeur qui n'est pas un multiple de 8, nous nous retrouvons avec une chaîne de bits au lieu d'un binaire. Par exemple,

bs = << 1 :: size(1) >>
IO.puts(bs)
IO.puts(is_binary(bs))
IO.puts(is_bitstring(bs))

Le programme ci-dessus génère le résultat suivant -

<< 1::size(1) >>
false
true

Cela signifie que la variable bsn'est pas un binaire mais plutôt une chaîne de bits. Nous pouvons également dire qu'un binaire est une chaîne de bits où le nombre de bits est divisible par 8. La correspondance de motifs fonctionne de la même manière sur les binaires ainsi que sur les chaînes de bits.

Une liste de caractères n'est rien de plus qu'une liste de caractères. Considérez le programme suivant pour comprendre la même chose.

IO.puts('Hello')
IO.puts(is_list('Hello'))

Le programme ci-dessus génère le résultat suivant -

Hello
true

Au lieu de contenir des octets, une liste de caractères contient les points de code des caractères entre guillemets simples. So while the double-quotes represent a string (i.e. a binary), singlequotes represent a char list (i.e. a list). Notez que IEx ne générera que des points de code en sortie si l'un des caractères est en dehors de la plage ASCII.

Les listes de caractères sont principalement utilisées lors de l'interfaçage avec Erlang, en particulier les anciennes bibliothèques qui n'acceptent pas les binaires comme arguments. Vous pouvez convertir une liste de caractères en chaîne et inversement en utilisant les fonctions to_string (char_list) et to_char_list (string) -

IO.puts(is_list(to_char_list("hełło")))
IO.puts(is_binary(to_string ('hełło')))

Le programme ci-dessus génère le résultat suivant -

true
true

NOTE - Les fonctions to_string et to_char_list sont polymorphes, c'est-à-dire qu'ils peuvent prendre plusieurs types d'entrées comme des atomes, des entiers et les convertir respectivement en chaînes et en listes de caractères.

Listes (liées)

Une liste chaînée est une liste hétérogène d'éléments qui sont stockés à différents emplacements de la mémoire et sont conservés à l'aide de références. Les listes liées sont des structures de données particulièrement utilisées dans la programmation fonctionnelle.

Elixir utilise des crochets pour spécifier une liste de valeurs. Les valeurs peuvent être de n'importe quel type -

[1, 2, true, 3]

Quand Elixir voit une liste de nombres ASCII imprimables, Elixir l'affiche sous forme de liste de caractères (littéralement une liste de caractères). Chaque fois que vous voyez une valeur dans IEx et que vous n'êtes pas sûr de ce que c'est, vous pouvez utiliser lei fonction pour récupérer des informations à ce sujet.

IO.puts([104, 101, 108, 108, 111])

Les caractères ci-dessus dans la liste sont tous imprimables. Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

hello

Vous pouvez également définir des listes dans l'autre sens, en utilisant des guillemets simples -

IO.puts(is_list('Hello'))

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

true

Gardez à l'esprit que les représentations entre guillemets simples et doubles ne sont pas équivalentes dans Elixir car elles sont représentées par des types différents.

Longueur d'une liste

Pour trouver la longueur d'une liste, nous utilisons la fonction length comme dans le programme suivant -

IO.puts(length([1, 2, :true, "str"]))

Le programme ci-dessus génère le résultat suivant -

4

Concaténation et soustraction

Deux listes peuvent être concaténées et soustraites à l'aide de la ++ et --les opérateurs. Prenons l'exemple suivant pour comprendre les fonctions.

IO.puts([1, 2, 3] ++ [4, 5, 6])
IO.puts([1, true, 2, false, 3, true] -- [true, false])

Cela vous donnera une chaîne concaténée dans le premier cas et une chaîne soustraite dans le second. Le programme ci-dessus génère le résultat suivant -

[1, 2, 3, 4, 5, 6]
[1, 2, 3, true]

Tête et queue d'une liste

La tête est le premier élément d'une liste et la queue est le reste d'une liste. Ils peuvent être récupérés avec les fonctionshd et tl. Attribuons une liste à une variable et récupérons sa tête et sa queue.

list = [1, 2, 3]
IO.puts(hd(list))
IO.puts(tl(list))

Cela nous donnera la tête et la queue de la liste en sortie. Le programme ci-dessus génère le résultat suivant -

1
[2, 3]

Note - Obtenir la tête ou la queue d'une liste vide est une erreur.

Autres fonctions de liste

La bibliothèque standard Elixir fournit de nombreuses fonctions pour gérer les listes. Nous allons jeter un œil à certains de ceux-ci ici. Vous pouvez consulter le reste ici Liste .

S.no. Nom et description de la fonction
1

delete(list, item)

Supprime l'élément donné de la liste. Renvoie une liste sans l'élément. Si l'élément apparaît plus d'une fois dans la liste, seule la première occurrence est supprimée.

2

delete_at(list, index)

Produit une nouvelle liste en supprimant la valeur à l'index spécifié. Les indices négatifs indiquent un décalage par rapport à la fin de la liste. Si l'index est hors limites, la liste d'origine est renvoyée.

3

first(list)

Renvoie le premier élément de la liste ou nil si la liste est vide.

4

flatten(list)

Aplatit la liste donnée de listes imbriquées.

5

insert_at(list, index, value)

Renvoie une liste avec une valeur insérée à l'index spécifié. Notez que l'index est plafonné à la longueur de la liste. Les indices négatifs indiquent un décalage par rapport à la fin de la liste.

6

last(list)

Renvoie le dernier élément de la liste ou nil si la liste est vide.

Tuples

Les tuples sont également des structures de données qui stockent un certain nombre d'autres structures en leur sein. Contrairement aux listes, ils stockent des éléments dans un bloc de mémoire contigu. Cela signifie que l'accès à un élément de tuple par index ou l'obtention de la taille de tuple est une opération rapide. Les index commencent à zéro.

Elixir utilise des accolades pour définir les tuples. Comme les listes, les tuples peuvent contenir n'importe quelle valeur -

{:ok, "hello"}

Longueur d'un tuple

Pour obtenir la longueur d'un tuple, utilisez le tuple_size fonctionne comme dans le programme suivant -

IO.puts(tuple_size({:ok, "hello"}))

Le programme ci-dessus génère le résultat suivant -

2

Ajout d'une valeur

Pour ajouter une valeur au tuple, utilisez la fonction Tuple.append -

tuple = {:ok, "Hello"}
Tuple.append(tuple, :world)

Cela créera et retournera un nouveau tuple: {: ok, "Hello",: world}

Insérer une valeur

Pour insérer une valeur à une position donnée, on peut soit utiliser la Tuple.insert_at fonction ou le put_elemfonction. Prenons l'exemple suivant pour comprendre la même chose -

tuple = {:bar, :baz}
new_tuple_1 = Tuple.insert_at(tuple, 0, :foo)
new_tuple_2 = put_elem(tuple, 1, :foobar)

Remarquerez que put_elem et insert_ata renvoyé de nouveaux tuples. Le tuple d'origine stocké dans la variable tuple n'a pas été modifié car les types de données Elixir sont immuables. En étant immuable, le code Elixir est plus facile à raisonner car vous n'avez jamais à vous inquiéter si un code particulier mute votre structure de données en place.

Tuples et listes

Quelle est la différence entre les listes et les tuples?

Les listes sont stockées en mémoire sous forme de listes chaînées, ce qui signifie que chaque élément d'une liste contient sa valeur et pointe vers l'élément suivant jusqu'à ce que la fin de la liste soit atteinte. Nous appelons chaque paire de valeur et de pointeur une cellule contre. Cela signifie que l'accès à la longueur d'une liste est une opération linéaire: nous devons parcourir toute la liste afin de déterminer sa taille. La mise à jour d'une liste est rapide tant que nous ajoutons des éléments au début.

Les tuples, par contre, sont stockés de manière contiguë dans la mémoire. Cela signifie que l'obtention de la taille du tuple ou l'accès à un élément par index est rapide. Cependant, la mise à jour ou l'ajout d'éléments aux tuples est coûteux car il nécessite de copier le tuple entier en mémoire.

Jusqu'à présent, nous n'avons pas discuté des structures de données associatives, c'est-à-dire des structures de données qui peuvent associer une certaine valeur (ou plusieurs valeurs) à une clé. Différents langages appellent ces fonctionnalités avec des noms différents comme des dictionnaires, des hachages, des tableaux associatifs, etc.

Dans Elixir, nous avons deux structures de données associatives principales: les listes de mots clés et les cartes. Dans ce chapitre, nous nous concentrerons sur les listes de mots clés.

Dans de nombreux langages de programmation fonctionnelle, il est courant d'utiliser une liste de tuples à 2 éléments comme représentation d'une structure de données associative. Dans Elixir, lorsque nous avons une liste de tuples et que le premier élément du tuple (c'est-à-dire la clé) est un atome, nous l'appelons une liste de mots clés. Prenons l'exemple suivant pour comprendre la même chose -

list = [{:a, 1}, {:b, 2}]

Elixir prend en charge une syntaxe spéciale pour définir de telles listes. On peut placer les deux points à la fin de chaque atome et se débarrasser entièrement des tuples. Par exemple,

list_1 = [{:a, 1}, {:b, 2}]
list_2 = [a: 1, b: 2]
IO.puts(list_1 == list_2)

Le programme ci-dessus générera le résultat suivant -

true

Ces deux éléments représentent une liste de mots clés. Puisque les listes de mots clés sont également des listes, nous pouvons utiliser toutes les opérations que nous avons utilisées sur les listes.

Pour récupérer la valeur associée à un atome dans la liste de mots clés, passez l'atome à [] après le nom de la liste -

list = [a: 1, b: 2]
IO.puts(list[:a])

Le programme ci-dessus génère le résultat suivant -

1

Les listes de mots-clés ont trois caractéristiques spéciales -

  • Les clés doivent être des atomes.
  • Les clés sont commandées, comme spécifié par le développeur.
  • Les clés peuvent être données plus d'une fois.

Afin de manipuler les listes de mots clés, Elixir fournit le module Keyword . N'oubliez pas, cependant, que les listes de mots clés sont simplement des listes et, en tant que telles, elles fournissent les mêmes caractéristiques de performances linéaires que les listes. Plus la liste est longue, plus il faudra de temps pour trouver une clé, compter le nombre d'éléments, etc. Pour cette raison, les listes de mots clés sont utilisées dans Elixir principalement comme des options. Si vous avez besoin de stocker de nombreux éléments ou de garantir des associés à une clé avec une seule valeur maximale, vous devez utiliser des cartes à la place.

Accéder à une clé

Pour accéder aux valeurs associées à une clé donnée, nous utilisons le Keyword.getfonction. Il renvoie la première valeur associée à la clé donnée. Pour obtenir toutes les valeurs, nous utilisons la fonction Keyword.get_values. Par exemple -

kl = [a: 1, a: 2, b: 3] 
IO.puts(Keyword.get(kl, :a)) 
IO.puts(Keyword.get_values(kl))

Le programme ci-dessus générera le résultat suivant -

1
[1, 2]

Insérer une clé

Pour ajouter une nouvelle valeur, utilisez Keyword.put_new. Si la clé existe déjà, sa valeur reste inchangée -

kl = [a: 1, a: 2, b: 3]
kl_new = Keyword.put_new(kl, :c, 5)
IO.puts(Keyword.get(kl_new, :c))

Lorsque le programme ci-dessus est exécuté, il produit une nouvelle liste de mots clés avec une clé supplémentaire, c et génère le résultat suivant -

5

Supprimer une clé

Si vous souhaitez supprimer toutes les entrées d'une clé, utilisez Keyword.delete; pour supprimer uniquement la première entrée d'une clé, utilisez Keyword.delete_first.

kl = [a: 1, a: 2, b: 3, c: 0]
kl = Keyword.delete_first(kl, :b)
kl = Keyword.delete(kl, :a)

IO.puts(Keyword.get(kl, :a))
IO.puts(Keyword.get(kl, :b))
IO.puts(Keyword.get(kl, :c))

Cela supprimera le premier b dans la liste et tous les adans la liste. Lorsque le programme ci-dessus est exécuté, il générera le résultat suivant -

0

Les listes de mots-clés sont un moyen pratique d'adresser le contenu stocké dans des listes par clé, mais en dessous, Elixir parcourt toujours la liste. Cela peut convenir si vous avez d'autres plans pour cette liste nécessitant de parcourir tout cela, mais cela peut être une surcharge inutile si vous prévoyez d'utiliser les clés comme seule approche des données.

C'est là que les cartes viennent à votre secours. Chaque fois que vous avez besoin d'un magasin clé-valeur, les cartes sont la structure de données «aller» dans Elixir.

Créer une carte

Une carte est créée en utilisant la syntaxe% {} -

map = %{:a => 1, 2 => :b}

Par rapport aux listes de mots clés, nous pouvons déjà voir deux différences -

  • Les cartes autorisent n'importe quelle valeur comme clé.
  • Les clés de Maps ne suivent aucun ordre.

Accéder à une clé

Pour accéder à la valeur associée à une clé, les cartes utilisent la même syntaxe que les listes de mots clés -

map = %{:a => 1, 2 => :b}
IO.puts(map[:a])
IO.puts(map[2])

Lorsque le programme ci-dessus est exécuté, il génère le résultat suivant -

1
b

Insérer une clé

Pour insérer une clé dans une carte, nous utilisons le Dict.put_new fonction qui prend la carte, la nouvelle clé et la nouvelle valeur comme arguments -

map = %{:a => 1, 2 => :b}
new_map = Dict.put_new(map, :new_val, "value") 
IO.puts(new_map[:new_val])

Cela insérera la paire clé-valeur :new_val - "value"dans une nouvelle carte. Lorsque le programme ci-dessus est exécuté, il génère le résultat suivant -

"value"

Mettre à jour une valeur

Pour mettre à jour une valeur déjà présente dans la carte, vous pouvez utiliser la syntaxe suivante -

map = %{:a => 1, 2 => :b}
new_map = %{ map | a: 25}
IO.puts(new_map[:a])

Lorsque le programme ci-dessus est exécuté, il génère le résultat suivant -

25

Correspondance de motif

Contrairement aux listes de mots clés, les cartes sont très utiles avec la correspondance de modèles. Lorsqu'une carte est utilisée dans un modèle, elle correspondra toujours à un sous-ensemble de la valeur donnée -

%{:a => a} = %{:a => 1, 2 => :b}
IO.puts(a)

Le programme ci-dessus génère le résultat suivant -

1

Cela correspondra a avec 1. Et par conséquent, il générera la sortie comme1.

Comme indiqué ci-dessus, une carte correspond tant que les clés du modèle existent dans la carte donnée. Par conséquent, une carte vide correspond à toutes les cartes.

Les variables peuvent être utilisées lors de l'accès, de la correspondance et de l'ajout de clés de carte -

n = 1
map = %{n => :one}
%{^n => :one} = %{1 => :one, 2 => :two, 3 => :three}

Le module Map fournit une API très similaire au module Keyword avec des fonctions pratiques pour manipuler les cartes. Vous pouvez utiliser des fonctions telles queMap.get, Map.delete, pour manipuler des cartes.

Cartes avec clés Atom

Les cartes sont livrées avec quelques propriétés intéressantes. Lorsque toutes les clés d'une carte sont des atomes, vous pouvez utiliser la syntaxe du mot-clé pour plus de commodité -

map = %{:a => 1, 2 => :b} 
IO.puts(map.a)

Une autre propriété intéressante des cartes est qu'elles fournissent leur propre syntaxe pour la mise à jour et l'accès aux clés atomiques -

map = %{:a => 1, 2 => :b}
IO.puts(map.a)

Le programme ci-dessus génère le résultat suivant -

1

Notez que pour accéder aux clés atom de cette manière, il doit exister ou le programme ne fonctionnera pas.

Dans Elixir, nous regroupons plusieurs fonctions en modules. Nous avons déjà utilisé différents modules dans les chapitres précédents tels que le module String, le module Bitwise, le module Tuple, etc.

Afin de créer nos propres modules dans Elixir, nous utilisons le defmodulemacro. Nous utilisons ledef macro pour définir les fonctions de ce module -

defmodule Math do
   def sum(a, b) do
      a + b
   end
end

Dans les sections suivantes, nos exemples vont devenir plus longs et il peut être difficile de tous les saisir dans le shell. Nous devons apprendre à compiler du code Elixir et également à exécuter des scripts Elixir.

Compilation

Il est toujours pratique d'écrire des modules dans des fichiers afin qu'ils puissent être compilés et réutilisés. Supposons que nous ayons un fichier nommé math.ex avec le contenu suivant -

defmodule Math do
   def sum(a, b) do
      a + b
   end
end

Nous pouvons compiler les fichiers en utilisant la commande -elixirc :

$ elixirc math.ex

Cela générera un fichier nommé Elixir.Math.beamcontenant le bytecode du module défini. Si nous commençonsiexencore une fois, notre définition de module sera disponible (à condition que iex soit démarré dans le même répertoire que le fichier bytecode). Par exemple,

IO.puts(Math.sum(1, 2))

Le programme ci-dessus générera le résultat suivant -

3

Mode scripté

En plus de l'extension de fichier Elixir .ex, Elixir prend également en charge .exsfichiers pour les scripts. Elixir traite les deux fichiers exactement de la même manière, la seule différence réside dans l'objectif..ex les fichiers sont destinés à être compilés tandis que les fichiers .exs sont utilisés pour scripting. Lorsqu'elles sont exécutées, les deux extensions compilent et chargent leurs modules en mémoire, bien que seulement.ex Les fichiers écrivent leur bytecode sur le disque au format de fichiers .beam.

Par exemple, si nous voulions exécuter le Math.sum dans le même fichier, nous pouvons utiliser le .exs de la manière suivante -

Math.exs

defmodule Math do
   def sum(a, b) do
      a + b
   end
end
IO.puts(Math.sum(1, 2))

Nous pouvons l'exécuter en utilisant la commande Elixir -

$ elixir math.exs

Le programme ci-dessus générera le résultat suivant -

3

Le fichier sera compilé en mémoire et exécuté, imprimant «3» comme résultat. Aucun fichier bytecode ne sera créé.

Imbrication de modules

Les modules peuvent être imbriqués dans Elixir. Cette fonctionnalité du langage nous aide à mieux organiser notre code. Pour créer des modules imbriqués, nous utilisons la syntaxe suivante -

defmodule Foo do
   #Foo module code here
   defmodule Bar do
      #Bar module code here
   end
end

L'exemple donné ci-dessus définira deux modules: Foo et Foo.Bar. Le second est accessible commeBar à l'intérieur Footant qu'ils sont dans la même portée lexicale. Si, plus tard, leBar module est déplacé en dehors de la définition du module Foo, il doit être référencé par son nom complet (Foo.Bar) ou un alias doit être défini à l'aide de la directive alias décrite dans le chapitre alias.

Note- Dans Elixir, il n'est pas nécessaire de définir le module Foo pour définir le module Foo.Bar, car le langage traduit tous les noms de modules en atomes. Vous pouvez définir des modules imbriqués de manière arbitraire sans définir de module dans la chaîne. Par exemple, vous pouvez définirFoo.Bar.Baz sans définir Foo ou Foo.Bar.

Afin de faciliter la réutilisation des logiciels, Elixir fournit trois directives - alias, require et import. Il fournit également une macro appelée use qui est résumée ci-dessous -

# Alias the module so it can be called as Bar instead of Foo.Bar
alias Foo.Bar, as: Bar

# Ensure the module is compiled and available (usually for macros)
require Foo

# Import functions from Foo so they can be called without the `Foo.` prefix
import Foo

# Invokes the custom code defined in Foo as an extension point
use Foo

Voyons maintenant en détail chaque directive.

alias

La directive alias vous permet de configurer des alias pour tout nom de module donné. Par exemple, si vous souhaitez donner un alias'Str' dans le module String, vous pouvez simplement écrire -

alias String, as: Str
IO.puts(Str.length("Hello"))

Le programme ci-dessus génère le résultat suivant -

5

Un alias est donné au String module comme Str. Maintenant, lorsque nous appelons une fonction en utilisant le littéral Str, elle fait en fait référence auStringmodule. Ceci est très utile lorsque nous utilisons des noms de modules très longs et que nous voulons les remplacer par des plus courts dans la portée actuelle.

NOTE - Alias MUST commencez par une majuscule.

Les alias ne sont valides que dans le lexical scope ils sont appelés. Par exemple, si vous avez 2 modules dans un fichier et que vous créez un alias dans l'un des modules, cet alias ne sera pas accessible dans le deuxième module.

Si vous donnez le nom d'un module intégré, comme String ou Tuple, comme alias à un autre module, pour accéder au module intégré, vous devrez le préfixer avec "Elixir.". Par exemple,

alias List, as: String
#Now when we use String we are actually using List.
#To use the string module: 
IO.puts(Elixir.String.length("Hello"))

Lorsque le programme ci-dessus est exécuté, il génère le résultat suivant -

5

exiger

Elixir fournit des macros comme mécanisme de méta-programmation (écriture de code qui génère du code).

Les macros sont des morceaux de code qui sont exécutés et développés au moment de la compilation. Cela signifie que pour utiliser une macro, nous devons garantir que son module et son implémentation sont disponibles lors de la compilation. Ceci est fait avec lerequire directif.

Integer.is_odd(3)

Lorsque le programme ci-dessus est exécuté, il générera le résultat suivant -

** (CompileError) iex:1: you must require Integer before invoking the macro Integer.is_odd/1

En Elixir, Integer.is_odd est défini comme un macro. Cette macro peut être utilisée comme garde. Cela signifie que, pour invoquerInteger.is_odd, nous aurons besoin du module Integer.

Utilisez le require Integer et exécutez le programme comme indiqué ci-dessous.

require Integer
Integer.is_odd(3)

Cette fois, le programme s'exécutera et produira la sortie comme: true.

En général, un module n'est pas requis avant l'utilisation, sauf si nous voulons utiliser les macros disponibles dans ce module. Une tentative d'appeler une macro qui n'a pas été chargée déclenchera une erreur. Notez que comme la directive alias, require a également une portée lexicale . Nous parlerons plus en détail des macros dans un chapitre ultérieur.

importer

Nous utilisons le importdirective pour accéder facilement aux fonctions ou aux macros d'autres modules sans utiliser le nom complet. Par exemple, si nous voulons utiliser leduplicate fonction du module List plusieurs fois, nous pouvons simplement l'importer.

import List, only: [duplicate: 2]

Dans ce cas, nous importons uniquement la fonction duplicate (avec l'argument de longueur de liste 2) depuis List. Bien que:only est facultative, son utilisation est recommandée afin d'éviter d'importer toutes les fonctions d'un module donné à l'intérieur de l'espace de noms. :except pourrait également être donné en option pour tout importer dans un module sauf une liste de fonctions.

le import directive prend également en charge :macros et :functions à donner à :only. Par exemple, pour importer toutes les macros, un utilisateur peut écrire -

import Integer, only: :macros

Notez que l'importation est aussi Lexically scopedtout comme les directives require et alias. Notez également que'import'ing a module also 'require's it.

utilisation

Bien que ce ne soit pas une directive, use est une macro étroitement liée à requirequi vous permet d'utiliser un module dans le contexte actuel. La macro use est fréquemment utilisée par les développeurs pour intégrer des fonctionnalités externes dans la portée lexicale actuelle, souvent des modules. Comprenons la directive use à travers un exemple -

defmodule Example do 
   use Feature, option: :value 
end

Use est une macro qui transforme ce qui précède en -

defmodule Example do
   require Feature
   Feature.__using__(option: :value)
end

le use Module requiert d'abord le module, puis appelle le __using__macro sur le module. Elixir a d'excellentes capacités de métaprogrammation et il a des macros pour générer du code au moment de la compilation. La macro _ _using__ est appelée dans l'instance ci-dessus et le code est injecté dans notre contexte local. Le contexte local est l'endroit où la macro use a été appelée au moment de la compilation.

Une fonction est un ensemble d'instructions organisées ensemble pour effectuer une tâche spécifique. Les fonctions de programmation fonctionnent principalement comme des fonctions en mathématiques. Vous donnez une entrée aux fonctions, elles génèrent une sortie basée sur l'entrée fournie.

Il existe 2 types de fonctions dans Elixir -

Fonction anonyme

Fonctions définies à l'aide du fn..end constructsont des fonctions anonymes. Ces fonctions sont parfois également appelées lambdas. Ils sont utilisés en les affectant à des noms de variables.

Fonction nommée

Fonctions définies à l'aide du def keywordsont des fonctions nommées. Ce sont des fonctions natives fournies dans Elixir.

Fonctions anonymes

Tout comme son nom l'indique, une fonction anonyme n'a pas de nom. Celles-ci sont fréquemment transmises à d'autres fonctions. Pour définir une fonction anonyme dans Elixir, nous avons besoin dufn et endmots clés. Dans ceux-ci, nous pouvons définir n'importe quel nombre de paramètres et de corps de fonction séparés par->. Par exemple,

sum = fn (a, b) -> a + b end
IO.puts(sum.(1, 5))

Lors de l'exécution du programme ci-dessus, est exécuté, il génère le résultat suivant -

6

Notez que ces fonctions ne sont pas appelées comme les fonctions nommées. Nous avons un '.'entre le nom de la fonction et ses arguments.

Utilisation de l'opérateur de capture

Nous pouvons également définir ces fonctions à l'aide de l'opérateur de capture. C'est une méthode plus simple pour créer des fonctions. Nous allons maintenant définir la fonction somme ci-dessus à l'aide de l'opérateur de capture,

sum = &(&1 + &2) 
IO.puts(sum.(1, 2))

Lorsque le programme ci-dessus est exécuté, il génère le résultat suivant -

3

Dans la version abrégée, nos paramètres ne sont pas nommés mais nous sont disponibles sous les noms & 1, & 2, & 3, etc.

Fonctions de correspondance de modèles

L'appariement de modèles n'est pas seulement limité aux variables et aux structures de données. Nous pouvons utiliser la correspondance de motifs pour rendre nos fonctions polymorphes. Par exemple, nous allons déclarer une fonction qui peut prendre 1 ou 2 entrées (dans un tuple) et les imprimer sur la console,

handle_result = fn
   {var1} -> IO.puts("#{var1} found in a tuple!")
   {var_2, var_3} -> IO.puts("#{var_2} and #{var_3} found!")
end
handle_result.({"Hey people"})
handle_result.({"Hello", "World"})

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

Hey people found in a tuple!
Hello and World found!

Fonctions nommées

Nous pouvons définir des fonctions avec des noms afin que nous puissions facilement nous y référer ultérieurement. Les fonctions nommées sont définies dans un module à l'aide du mot-clé def. Les fonctions nommées sont toujours définies dans un module. Pour appeler des fonctions nommées, nous devons les référencer en utilisant leur nom de module.

Voici la syntaxe des fonctions nommées -

def function_name(argument_1, argument_2) do
   #code to be executed when function is called
end

Définissons maintenant notre fonction nommée somme dans le module Math.

defmodule Math do
   def sum(a, b) do
      a + b
   end
end

IO.puts(Math.sum(5, 6))

Lors de l'exécution du programme ci-dessus, il produit le résultat suivant -

11

Pour les fonctions à 1 ligne, il existe une notation abrégée pour définir ces fonctions, en utilisant do:. Par exemple -

defmodule Math do
   def sum(a, b), do: a + b
end
IO.puts(Math.sum(5, 6))

Lors de l'exécution du programme ci-dessus, il produit le résultat suivant -

11

Fonctions privées

Elixir nous offre la possibilité de définir des fonctions privées accessibles depuis le module dans lequel elles sont définies. Pour définir une fonction privée, utilisezdefp au lieu de def. Par exemple,

defmodule Greeter do
   def hello(name), do: phrase <> name
   defp phrase, do: "Hello "
end

Greeter.hello("world")

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

Hello world

Mais si nous essayons simplement d'appeler explicitement la fonction de phrase, en utilisant le Greeter.phrase() fonction, cela provoquera une erreur.

Arguments par défaut

Si nous voulons une valeur par défaut pour un argument, nous utilisons la argument \\ value syntaxe -

defmodule Greeter do
   def hello(name, country \\ "en") do
      phrase(country) <> name
   end

   defp phrase("en"), do: "Hello, "
   defp phrase("es"), do: "Hola, "
end

Greeter.hello("Ayush", "en")
Greeter.hello("Ayush")
Greeter.hello("Ayush", "es")

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

Hello, Ayush
Hello, Ayush
Hola, Ayush

La récursivité est une méthode dans laquelle la solution à un problème dépend des solutions aux instances plus petites du même problème. La plupart des langages de programmation informatique prennent en charge la récursivité en permettant à une fonction de s'appeler elle-même dans le texte du programme.

Idéalement, les fonctions récursives ont une condition de fin. Cette condition de fin, également connue sous le nom de cas de base, arrête de rentrer dans la fonction et d'ajouter des appels de fonction à la pile. C'est là que s'arrête l'appel de fonction récursive. Prenons l'exemple suivant pour mieux comprendre la fonction récursive.

defmodule Math do
   def fact(res, num) do
   if num === 1 do
      res
   else
      new_res = res * num
      fact(new_res, num-1)
      end
   end
end

IO.puts(Math.fact(1,5))

Lorsque le programme ci-dessus est exécuté, il génère le résultat suivant -

120

Donc, dans la fonction ci-dessus, Math.fact, nous calculons la factorielle d'un nombre. Notez que nous appelons la fonction en elle-même. Voyons maintenant comment cela fonctionne.

Nous lui avons fourni 1 et le nombre dont nous voulons calculer la factorielle. La fonction vérifie si le nombre est 1 ou non et renvoie res s'il est 1(Ending condition). Sinon, il crée une variable new_res et lui affecte la valeur de res * current num. Il renvoie la valeur renvoyée par notre fonction call fact (new_res, num-1) . Cela se répète jusqu'à ce que nous obtenions num comme 1. Une fois que cela se produit, nous obtenons le résultat.

Prenons un autre exemple, imprimant chaque élément de la liste un par un. Pour ce faire, nous utiliserons lehd et tl fonctions des listes et correspondance de motifs dans les fonctions -

a = ["Hey", 100, 452, :true, "People"]
defmodule ListPrint do
   def print([]) do
   end
   def print([head | tail]) do 
      IO.puts(head)
      print(tail)
   end
end

ListPrint.print(a)

La première fonction d'impression est appelée lorsque nous avons une liste vide(ending condition). Sinon, alors la deuxième fonction d'impression sera appelée qui divisera la liste en 2 et affectera le premier élément de la liste à la tête et le reste de la liste à la queue. La tête est alors imprimée et nous appelons à nouveau la fonction d'impression avec le reste de la liste, c'est-à-dire la queue. Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

Hey
100
452
true
People

En raison de l'immuabilité, les boucles dans Elixir (comme dans tout langage de programmation fonctionnel) sont écrites différemment des langages impératifs. Par exemple, dans un langage impératif comme C, vous écrirez -

for(i = 0; i < 10; i++) {
   printf("%d", array[i]);
}

Dans l'exemple donné ci-dessus, nous mutons à la fois le tableau et la variable i. La mutation n'est pas possible dans Elixir. Au lieu de cela, les langages fonctionnels reposent sur la récursivité: une fonction est appelée récursivement jusqu'à ce qu'une condition soit atteinte qui empêche l'action récursive de se poursuivre. Aucune donnée n'est mutée dans ce processus.

Écrivons maintenant une boucle simple utilisant la récursivité qui affiche bonjour n fois.

defmodule Loop do
   def print_multiple_times(msg, n) when n <= 1 do
      IO.puts msg
   end

   def print_multiple_times(msg, n) do
      IO.puts msg
      print_multiple_times(msg, n - 1)
   end
end

Loop.print_multiple_times("Hello", 10)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

Hello
Hello
Hello
Hello
Hello
Hello
Hello
Hello
Hello
Hello

Nous avons utilisé les techniques de correspondance de modèles de fonction et la récursivité pour implémenter avec succès une boucle. Les définitions récursives sont difficiles à comprendre mais la conversion de boucles en récursivité est facile.

Elixir nous fournit le Enum module. Ce module est utilisé pour les appels en boucle les plus itératifs car il est beaucoup plus facile de les utiliser que d'essayer de trouver des définitions récursives pour les mêmes. Nous en discuterons dans le prochain chapitre. Vos propres définitions récursives ne doivent être utilisées que lorsque vous ne trouvez pas de solution utilisant ce module. Ces fonctions sont optimisées pour les appels de queue et assez rapides.

Un énumérable est un objet qui peut être énuméré. «Énuméré» signifie compter les membres d'un ensemble / collection / catégorie un par un (généralement dans l'ordre, généralement par nom).

Elixir fournit le concept d'énumérables et le module Enum pour travailler avec eux. Les fonctions du module Enum sont limitées, comme son nom l'indique, à l'énumération des valeurs dans les structures de données. Un exemple de structure de données énumérables est une liste, un tuple, une carte, etc. Le module Enum nous fournit un peu plus de 100 fonctions pour traiter les énumérations. Nous discuterons de quelques fonctions importantes dans ce chapitre.

Toutes ces fonctions prennent un énumérable comme premier élément et une fonction comme second et fonctionnent dessus. Les fonctions sont décrites ci-dessous.

tout?

Quand nous utilisons all? fonction, la collection entière doit être évaluée à vrai sinon faux sera retourné. Par exemple, pour vérifier si tous les éléments de la liste sont des nombres impairs, alors.

res = Enum.all?([1, 2, 3, 4], fn(s) -> rem(s,2) == 1 end) 
IO.puts(res)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

false

En effet, tous les éléments de cette liste ne sont pas étranges.

tout?

Comme son nom l'indique, cette fonction renvoie true si un élément de la collection est évalué à true. Par exemple -

res = Enum.any?([1, 2, 3, 4], fn(s) -> rem(s,2) == 1 end)
IO.puts(res)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

true

tronçon

Cette fonction divise notre collection en petits morceaux de la taille fournie comme deuxième argument. Par exemple -

res = Enum.chunk([1, 2, 3, 4, 5, 6], 2)
IO.puts(res)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

[[1, 2], [3, 4], [5, 6]]

chaque

Il peut être nécessaire de parcourir une collection sans produire de nouvelle valeur, dans ce cas, nous utilisons la each fonction -

Enum.each(["Hello", "Every", "one"], fn(s) -> IO.puts(s) end)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

Hello
Every
one

carte

Pour appliquer notre fonction à chaque élément et produire une nouvelle collection, nous utilisons la fonction map. C'est l'une des constructions les plus utiles en programmation fonctionnelle car elle est assez expressive et courte. Prenons un exemple pour comprendre cela. Nous doublerons les valeurs stockées dans une liste et la stockerons dans une nouvelle listeres -

res = Enum.map([2, 5, 3, 6], fn(a) -> a*2 end)
IO.puts(res)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

[4, 10, 6, 12]

réduire

le reduceLa fonction nous aide à réduire notre énumérable à une valeur unique. Pour ce faire, nous fournissons un accumulateur optionnel (5 dans cet exemple) à passer dans notre fonction; si aucun accumulateur n'est fourni, la première valeur est utilisée -

res = Enum.reduce([1, 2, 3, 4], 5, fn(x, accum) -> x + accum end)
IO.puts(res)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

15

L'accumulateur est la valeur initiale transmise au fn. À partir du deuxième appel, la valeur renvoyée par l'appel précédent est transmise en tant que cumul. Nous pouvons également utiliser réduire sans l'accumulateur -

res = Enum.reduce([1, 2, 3, 4], fn(x, accum) -> x + accum end)
IO.puts(res)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

10

uniq

La fonction uniq supprime les doublons de notre collection et renvoie uniquement l'ensemble des éléments de la collection. Par exemple -

res = Enum.uniq([1, 2, 2, 3, 3, 3, 4, 4, 4, 4])
IO.puts(res)

Lors de l'exécution du programme ci-dessus, il produit le résultat suivant -

[1, 2, 3, 4]

Évaluation avide

Toutes les fonctions du module Enum sont impatientes. De nombreuses fonctions attendent un énumérable et renvoient une liste. Cela signifie que lors de l'exécution de plusieurs opérations avec Enum, chaque opération va générer une liste intermédiaire jusqu'à ce que nous atteignions le résultat. Prenons l'exemple suivant pour comprendre cela -

odd? = &(odd? = &(rem(&1, 2) != 0) 
res = 1..100_000 |> Enum.map(&(&1 * 3)) |> Enum.filter(odd?) |> Enum.sum 
IO.puts(res)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

7500000000

L'exemple ci-dessus a un pipeline d'opérations. Nous commençons par une plage puis multiplions chaque élément de la plage par 3. Cette première opération va maintenant créer et retourner une liste de 100_000 éléments. Ensuite, nous conservons tous les éléments impairs de la liste, générant une nouvelle liste, maintenant avec 50_000 éléments, puis nous additionnons toutes les entrées.

le |> Le symbole utilisé dans l'extrait ci-dessus est le pipe operator: il prend simplement la sortie de l'expression sur son côté gauche et le passe comme premier argument à l'appel de fonction sur son côté droit. C'est similaire à Unix | opérateur. Son objectif est de mettre en évidence le flux de données transformé par une série de fonctions.

Sans le pipe opérateur, le code semble compliqué -

Enum.sum(Enum.filter(Enum.map(1..100_000, &(&1 * 3)), odd?))

Nous avons de nombreuses autres fonctions, cependant, seules quelques-unes importantes ont été décrites ici.

De nombreuses fonctions attendent un énumérable et renvoient un listretour. Cela signifie que, tout en effectuant plusieurs opérations avec Enum, chaque opération va générer une liste intermédiaire jusqu'à ce que nous atteignions le résultat.

Les flux prennent en charge les opérations paresseuses par opposition aux opérations désirées par les énumérations. En bref,streams are lazy, composable enumerables. Cela signifie que Streams n'effectue aucune opération à moins que cela ne soit absolument nécessaire. Prenons un exemple pour comprendre cela -

odd? = &(rem(&1, 2) != 0)
res = 1..100_000 |> Stream.map(&(&1 * 3)) |> Stream.filter(odd?) |> Enum.sum
IO.puts(res)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

7500000000

Dans l'exemple donné ci-dessus, 1..100_000 |> Stream.map(&(&1 * 3))renvoie un type de données, un flux réel, qui représente le calcul de la carte sur la plage 1..100_000. Il n'a pas encore évalué cette représentation. Au lieu de générer des listes intermédiaires, les flux créent une série de calculs qui ne sont appelés que lorsque nous transmettons le flux sous-jacent au module Enum. Les flux sont utiles lorsque vous travaillez avec des collections volumineuses, voire infinies.

Les flux et les énumérations ont de nombreuses fonctions en commun. Les flux fournissent principalement les mêmes fonctions fournies par le module Enum qui a généré des listes que leurs valeurs de retour après avoir effectué des calculs sur les énumérables d'entrée. Certains d'entre eux sont répertoriés dans le tableau suivant -

Sr.No. Fonction et sa description
1

chunk(enum, n, step, leftover \\ nil)

Diffuse l'énumérable en morceaux, contenant chacun n éléments, où chaque nouveau bloc commence les éléments d'étape dans l'énumérable.

2

concat(enumerables)

Crée un flux qui énumère chaque énumérable dans un énumérable.

3

each(enum, fun)

Exécute la fonction donnée pour chaque élément.

4

filter(enum, fun)

Crée un flux qui filtre les éléments en fonction de la fonction donnée lors de l'énumération.

5

map(enum, fun)

Crée un flux qui appliquera la fonction donnée lors de l'énumération.

6

drop(enum, n)

Supprime paresseusement les n éléments suivants de l'énumérable.

Les structures sont des extensions construites sur des cartes qui fournissent des vérifications à la compilation et des valeurs par défaut.

Définition des structures

Pour définir une structure, la construction defstruct est utilisée -

defmodule User do
   defstruct name: "John", age: 27
end

La liste de mots clés utilisée avec defstruct définit les champs que la structure aura avec leurs valeurs par défaut. Les structures prennent le nom du module dans lequel elles sont définies. Dans l'exemple donné ci-dessus, nous avons défini une structure nommée User. Nous pouvons maintenant créer des structures utilisateur en utilisant une syntaxe similaire à celle utilisée pour créer des cartes -

new_john = %User{})
ayush = %User{name: "Ayush", age: 20}
megan = %User{name: "Megan"})

Le code ci-dessus générera trois structures différentes avec des valeurs -

%User{age: 27, name: "John"}
%User{age: 20, name: "Ayush"}
%User{age: 27, name: "Megan"}

Les structures fournissent des garanties lors de la compilation que seuls les champs (et tous) définis via defstruct seront autorisés à exister dans une structure. Vous ne pouvez donc pas définir vos propres champs une fois que vous avez créé la structure dans le module.

Accès et mise à jour des structures

Lorsque nous avons discuté des cartes, nous avons montré comment nous pouvons accéder et mettre à jour les champs d'une carte. Les mêmes techniques (et la même syntaxe) s'appliquent également aux structures. Par exemple, si nous voulons mettre à jour l'utilisateur que nous avons créé dans l'exemple précédent, alors -

defmodule User do
   defstruct name: "John", age: 27
end
john = %User{}
#john right now is: %User{age: 27, name: "John"}

#To access name and age of John, 
IO.puts(john.name)
IO.puts(john.age)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

John
27

Pour mettre à jour une valeur dans une structure, nous utiliserons à nouveau la même procédure que celle utilisée dans le chapitre map,

meg = %{john | name: "Meg"}

Les structures peuvent également être utilisées dans la correspondance de modèles, à la fois pour la correspondance sur la valeur de clés spécifiques et pour garantir que la valeur correspondante est une structure du même type que la valeur correspondante.

Les protocoles sont un mécanisme pour réaliser le polymorphisme dans Elixir. La distribution sur un protocole est disponible pour tout type de données tant qu'il implémente le protocole.

Prenons un exemple d'utilisation de protocoles. Nous avons utilisé une fonction appeléeto_stringdans les chapitres précédents pour convertir d'autres types en type chaîne. C'est en fait un protocole. Il agit en fonction de l'entrée qui est donnée sans produire d'erreur. Cela peut sembler être en train de discuter des fonctions de correspondance de modèles, mais au fur et à mesure que nous progressons, cela s'avère différent.

Prenons l'exemple suivant pour mieux comprendre le mécanisme du protocole.

Créons un protocole qui affichera si l'entrée donnée est vide ou non. Nous appellerons ce protocoleblank?.

Définition d'un protocole

Nous pouvons définir un protocole dans Elixir de la manière suivante -

defprotocol Blank do
   def blank?(data)
end

Comme vous pouvez le voir, nous n'avons pas besoin de définir un corps pour la fonction. Si vous êtes familier avec les interfaces dans d'autres langages de programmation, vous pouvez considérer un protocole comme essentiellement la même chose.

Donc, ce protocole dit que tout ce qui le met en œuvre doit avoir un empty?fonction, bien que ce soit à l'implémenteur de savoir comment la fonction répond. Avec le protocole défini, comprenons comment ajouter quelques implémentations.

Mettre en œuvre un protocole

Puisque nous avons défini un protocole, nous devons maintenant lui dire comment gérer les différentes entrées qu'il pourrait obtenir. Penchons-nous sur l'exemple que nous avions pris plus tôt. Nous implémenterons le protocole vierge pour les listes, les cartes et les chaînes. Cela montrera si la chose que nous avons passée est vide ou non.

#Defining the protocol
defprotocol Blank do
   def blank?(data)
end

#Implementing the protocol for lists
defimpl Blank, for: List do
   def blank?([]), do: true
   def blank?(_), do: false
end

#Implementing the protocol for strings
defimpl Blank, for: BitString do
   def blank?(""), do: true
   def blank?(_), do: false
end

#Implementing the protocol for maps
defimpl Blank, for: Map do
   def blank?(map), do: map_size(map) == 0
end

IO.puts(Blank.blank? [])
IO.puts(Blank.blank? [:true, "Hello"])
IO.puts(Blank.blank? "")
IO.puts(Blank.blank? "Hi")

Vous pouvez implémenter votre protocole pour autant ou aussi peu de types que vous le souhaitez, tout ce qui est logique pour l'utilisation de votre protocole. C'était un cas d'utilisation assez basique des protocoles. Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

true
false
true
false

Note - Si vous utilisez ceci pour des types autres que ceux pour lesquels vous avez défini le protocole, cela produira une erreur.

File IO fait partie intégrante de tout langage de programmation car il permet au langage d'interagir avec les fichiers du système de fichiers. Dans ce chapitre, nous aborderons deux modules - Chemin et Fichier.

Le module Path

le pathmodule est un très petit module qui peut être considéré comme un module d'aide pour les opérations du système de fichiers. La majorité des fonctions du module File attendent des chemins comme arguments. Le plus souvent, ces chemins seront des binaires réguliers. Le module Path fournit des fonctionnalités pour travailler avec de tels chemins. Il est préférable d'utiliser les fonctions du module Path plutôt que de simplement manipuler les binaires, car le module Path prend en charge différents systèmes d'exploitation de manière transparente. Il est à noter qu'Elixir convertira automatiquement les barres obliques (/) en barres obliques inverses (\) sous Windows lors de l'exécution d'opérations sur les fichiers.

Prenons l'exemple suivant pour mieux comprendre le module Path -

IO.puts(Path.join("foo", "bar"))

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

foo/bar

Le module chemin fournit de nombreuses méthodes. Vous pouvez consulter les différentes méthodes ici . Ces méthodes sont fréquemment utilisées si vous effectuez de nombreuses opérations de manipulation de fichiers.

Le module de fichiers

Le module de fichiers contient des fonctions qui nous permettent d'ouvrir des fichiers en tant que périphériques IO. Par défaut, les fichiers sont ouverts en mode binaire, ce qui oblige les développeurs à utiliser leIO.binread et IO.binwritefonctions du module IO. Créons un fichier appelénewfile et écrivez-y des données.

{:ok, file} = File.read("newfile", [:write]) 
# Pattern matching to store returned stream
IO.binwrite(file, "This will be written to the file")

Si vous ouvrez le fichier dans lequel nous venons d'écrire, le contenu sera affiché de la manière suivante -

This will be written to the file

Voyons maintenant comment utiliser le module de fichiers.

Ouvrir un fichier

Pour ouvrir un fichier, nous pouvons utiliser l'une des 2 fonctions suivantes -

{:ok, file} = File.open("newfile")
file = File.open!("newfile")

Comprenons maintenant la différence entre le File.open fonction et le File.open!() fonction.

  • le File.openLa fonction renvoie toujours un tuple. Si le fichier est ouvert avec succès, il renvoie la première valeur du tuple comme:oket la deuxième valeur est littérale de type io_device. Si une erreur est provoquée, il renverra un tuple avec la première valeur comme:error et deuxième valeur comme raison.

  • le File.open!() la fonction retournera un io_devicesi le fichier est ouvert avec succès, sinon, une erreur sera générée. REMARQUE: c'est le modèle suivi dans toutes les fonctions du module de fichiers que nous allons discuter.

Nous pouvons également spécifier les modes dans lesquels nous voulons ouvrir ce fichier. Pour ouvrir un fichier en lecture seule et en mode d'encodage utf-8, nous utilisons le code suivant -

file = File.open!("newfile", [:read, :utf8])

Ecrire dans un fichier

Nous avons deux façons d'écrire dans des fichiers. Voyons le premier utilisant la fonction d'écriture du module Fichier.

File.write("newfile", "Hello")

Mais cela ne doit pas être utilisé si vous effectuez plusieurs écritures dans le même fichier. Chaque fois que cette fonction est appelée, un descripteur de fichier est ouvert et un nouveau processus est généré pour écrire dans le fichier. Si vous effectuez plusieurs écritures en boucle, ouvrez le fichier viaFile.openet écrivez-y en utilisant les méthodes du module IO. Prenons un exemple pour comprendre la même chose -

#Open the file in read, write and utf8 modes. 
file = File.open!("newfile_2", [:read, :utf8, :write])

#Write to this "io_device" using standard IO functions
IO.puts(file, "Random text")

Vous pouvez utiliser d'autres méthodes de module IO comme IO.write et IO.binwrite pour écrire dans des fichiers ouverts en tant que io_device.

Lecture à partir d'un fichier

Nous avons deux façons de lire des fichiers. Voyons le premier en utilisant la fonction de lecture du module Fichier.

IO.puts(File.read("newfile"))

Lors de l'exécution de ce code, vous devriez obtenir un tuple avec le premier élément comme :ok et le second comme contenu de newfile

Nous pouvons également utiliser le File.read! fonction pour simplement récupérer le contenu des fichiers.

Fermer un fichier ouvert

Chaque fois que vous ouvrez un fichier à l'aide de la fonction File.open, une fois que vous avez fini de l'utiliser, vous devez le fermer en utilisant le File.close fonction -

File.close(file)

Dans Elixir, tout le code s'exécute à l'intérieur des processus. Les processus sont isolés les uns des autres, s'exécutent simultanément et communiquent via la transmission de messages. Les processus d'Elixir ne doivent pas être confondus avec les processus du système d'exploitation. Les processus d'Elixir sont extrêmement légers en termes de mémoire et de CPU (contrairement aux threads dans de nombreux autres langages de programmation). Pour cette raison, il n'est pas rare que des dizaines, voire des centaines de milliers de processus s'exécutent simultanément.

Dans ce chapitre, nous allons découvrir les constructions de base pour générer de nouveaux processus, ainsi que pour envoyer et recevoir des messages entre différents processus.

La fonction Spawn

Le moyen le plus simple de créer un nouveau processus est d'utiliser le spawnfonction. lespawnaccepte une fonction qui sera exécutée dans le nouveau processus. Par exemple -

pid = spawn(fn -> 2 * 2 end)
Process.alive?(pid)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

false

La valeur de retour de la fonction spawn est un PID. Il s'agit d'un identifiant unique pour le processus et donc si vous exécutez le code au-dessus de votre PID, ce sera différent. Comme vous pouvez le voir dans cet exemple, le processus est mort lorsque nous vérifions s'il est vivant. En effet, le processus se terminera dès qu'il aura terminé d'exécuter la fonction donnée.

Comme déjà mentionné, tous les codes Elixir s'exécutent à l'intérieur des processus. Si vous exécutez la fonction auto, vous verrez le PID de votre session en cours -

pid = self
 
Process.alive?(pid)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

true

Message passant

Nous pouvons envoyer des messages à un processus avec send et recevez-les avec receive. Passons un message au processus actuel et recevons-le sur le même.

send(self(), {:hello, "Hi people"})

receive do
   {:hello, msg} -> IO.puts(msg)
   {:another_case, msg} -> IO.puts("This one won't match!")
end

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

Hi people

Nous avons envoyé un message au processus actuel en utilisant la fonction d'envoi et l'avons passé au PID de self. Ensuite, nous avons traité le message entrant en utilisant lereceive fonction.

Lorsqu'un message est envoyé à un processus, le message est stocké dans le process mailbox. Le bloc de réception parcourt la boîte aux lettres de processus en cours à la recherche d'un message correspondant à l'un des modèles donnés. Le bloc de réception prend en charge les gardes et de nombreuses clauses, telles que case.

S'il n'y a aucun message dans la boîte aux lettres correspondant à l'un des modèles, le processus en cours attendra jusqu'à ce qu'un message correspondant arrive. Un délai d'expiration peut également être spécifié. Par exemple,

receive do
   {:hello, msg}  -> msg
after
   1_000 -> "nothing after 1s"
end

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

nothing after 1s

NOTE - Un délai d'expiration de 0 peut être donné lorsque vous vous attendez déjà à ce que le message soit dans la boîte aux lettres.

Liens

La forme la plus courante de frai dans Elixir est en fait via spawn_linkfonction. Avant de jeter un œil à un exemple avec spawn_link, voyons ce qui se passe lorsqu'un processus échoue.

spawn fn -> raise "oops" end

Lorsque le programme ci-dessus est exécuté, il produit l'erreur suivante -

[error] Process #PID<0.58.00> raised an exception
** (RuntimeError) oops
   :erlang.apply/2

Il a enregistré une erreur mais le processus de frai est toujours en cours. C'est parce que les processus sont isolés. Si nous voulons que l'échec d'un processus se propage à un autre, nous devons les lier. Cela peut être fait avec lespawn_linkfonction. Prenons un exemple pour comprendre la même chose -

spawn_link fn -> raise "oops" end

Lorsque le programme ci-dessus est exécuté, il produit l'erreur suivante -

** (EXIT from #PID<0.41.0>) an exception was raised:
   ** (RuntimeError) oops
      :erlang.apply/2

Si vous exécutez ceci dans iexshell alors le shell gère cette erreur et ne se ferme pas. Mais si vous exécutez en créant d'abord un fichier de script, puis en utilisantelixir <file-name>.exs, le processus parent sera également interrompu en raison de cet échec.

Les processus et les liens jouent un rôle important lors de la création de systèmes tolérants aux pannes. Dans les applications Elixir, nous lions souvent nos processus à des superviseurs qui détectent quand un processus meurt et démarrent un nouveau processus à sa place. Cela n'est possible que parce que les processus sont isolés et ne partagent rien par défaut. Et comme les processus sont isolés, il n'y a aucun moyen qu'un échec dans un processus se bloque ou corrompe l'état d'un autre. Alors que d'autres langages nous obligeront à attraper / gérer les exceptions; dans Elixir, nous sommes en fait d'accord pour laisser les processus échouer parce que nous attendons des superviseurs qu'ils redémarrent correctement nos systèmes.

Etat

Si vous créez une application qui nécessite un état, par exemple, pour conserver la configuration de votre application, ou si vous devez analyser un fichier et le conserver en mémoire, où le stockeriez-vous? La fonctionnalité de processus d'Elixir peut être utile lorsque vous faites de telles choses.

Nous pouvons écrire des processus qui bouclent indéfiniment, maintiennent l'état et envoient et reçoivent des messages. À titre d'exemple, écrivons un module qui démarre de nouveaux processus qui fonctionnent comme un magasin clé-valeur dans un fichier nommékv.exs.

defmodule KV do
   def start_link do
      Task.start_link(fn -> loop(%{}) end)
   end

   defp loop(map) do
      receive do
         {:get, key, caller} ->
         send caller, Map.get(map, key)
         loop(map)
         {:put, key, value} ->
         loop(Map.put(map, key, value))
      end
   end
end

Notez que le start_link la fonction démarre un nouveau processus qui exécute le loopfonction, en commençant par une carte vide. leloopLa fonction attend ensuite les messages et exécute l'action appropriée pour chaque message. Dans le cas d'un:getmessage, il renvoie un message à l'appelant et appelle à nouveau en boucle, pour attendre un nouveau message. Tandis que le:put le message invoque en fait loop avec une nouvelle version de la carte, avec la clé et la valeur données stockées.

Examinons maintenant ce qui suit -

iex kv.exs

Maintenant tu devrais être dans ton iexcoquille. Pour tester notre module, essayez ce qui suit -

{:ok, pid} = KV.start_link

# pid now has the pid of our new process that is being 
# used to get and store key value pairs 

# Send a KV pair :hello, "Hello" to the process
send pid, {:put, :hello, "Hello"}

# Ask for the key :hello
send pid, {:get, :hello, self()}

# Print all the received messages on the current process.
flush()

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

"Hello"

Dans ce chapitre, nous allons explorer les sigils, les mécanismes fournis par le langage pour travailler avec des représentations textuelles. Les sigils commencent par le caractère tilde (~) qui est suivi d'une lettre (qui identifie le sigil) puis d'un délimiteur; facultativement, des modificateurs peuvent être ajoutés après le délimiteur final.

Regex

Les regex dans Elixir sont des sceaux. Nous avons vu leur utilisation dans le chapitre String. Prenons à nouveau un exemple pour voir comment nous pouvons utiliser les regex dans Elixir.

# A regular expression that matches strings which contain "foo" or
# "bar":
regex = ~r/foo|bar/
IO.puts("foo" =~ regex)
IO.puts("baz" =~ regex)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

true
false

Les sigils prennent en charge 8 délimiteurs différents -

~r/hello/
~r|hello|
~r"hello"
~r'hello'
~r(hello)
~r[hello]
~r{hello}
~r<hello>

La raison derrière la prise en charge de différents délimiteurs est que différents délimiteurs peuvent être plus adaptés à différents sigils. Par exemple, l'utilisation de parenthèses pour les expressions régulières peut être un choix déroutant car elles peuvent être mélangées avec les parenthèses à l'intérieur de l'expression régulière. Cependant, les parenthèses peuvent être utiles pour d'autres sigils, comme nous le verrons dans la section suivante.

Elixir prend en charge les expressions régulières compatibles Perl et prend également en charge les modificateurs. Vous pouvez en savoir plus sur l'utilisation des expressions régulières ici .

Chaînes, listes de caractères et listes de mots

En plus des expressions régulières, Elixir a 3 autres sceaux intégrés. Jetons un coup d'œil aux sigils.

Cordes

Le sigil ~ s est utilisé pour générer des chaînes, comme le sont les guillemets doubles. Le sigil ~ s est utile, par exemple, lorsqu'une chaîne contient à la fois des guillemets doubles et simples -

new_string = ~s(this is a string with "double" quotes, not 'single' ones)
IO.puts(new_string)

Ce sceau génère des chaînes. Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

"this is a string with \"double\" quotes, not 'single' ones"

Listes de caractères

Le sigil ~ c est utilisé pour générer des listes de caractères -

new_char_list = ~c(this is a char list containing 'single quotes')
IO.puts(new_char_list)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

this is a char list containing 'single quotes'

Listes de mots

Le sigil ~ w est utilisé pour générer des listes de mots (les mots ne sont que des chaînes régulières). À l'intérieur du sigil ~ w, les mots sont séparés par des espaces.

new_word_list = ~w(foo bar bat)
IO.puts(new_word_list)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

foobarbat

Le sigil ~ w accepte également le c, s et a les modificateurs (pour les listes de caractères, les chaînes et les atomes, respectivement), qui spécifient le type de données des éléments de la liste résultante -

new_atom_list = ~w(foo bar bat)a
IO.puts(new_atom_list)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

[:foo, :bar, :bat]

Interpolation et échappée dans les sigils

Outre les sigils minuscules, Elixir prend en charge les sigils majuscules pour traiter les caractères d'échappement et l'interpolation. Alors que ~ s et ~ S renverront des chaînes, le premier autorise les codes d'échappement et l'interpolation alors que le second ne le fait pas. Prenons un exemple pour comprendre cela -

~s(String with escape codes \x26 #{"inter" <> "polation"})
# "String with escape codes & interpolation"
~S(String without escape codes \x26 without #{interpolation})
# "String without escape codes \\x26 without \#{interpolation}"

Sigils personnalisés

Nous pouvons facilement créer nos propres sigils personnalisés. Dans cet exemple, nous allons créer un sigil pour convertir une chaîne en majuscules.

defmodule CustomSigil do
   def sigil_u(string, []), do: String.upcase(string)
end

import CustomSigil

IO.puts(~u/tutorials point/)

Lorsque nous exécutons le code ci-dessus, il produit le résultat suivant -

TUTORIALS POINT

Nous définissons d'abord un module appelé CustomSigil et dans ce module, nous avons créé une fonction appelée sigil_u. Comme il n'y a pas de sigil ~ u existant dans l'espace sigil existant, nous allons l'utiliser. Le _u indique que nous souhaitons utiliser u comme caractère après le tilde. La définition de la fonction doit prendre deux arguments, une entrée et une liste.

Les compréhensions de listes sont du sucre syntaxique pour parcourir les énumérables dans Elixir. Dans ce chapitre, nous utiliserons les compréhensions pour l'itération et la génération.

Basiques

Quand nous avons regardé le module Enum dans le chapitre énumérables, nous sommes tombés sur la fonction map.

Enum.map(1..3, &(&1 * 2))

Dans cet exemple, nous passerons une fonction comme deuxième argument. Chaque élément de la plage sera passé dans la fonction, puis une nouvelle liste sera renvoyée contenant les nouvelles valeurs.

La cartographie, le filtrage et la transformation sont des actions très courantes dans Elixir et il existe donc une manière légèrement différente d'obtenir le même résultat que l'exemple précédent -

for n <- 1..3, do: n * 2

Lorsque nous exécutons le code ci-dessus, il produit le résultat suivant -

[2, 4, 6]

Le deuxième exemple est une compréhension, et comme vous pouvez probablement le voir, il s'agit simplement de sucre syntaxique pour ce que vous pouvez également réaliser si vous utilisez le Enum.mapfonction. Cependant, il n'y a pas de réels avantages à utiliser une compréhension sur une fonction du module Enum en termes de performances.

Les compréhensions ne se limitent pas aux listes mais peuvent être utilisées avec tous les énumérables.

Filtre

Vous pouvez considérer les filtres comme une sorte de garde pour les compréhensions. Lorsqu'une valeur filtrée revientfalse ou nilil est exclu de la liste finale. Passons en boucle sur une plage et ne nous préoccupons que des nombres pairs. Nous utiliserons leis_even fonction du module Integer pour vérifier si une valeur est paire ou non.

import Integer
IO.puts(for x <- 1..10, is_even(x), do: x)

Lorsque le code ci-dessus est exécuté, il produit le résultat suivant -

[2, 4, 6, 8, 10]

Nous pouvons également utiliser plusieurs filtres dans la même compréhension. Ajoutez un autre filtre de votre choix après leis_even filtre séparé par une virgule.

: en option

Dans les exemples ci-dessus, toutes les compréhensions ont renvoyé des listes comme résultat. Cependant, le résultat d'une compréhension peut être inséré dans différentes structures de données en passant le:into option à la compréhension.

Par exemple, un bitstring générateur peut être utilisé avec l'option: into afin de supprimer facilement tous les espaces d'une chaîne -

IO.puts(for <<c <- " hello world ">>, c != ?\s, into: "", do: <<c>>)

Lorsque le code ci-dessus est exécuté, il produit le résultat suivant -

helloworld

Le code ci-dessus supprime tous les espaces de la chaîne en utilisant c != ?\s filter puis en utilisant l'option: into, il met tous les caractères retournés dans une chaîne.

Elixir est un langage typé dynamiquement, donc tous les types d'Elixir sont déduits par le runtime. Néanmoins, Elixir est livré avec des typespecs, qui sont une notation utilisée pourdeclaring custom data types and declaring typed function signatures (specifications).

Spécifications des fonctions (spécifications)

Par défaut, Elixir fournit des types de base, tels que des entiers ou des pid, ainsi que des types complexes: par exemple, le round, qui arrondit un flottant à son entier le plus proche, prend un nombre comme argument (un entier ou un flottant) et renvoie un entier. Dans la documentation associée , la signature dactylographiée ronde s'écrit -

round(number) :: integer

La description ci-dessus implique que la fonction à gauche prend comme argument ce qui est spécifié entre parenthèses et retourne ce qui est à droite de ::, c'est-à-dire Integer. Les spécifications de fonction sont écrites avec le@specdirective, placée juste avant la définition de la fonction. La fonction round peut s'écrire -

@spec round(number) :: integer
def round(number), do: # Function implementation
...

Les spécifications de type prennent également en charge les types complexes, par exemple, si vous souhaitez renvoyer une liste d'entiers, vous pouvez utiliser [Integer]

Types personnalisés

Bien qu'Elixir fournisse de nombreux types intégrés utiles, il est pratique de définir des types personnalisés le cas échéant. Cela peut être fait lors de la définition de modules via la directive @type. Prenons un exemple pour comprendre la même chose -

defmodule FunnyCalculator do
   @type number_with_joke :: {number, String.t}

   @spec add(number, number) :: number_with_joke
   def add(x, y), do: {x + y, "You need a calculator to do that?"}

   @spec multiply(number, number) :: number_with_joke
   def multiply(x, y), do: {x * y, "It is like addition on steroids."}
end

{result, comment} = FunnyCalculator.add(10, 20)
IO.puts(result)
IO.puts(comment)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

30
You need a calculator to do that?

NOTE - Les types personnalisés définis via @type sont exportés et disponibles en dehors du module dans lequel ils sont définis. Si vous souhaitez conserver un type personnalisé privé, vous pouvez utiliser le @typep directive au lieu de @type.

Les comportements dans Elixir (et Erlang) sont un moyen de séparer et d'abstraire la partie générique d'un composant (qui devient le module de comportement) de la partie spécifique (qui devient le module de rappel). Les comportements permettent de -

  • Définissez un ensemble de fonctions qui doivent être implémentées par un module.
  • Assurez-vous qu'un module implémente toutes les fonctions de cet ensemble.

Si vous devez le faire, vous pouvez penser à des comportements tels que des interfaces dans des langages orientés objet comme Java: un ensemble de signatures de fonction qu'un module doit implémenter.

Définir un comportement

Prenons un exemple pour créer notre propre comportement, puis utilisons ce comportement générique pour créer un module. Nous définirons un comportement qui salue les gens bonjour et au revoir dans différentes langues.

defmodule GreetBehaviour do
   @callback say_hello(name :: string) :: nil
   @callback say_bye(name :: string) :: nil
end

le @callbackdirective est utilisée pour lister les fonctions que l'adoption de modules devra définir. Il spécifie également le non. d'arguments, leur type et leurs valeurs de retour.

Adopter un comportement

Nous avons défini avec succès un comportement. Nous allons maintenant l'adopter et l'implémenter dans plusieurs modules. Créons deux modules implémentant ce comportement en anglais et en espagnol.

defmodule GreetBehaviour do
   @callback say_hello(name :: string) :: nil
   @callback say_bye(name :: string) :: nil
end

defmodule EnglishGreet do
   @behaviour GreetBehaviour
   def say_hello(name), do: IO.puts("Hello " <> name)
   def say_bye(name), do: IO.puts("Goodbye, " <> name)
end

defmodule SpanishGreet do
   @behaviour GreetBehaviour
   def say_hello(name), do: IO.puts("Hola " <> name)
   def say_bye(name), do: IO.puts("Adios " <> name)
end

EnglishGreet.say_hello("Ayush")
EnglishGreet.say_bye("Ayush")
SpanishGreet.say_hello("Ayush")
SpanishGreet.say_bye("Ayush")

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

Hello Ayush
Goodbye, Ayush
Hola Ayush
Adios Ayush

Comme vous l'avez déjà vu, nous adoptons un comportement utilisant le @behaviourdirective dans le module. Nous devons définir toutes les fonctions implémentées dans le comportement pour tous les modules enfants . Cela peut à peu près être considéré comme équivalent aux interfaces dans les langages POO.

Elixir a trois mécanismes d'erreur: les erreurs, les lancers et les sorties. Explorons chaque mécanisme en détail.

Erreur

Des erreurs (ou exceptions) sont utilisées lorsque des choses exceptionnelles se produisent dans le code. Un exemple d'erreur peut être récupéré en essayant d'ajouter un nombre dans une chaîne -

IO.puts(1 + "Hello")

Lorsque le programme ci-dessus est exécuté, il produit l'erreur suivante -

** (ArithmeticError) bad argument in arithmetic expression
   :erlang.+(1, "Hello")

C'était un exemple d'erreur intégrée.

Relever des erreurs

nous pouvons raiseerreurs en utilisant les fonctions de montée. Prenons un exemple pour comprendre la même chose -

#Runtime Error with just a message
raise "oops"  # ** (RuntimeError) oops

D'autres erreurs peuvent être déclenchées en passant le nom de l'erreur et une liste d'arguments de mot-clé

#Other error type with a message
raise ArgumentError, message: "invalid argument foo"

Vous pouvez également définir vos propres erreurs et les relever. Prenons l'exemple suivant -

defmodule MyError do
   defexception message: "default message"
end

raise MyError  # Raises error with default message
raise MyError, message: "custom message"  # Raises error with custom message

Récupérer les erreurs

Nous ne voulons pas que nos programmes s'arrêtent brusquement, mais plutôt les erreurs doivent être traitées avec soin. Pour cela, nous utilisons la gestion des erreurs. nousrescue erreurs en utilisant le try/rescueconstruction. Prenons l'exemple suivant pour comprendre la même chose -

err = try do
   raise "oops"
rescue
   e in RuntimeError -> e
end

IO.puts(err.message)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

oops

Nous avons géré les erreurs dans l'instruction de sauvetage en utilisant la correspondance de modèle. Si nous n'avons aucune utilisation de l'erreur et que nous voulons simplement l'utiliser à des fins d'identification, nous pouvons également utiliser le formulaire -

err = try do
   1 + "Hello"
rescue
   RuntimeError -> "You've got a runtime error!"
   ArithmeticError -> "You've got a Argument error!"
end

IO.puts(err)

Lors de l'exécution du programme ci-dessus, il produit le résultat suivant -

You've got a Argument error!

NOTE- La plupart des fonctions de la bibliothèque standard Elixir sont implémentées deux fois, une fois renvoyant des tuples et l'autre provoquant des erreurs. Par exemple, leFile.read et le File.read!les fonctions. Le premier retournait un tuple si le fichier était lu avec succès et si une erreur était rencontrée, ce tuple était utilisé pour donner la raison de l'erreur. Le second a soulevé une erreur si une erreur était rencontrée.

Si nous utilisons la première approche fonctionnelle, nous devons utiliser le cas pour le modèle correspondant à l'erreur et agir en conséquence. Dans le second cas, nous utilisons l'approche try rescue pour le code sujet aux erreurs et traitons les erreurs en conséquence.

Jette

Dans Elixir, une valeur peut être lancée et attrapée plus tard. Throw et Catch sont réservés aux situations où il n'est pas possible de récupérer une valeur à moins d'utiliser throw et catch.

Les instances sont assez rares en pratique sauf lors de l'interfaçage avec des bibliothèques. Par exemple, supposons maintenant que le module Enum ne fournissait aucune API pour trouver une valeur et que nous devions trouver le premier multiple de 13 dans une liste de nombres -

val = try do
   Enum.each 20..100, fn(x) ->
      if rem(x, 13) == 0, do: throw(x)
   end
   "Got nothing"
catch
   x -> "Got #{x}"
end

IO.puts(val)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

Got 26

Sortie

Lorsqu'un processus meurt de «causes naturelles» (par exemple, des exceptions non gérées), il envoie un signal de sortie. Un processus peut également mourir en envoyant explicitement un signal de sortie. Prenons l'exemple suivant -

spawn_link fn -> exit(1) end

Dans l'exemple ci-dessus, le processus lié est mort en envoyant un signal de sortie avec la valeur 1. Notez que la sortie peut également être «interceptée» en utilisant try / catch. Par exemple -

val = try do
   exit "I am exiting"
catch
   :exit, _ -> "not really"
end

IO.puts(val)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

not really

Après

Parfois, il est nécessaire de s'assurer qu'une ressource est nettoyée après une action qui peut potentiellement provoquer une erreur. La construction try / after vous permet de faire cela. Par exemple, nous pouvons ouvrir un fichier et utiliser une clause after pour le fermer, même si quelque chose ne va pas.

{:ok, file} = File.open "sample", [:utf8, :write]
try do
   IO.write file, "olá"
   raise "oops, something went wrong"
after
   File.close(file)
end

Lorsque nous exécutons ce programme, cela nous donnera une erreur. Mais leafter garantit que le descripteur de fichier est fermé lors d'un tel événement.

Les macros sont l'une des fonctionnalités les plus avancées et les plus puissantes d'Elixir. Comme pour toutes les fonctionnalités avancées de n'importe quel langage, les macros doivent être utilisées avec parcimonie. Ils permettent d'effectuer de puissantes transformations de code au moment de la compilation. Nous allons maintenant comprendre ce que sont les macros et comment les utiliser en bref.

Citation

Avant de commencer à parler de macros, examinons d'abord les composants internes d'Elixir. Un programme Elixir peut être représenté par ses propres structures de données. Le bloc de construction d'un programme Elixir est un tuple avec trois éléments. Par exemple, l'appel de fonction sum (1, 2, 3) est représenté en interne par -

{:sum, [], [1, 2, 3]}

Le premier élément est le nom de la fonction, le second est une liste de mots clés contenant des métadonnées et le troisième est la liste d'arguments. Vous pouvez obtenir ceci comme sortie dans le shell iex si vous écrivez ce qui suit -

quote do: sum(1, 2, 3)

Les opérateurs sont également représentés comme de tels tuples. Les variables sont également représentées à l'aide de ces triplets, sauf que le dernier élément est un atome, au lieu d'une liste. En citant des expressions plus complexes, nous pouvons voir que le code est représenté dans de tels tuples, qui sont souvent imbriqués les uns dans les autres dans une structure ressemblant à un arbre. De nombreuses langues appelleraient de telles représentations unAbstract Syntax Tree (AST). Elixir appelle ces expressions citées.

Fin de citation

Maintenant que nous pouvons récupérer la structure interne de notre code, comment la modifier? Pour injecter un nouveau code ou de nouvelles valeurs, nous utilisonsunquote. Lorsque nous décompressons une expression, elle sera évaluée et injectée dans l'AST. Prenons un exemple (dans iex shell) pour comprendre le concept -

num = 25

quote do: sum(15, num)

quote do: sum(15, unquote(num))

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

{:sum, [], [15, {:num, [], Elixir}]}
{:sum, [], [15, 25]}

Dans l'exemple de l'expression quote, elle ne remplace pas automatiquement num par 25. Nous devons décompresser cette variable si nous voulons modifier l'AST.

Macros

Alors maintenant que nous sommes familiers avec les guillemets et les guillemets, nous pouvons explorer la métaprogrammation dans Elixir à l'aide de macros.

Dans le plus simple des termes, les macros sont des fonctions spéciales conçues pour renvoyer une expression entre guillemets qui sera insérée dans notre code d'application. Imaginez que la macro soit remplacée par l'expression entre guillemets plutôt que appelée comme une fonction. Avec les macros, nous avons tout le nécessaire pour étendre Elixir et ajouter dynamiquement du code à nos applications

Laissez-nous implémenter à moins que comme une macro. Nous commencerons par définir la macro à l'aide dudefmacromacro. N'oubliez pas que notre macro doit renvoyer une expression entre guillemets.

defmodule OurMacro do
   defmacro unless(expr, do: block) do
      quote do
         if !unquote(expr), do: unquote(block)
      end
   end
end

require OurMacro

OurMacro.unless true, do: IO.puts "True Expression"

OurMacro.unless false, do: IO.puts "False expression"

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

False expression

Ce qui se passe ici, c'est que notre code est remplacé par le code cité retourné par la macro sauf . Nous avons décompressé l'expression pour l'évaluer dans le contexte actuel et également décompressé le bloc do pour l'exécuter dans son contexte. Cet exemple nous montre la métaprogrammation à l'aide de macros dans elixir.

Les macros peuvent être utilisées dans des tâches beaucoup plus complexes, mais doivent être utilisées avec parcimonie. En effet, la métaprogrammation en général est considérée comme une mauvaise pratique et ne doit être utilisée que lorsque cela est nécessaire.

Elixir offre une excellente interopérabilité avec les bibliothèques Erlang. Parlons de quelques bibliothèques en bref.

Le module binaire

Le module Elixir String intégré gère les binaires encodés en UTF-8. Le module binaire est utile lorsque vous traitez des données binaires qui ne sont pas nécessairement encodées en UTF-8. Prenons un exemple pour mieux comprendre le module binaire -

# UTF-8
IO.puts(String.to_char_list("Ø"))

# binary
IO.puts(:binary.bin_to_list "Ø")

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

[216]
[195, 152]

L'exemple ci-dessus montre la différence; le module String renvoie des points de code UTF-8, tandis que: binary traite les octets de données brutes.

Le module crypto

Le module crypto contient des fonctions de hachage, des signatures numériques, du cryptage et plus encore. Ce module ne fait pas partie de la bibliothèque standard Erlang, mais est inclus avec la distribution Erlang. Cela signifie que vous devez lister: crypto dans la liste des applications de votre projet chaque fois que vous l'utilisez. Voyons un exemple utilisant le module crypto -

IO.puts(Base.encode16(:crypto.hash(:sha256, "Elixir")))

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

3315715A7A3AD57428298676C5AE465DADA38D951BDFAC9348A8A31E9C7401CB

Le module Digraph

Le module digraph contient des fonctions permettant de traiter des graphes dirigés constitués de sommets et d'arêtes. Après avoir construit le graphe, les algorithmes qui s'y trouvent aideront à trouver, par exemple, le chemin le plus court entre deux sommets ou des boucles dans le graphe. Notez que les fonctionsin :digraph modifier indirectement la structure du graphe comme effet secondaire, tout en renvoyant les sommets ou arêtes ajoutés.

digraph = :digraph.new()
coords = [{0.0, 0.0}, {1.0, 0.0}, {1.0, 1.0}]
[v0, v1, v2] = (for c <- coords, do: :digraph.add_vertex(digraph, c))
:digraph.add_edge(digraph, v0, v1)
:digraph.add_edge(digraph, v1, v2)
for point <- :digraph.get_short_path(digraph, v0, v2) do 
   {x, y} = point
   IO.puts("#{x}, #{y}")
end

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

0.0, 0.0
1.0, 0.0
1.0, 1.0

Le module mathématique

Le module mathématique contient des opérations mathématiques courantes couvrant les fonctions trigonométriques, exponentielles et logarithmiques. Prenons l'exemple suivant pour comprendre le fonctionnement du module Math -

# Value of pi
IO.puts(:math.pi())

# Logarithm
IO.puts(:math.log(7.694785265142018e23))

# Exponentiation
IO.puts(:math.exp(55.0))

#...

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

3.141592653589793
55.0
7.694785265142018e23

Le module de file d'attente

La file d'attente est une structure de données qui implémente efficacement les files d'attente FIFO (premier entré premier sorti). L'exemple suivant montre comment fonctionne un module de file d'attente -

q = :queue.new
q = :queue.in("A", q)
q = :queue.in("B", q)
{{:value, val}, q} = :queue.out(q)
IO.puts(val)
{{:value, val}, q} = :queue.out(q)
IO.puts(val)

Lorsque le programme ci-dessus est exécuté, il produit le résultat suivant -

A
B