F # - Guide rapide

F # est un langage de programmation fonctionnel. Pour comprendre les constructions F #, vous devez lire quelques lignes sur le paradigme de programmation nomméFunctional Programming.

La programmation fonctionnelle traite les programmes informatiques comme des fonctions mathématiques. En programmation fonctionnelle, l'accent serait mis sur les constantes et les fonctions, plutôt que sur les variables et les états. Parce que les fonctions et les constantes sont des choses qui ne changent pas.

En programmation fonctionnelle, vous écrirez des programmes modulaires, c'est-à-dire que les programmes comprendraient des fonctions qui prendront d'autres fonctions en entrée.

Les programmes écrits en langage de programmation fonctionnel ont tendance à être concis.

À propos de F #

Voici les informations de base sur F # -

  • Il a été développé en 2005 chez Microsoft Research.
  • Il fait partie de la famille Microsoft du langage .Net.
  • C'est un langage de programmation fonctionnel.
  • Il est basé sur le langage de programmation fonctionnel OCaml.

Caractéristiques de F #

  • C'est une implémentation .Net d'OCaml.

  • Il compile le code octet .Net CLI (Common Language Interface) ou MSIL (Microsoft Intermediate Language) qui s'exécute sur CLR (Common Language Runtime).

  • Il fournit une inférence de type.

  • Il fournit des constructions de correspondance de modèles riches.

  • Il a des capacités interactives de script et de débogage.

  • Il permet d'écrire des fonctions d'ordre supérieur.

  • Il fournit un modèle d'objet bien développé.

Utilisation de F #

F # est normalement utilisé dans les domaines suivants -

  • Faire un modèle scientifique
  • Résolution de problèmes mathématiques
  • Travaux de recherche en intelligence artificielle
  • Modélisation financière
  • Conception graphique
  • Conception du processeur
  • Programmation du compilateur
  • Telecommunications

Il est également utilisé dans les applications CRUD, les pages Web, les jeux d'interface graphique et d'autres programmes à usage général.

Les outils nécessaires à la programmation F # sont traités dans ce chapitre.

Environnement de développement intégré (IDE) pour F #

Microsoft fournit Visual Studio 2013 pour la programmation F #.

L'édition communautaire gratuite de Visual Studio 2013 est disponible sur le site Web officiel de Microsoft. La communauté Visual Studio 2013 et les versions ultérieures sont fournies avec les outils Visual F #. Détails de l'installation disponibles sur Asp.net Tutorial . Les outils Visual F # incluent le compilateur de ligne de commande (fsc.exe) et F # Interactive (fsi.exe).

En utilisant ces outils, vous pouvez écrire toutes sortes de programmes F #, des simples applications de ligne de commande aux applications plus complexes. Vous pouvez également écrire des fichiers de code source F # à l'aide d'un éditeur de texte de base, tel que le Bloc-notes, et compiler le code dans des assemblys à l'aide du compilateur de ligne de commande.

Vous pouvez le télécharger à partir de Microsoft Visual Studio. Il est automatiquement installé sur votre machine.

Ecrire des programmes F # sur des liens

Veuillez visiter le site officiel de F # pour les dernières instructions pour obtenir les outils sous forme de paquet Debian ou les compiler directement à partir de la source - https://fsharp.org/use/linux/.

F # est un langage de programmation fonctionnelle.

En F #, les fonctions fonctionnent comme des types de données. Vous pouvez déclarer et utiliser une fonction de la même manière que toute autre variable.

En général, une application F # n'a pas de point d'entrée spécifique. Le compilateur exécute toutes les instructions de niveau supérieur dans le fichier de haut en bas.

Cependant, pour suivre le style de programmation procédurale, de nombreuses applications conservent une seule instruction de niveau supérieur qui appelle la boucle principale.

Le code suivant montre un simple programme F # -

open System
(* This is a multi-line comment *)
// This is a single-line comment

let sign num =
   if num > 0 then "positive"
   elif num < 0 then "negative"
   else "zero"

let main() =
   Console.WriteLine("sign 5: {0}", (sign 5))

main()

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

sign 5: positive

Veuillez noter que -

  • Un fichier de code F # peut commencer par un certain nombre de open instructions utilisées pour importer des espaces de noms.

  • Le corps des fichiers comprend d'autres fonctions qui implémentent la logique métier de l'application.

  • La boucle principale contient les principales instructions exécutables.

Vous avez vu la structure de base d'un programme F #, il sera donc facile de comprendre les autres blocs de construction de base du langage de programmation F #.

Jetons en F #

Un programme F # se compose de plusieurs jetons. Un jeton peut être un mot-clé, un identifiant, une constante, une chaîne littérale ou un symbole. Nous pouvons classer les jetons F # en deux types -

  • Keywords
  • Symbole et opérateurs

Mots-clés F #

Le tableau suivant présente les mots-clés et de brèves descriptions des mots-clés. Nous discuterons de l'utilisation de ces mots-clés dans les chapitres suivants.

Mot-clé La description
abstract Indique une méthode qui n'a pas d'implémentation dans le type dans lequel elle est déclarée ou qui est virtuelle et a une implémentation par défaut.
and Utilisé dans les liaisons mutuellement récursives, dans les déclarations de propriété et avec plusieurs contraintes sur les paramètres génériques.
as Utilisé pour donner à l'objet de classe courant un nom d'objet. Également utilisé pour donner un nom à un motif entier dans une correspondance de motif.
assert Utilisé pour vérifier le code pendant le débogage.
base Utilisé comme nom de l'objet de classe de base.
begin Dans une syntaxe détaillée, indique le début d'un bloc de code.
class Dans une syntaxe détaillée, indique le début d'une définition de classe.
default Indique une implémentation d'une méthode abstraite; utilisé avec une déclaration de méthode abstraite pour créer une méthode virtuelle.
delegate Utilisé pour déclarer un délégué.
do Utilisé dans les constructions en boucle ou pour exécuter du code impératif.
done Dans une syntaxe détaillée, indique la fin d'un bloc de code dans une expression en boucle.
downcast Utilisé pour convertir en un type inférieur dans la chaîne d'héritage.
downto Dans un for expression, utilisée lors du comptage en sens inverse.
elif Utilisé dans le branchement conditionnel. Une forme courte de else if.
else Utilisé dans le branchement conditionnel.
end

Dans les définitions de type et les extensions de type, indique la fin d'une section de définitions de membre.

Dans une syntaxe détaillée, utilisé pour spécifier la fin d'un bloc de code commençant par le mot-clé begin.

exception Utilisé pour déclarer un type d'exception.
extern Indique qu'un élément de programme déclaré est défini dans un autre binaire ou assembly.
false Utilisé comme littéral booléen.
finally Utilisé avec essayer d'introduire un bloc de code qui s'exécute indépendamment du fait qu'une exception se produise.
for Utilisé dans les constructions en boucle.
fun Utilisé dans les expressions lambda, également appelées fonctions anonymes.
function Utilisé comme une alternative plus courte au mot-clé fun et une expression de correspondance dans une expression lambda qui a une correspondance de modèle sur un seul argument.
global Utilisé pour référencer l'espace de noms .NET de niveau supérieur.
if Utilisé dans les constructions de branchement conditionnel.
in Utilisé pour les expressions de séquence et, dans la syntaxe détaillée, pour séparer les expressions des liaisons.
inherit Utilisé pour spécifier une classe de base ou une interface de base.
inline Utilisé pour indiquer une fonction qui doit être intégrée directement dans le code de l'appelant.
interface Utilisé pour déclarer et implémenter des interfaces.
internal Utilisé pour spécifier qu'un membre est visible à l'intérieur d'un assemblage mais pas à l'extérieur.
lazy Utilisé pour spécifier un calcul à effectuer uniquement lorsqu'un résultat est nécessaire.
let Utilisé pour associer ou lier un nom à une valeur ou une fonction.
let! Utilisé dans les flux de travail asynchrones pour lier un nom au résultat d'un calcul asynchrone ou, dans d'autres expressions de calcul, utilisé pour lier un nom à un résultat, qui est du type de calcul.
match Utilisé pour créer une branche en comparant une valeur à un modèle.
member Utilisé pour déclarer une propriété ou une méthode dans un type d'objet.
module Utilisé pour associer un nom à un groupe de types, valeurs et fonctions associés, pour le séparer logiquement des autres codes.
mutable Utilisé pour déclarer une variable, c'est-à-dire une valeur qui peut être modifiée.
namespace Utilisé pour associer un nom à un groupe de types et de modules associés, pour le séparer logiquement des autres codes.
new

Utilisé pour déclarer, définir ou appeler un constructeur qui crée ou qui peut créer un objet.

Également utilisé dans les contraintes de paramètres génériques pour indiquer qu'un type doit avoir un certain constructeur.

not Pas vraiment un mot-clé. Cependant, pas struct in combination est utilisé comme contrainte de paramètre générique.
null

Indique l'absence d'objet.

Également utilisé dans les contraintes de paramètres génériques.

of Utilisé dans les unions discriminées pour indiquer le type de catégories de valeurs, et dans les déclarations de délégué et d'exception.
open Utilisé pour rendre le contenu d'un espace de noms ou d'un module disponible sans qualification.
or

Utilisé avec des conditions booléennes comme opérateur booléen ou. Équivalent à ||.

Également utilisé dans les contraintes de membre.

override Utilisé pour implémenter une version d'une méthode abstraite ou virtuelle qui diffère de la version de base.
private Restreint l'accès à un membre au code du même type ou module.
public Permet d'accéder à un membre depuis l'extérieur du type.
rec Utilisé pour indiquer qu'une fonction est récursive.
return Utilisé pour indiquer une valeur à fournir comme résultat d'une expression de calcul.
return! Utilisé pour indiquer une expression de calcul qui, lorsqu'elle est évaluée, fournit le résultat de l'expression de calcul contenant.
select Utilisé dans les expressions de requête pour spécifier les champs ou colonnes à extraire. Notez qu'il s'agit d'un mot-clé contextuel, ce qui signifie qu'il ne s'agit pas réellement d'un mot réservé et qu'il agit uniquement comme un mot-clé dans le contexte approprié.
static Utilisé pour indiquer une méthode ou une propriété qui peut être appelée sans instance d'un type, ou un membre de valeur qui est partagé entre toutes les instances d'un type.
struct

Utilisé pour déclarer un type de structure.

Également utilisé dans les contraintes de paramètres génériques.

Utilisé pour la compatibilité OCaml dans les définitions de module.

then

Utilisé dans les expressions conditionnelles.

Également utilisé pour effectuer des effets secondaires après la construction de l'objet.

to Utilisé dans les boucles for pour indiquer une plage.
true Utilisé comme littéral booléen.
try Utilisé pour introduire un bloc de code susceptible de générer une exception. Utilisé avec ou enfin.
type Utilisé pour déclarer une classe, un enregistrement, une structure, une union discriminée, un type d'énumération, une unité de mesure ou une abréviation de type.
upcast Utilisé pour convertir en un type plus élevé dans la chaîne d'héritage.
use Utilisé à la place de let pour les valeurs qui nécessitent l'appel de Dispose pour libérer des ressources.
use! Utilisé au lieu de laisser! dans les workflows asynchrones et autres expressions de calcul pour les valeurs qui nécessitent l'appel de Dispose pour libérer des ressources.
val Utilisé dans une signature pour indiquer une valeur, ou dans un type pour déclarer un membre, dans des situations limitées.
void Indique le type void .NET. Utilisé lors de l'interopérabilité avec d'autres langages .NET.
when Utilisé pour les conditions booléennes (lorsque les gardes) sur les correspondances de modèle et pour introduire une clause de contrainte pour un paramètre de type générique.
while Introduit une construction en boucle.
with Utilisé avec le mot-clé match dans les expressions de correspondance de modèle. Également utilisé dans les expressions d'objet, les expressions de copie d'enregistrement et les extensions de type pour introduire des définitions de membre et pour introduire des gestionnaires d'exceptions.
yield Utilisé dans une expression de séquence pour produire une valeur pour une séquence.
yield! Utilisé dans une expression de calcul pour ajouter le résultat d'une expression de calcul donnée à une collection de résultats pour l'expression de calcul contenant.

Certains mots-clés réservés proviennent du langage OCaml -

asr terre lor lsl lsr lxor mod sig

Certains autres mots-clés réservés sont conservés pour l'expansion future de F #.

atomique Pause vérifié composant const contrainte constructor
continue eager event external fixed functor include
method mixin object parallel process protected pure
sealed tailcall trait virtual volatile

Comments in F#

F# provides two types of comments −

  • One line comment starts with // symbol.
  • Multi line comment starts with (* and ends with *).

A Basic Program and Application Entry Point in F#

Generally, you don’t have any explicit entry point for F# programs. When you compile an F# application, the last file provided to the compiler becomes the entry point and all top level statements in that file are executed from top to bottom.

A well-written program should have a single top-level statement that would call the main loop of the program.

A very minimalistic F# program that would display ‘Hello World’ on the screen −

(* This is a comment *)
(* Sample Hello World program using F# *)
printfn "Hello World!"

When you compile and execute the program, it yields the following output −

Hello World!

The data types in F# can be classified as follows −

  • Integral types
  • Floating point types
  • Text types
  • Other types

Integral Data Type

The following table provides the integral data types of F#. These are basically integer data types.

F# Type Size Range Example Remarks
sbyte 1 byte -128 to 127

42y

-11y

8-bit signed integer
byte 1 byte 0 to 255

42uy

200uy

8-bit unsigned integer
int16 2 bytes -32768 to 32767

42s

-11s

16-bit signed integer
uint16 2 bytes 0 to 65,535

42us

200us

16-bit unsigned integer
int/int32 4 bytes -2,147,483,648 to 2,147,483,647

42

-11

32-bit signed integer
uint32 4 bytes 0 to 4,294,967,295

42u

200u

32-bit unsigned integer
int64 8 bytes -9,223,372,036,854,775,808 to 9,223,372,036,854,775,807

42L

-11L

64-bit signed integer
uint64 8 bytes 0 to 18,446,744,073,709,551,615

42UL

200UL

64-bit unsigned integer
bigint At least 4 bytes any integer

42I

1499999

9999999

9999999

9999999

9999I

arbitrary precision integer

Example

(* single byte integer *)
let x = 268.97f
let y = 312.58f
let z = x + y

printfn "x: %f" x
printfn "y: %f" y
printfn "z: %f" z

(* unsigned 8-bit natural number *)

let p = 2uy
let q = 4uy
let r = p + q

printfn "p: %i" p
printfn "q: %i" q
printfn "r: %i" r

(* signed 16-bit integer *)

let a = 12s
let b = 24s
let c = a + b

printfn "a: %i" a
printfn "b: %i" b
printfn "c: %i" c

(* signed 32-bit integer *)

let d = 212l
let e = 504l
let f = d + e

printfn "d: %i" d
printfn "e: %i" e
printfn "f: %i" f

When you compile and execute the program, it yields the following output −

x: 1
y: 2
z: 3
p: 2
q: 4
r: 6
a: 12
b: 24
c: 36
d: 212
e: 504
f: 716

Floating Point Data Types

The following table provides the floating point data types of F#.

F# Type Size Range Example Remarks
float32 4 bytes ±1.5e-45 to ±3.4e38

42.0F

-11.0F

32-bit signed floating point number (7 significant digits)
float 8 bytes ±5.0e-324 to ±1.7e308

42.0

-11.0

64-bit signed floating point number (15-16 significant digits)
decimal 16 bytes ±1.0e-28 to ±7.9e28

42.0M

-11.0M

128-bit signed floating point number (28-29 significant digits)
BigRational At least 4 bytes Any rational number.

42N

-11N

Arbitrary precision rational number. Using this type requires a reference to FSharp.PowerPack.dll.

Example

(* 32-bit signed floating point number *)
(* 7 significant digits *)

let d = 212.098f
let e = 504.768f
let f = d + e

printfn "d: %f" d
printfn "e: %f" e
printfn "f: %f" f

(* 64-bit signed floating point number *)
(* 15-16 significant digits *)
let x = 21290.098
let y = 50446.768
let z = x + y

printfn "x: %g" x
printfn "y: %g" y
printfn "z: %g" z

When you compile and execute the program, it yields the following output −

d: 212.098000
e: 504.768000
f: 716.866000
x: 21290.1
y: 50446.8
z: 71736.9

Text Data Types

The following table provides the text data types of F#.

F# Type Size Range Example Remarks
char 2 bytes U+0000 to U+ffff

'x'

'\t'

Single unicode characters
string 20 + (2 * string's length) bytes 0 to about 2 billion characters

"Hello"

"World"

Unicode text

Example

let choice = 'y'
let name = "Zara Ali"
let org = "Tutorials Point"

printfn "Choice: %c" choice
printfn "Name: %s" name
printfn "Organisation: %s" org

When you compile and execute the program, it yields the following output −

Choice: y
Name: Zara Ali
Organisation: Tutorials Point

Other Data Types

The following table provides some other data types of F#.

F# Type Size Range Example Remarks
bool 1 byte Only two possible values, true or false

true

false

Stores boolean values

Example

let trueVal = true
let falseVal = false

printfn "True Value: %b" (trueVal)
printfn "False Value: %b" (falseVal)

When you compile and execute the program, it yields the following output −

True Value: true
False Value: false

A variable is a name given to a storage area that our programs can manipulate. Each variable has a specific type, which determines the size and layout of the variable's memory; the range of values that can be stored within that memory; and the set of operations that can be applied to the variable.

Variable Declaration in F#

The let keyword is used for variable declaration −

For example,

let x = 10

It declares a variable x and assigns the value 10 to it.

You can also assign an expression to a variable −

let x = 10
let y = 20
let z = x + y

The following example illustrates the concept −

Example

let x = 10
let y = 20
let z = x + y

printfn "x: %i" x
printfn "y: %i" y
printfn "z: %i" z

When you compile and execute the program, it yields the following output −

x: 10
y: 20
z: 30

Variables in F# are immutable, which means once a variable is bound to a value, it can’t be changed. They are actually compiled as static read-only properties.

The following example demonstrates this.

Example

let x = 10
let y = 20
let z = x + y

printfn "x: %i" x
printfn "y: %i" y
printfn "z: %i" z

let x = 15
let y = 20
let z = x + y

printfn "x: %i" x
printfn "y: %i" y
printfn "z: %i" z

When you compile and execute the program, it shows the following error message −

Duplicate definition of value 'x'
Duplicate definition of value 'Y'
Duplicate definition of value 'Z'

Variable Definition With Type Declaration

A variable definition tells the compiler where and how much storage for the variable should be created. A variable definition may specify a data type and contains a list of one or more variables of that type as shown in the following example.

Example

let x:int32 = 10
let y:int32 = 20
let z:int32 = x + y

printfn "x: %d" x
printfn "y: %d" y
printfn "z: %d" z

let p:float = 15.99
let q:float = 20.78
let r:float = p + q

printfn "p: %g" p
printfn "q: %g" q
printfn "r: %g" r

When you compile and execute the program, it shows the following error message −

x: 10
y: 20
z: 30
p: 15.99
q: 20.78
r: 36.77

Mutable Variables

At times you need to change the values stored in a variable. To specify that there could be a change in the value of a declared and assigned variable, in later part of a program, F# provides the mutable keyword. You can declare and assign mutable variables using this keyword, whose values you will change.

The mutable keyword allows you to declare and assign values in a mutable variable.

You can assign some initial value to a mutable variable using the let keyword. However, to assign new subsequent value to it, you need to use the operator.

For example,

let mutable x = 10
x ← 15

The following example will clear the concept −

Example

let mutable x = 10
let y = 20
let mutable z = x + y

printfn "Original Values:"
printfn "x: %i" x
printfn "y: %i" y
printfn "z: %i" z

printfn "Let us change the value of x"
printfn "Value of z will change too."

x <- 15
z <- x + y

printfn "New Values:"
printfn "x: %i" x
printfn "y: %i" y
printfn "z: %i" z

When you compile and execute the program, it yields the following output −

Original Values:
x: 10
y: 20
z: 30
Let us change the value of x
Value of z will change too.
New Values:
x: 15
y: 20
z: 35

An operator is a symbol that tells the compiler to perform specific mathematical or logical manipulations. F# is rich in built-in operators and provides the following types of operators −

  • Arithmetic Operators
  • Comparison Operators
  • Boolean Operators
  • Bitwise Operators

Arithmetic Operators

The following table shows all the arithmetic operators supported by F# language. Assume variable A holds 10 and variable B holds 20 then −

Show Example

Operator Description Example
+ Adds two operands A + B will give 30
- Subtracts second operand from the first A - B will give -10
* Multiplies both operands A * B will give 200
/ Divides numerator by de-numerator B / A will give 2
% Modulus Operator and remainder of after an integer division B % A will give 0
** Exponentiation Operator, raises an operand to the power of another B**A will give 2010

Comparison Operators

The following table shows all the comparison operators supported by F# language. These binary comparison operators are available for integral and floating-point types. These operators return values of type bool.

Assume variable A holds 10 and variable B holds 20, then −

Show Example

Operator Description Example
= Checks if the values of two operands are equal or not, if yes then condition becomes true. (A == B) is not true.
<> Checks if the values of two operands are equal or not, if values are not equal then condition becomes true. (A <> B) is true.
> Checks if the value of left operand is greater than the value of right operand, if yes then condition becomes true. (A > B) is not true.
< Checks if the value of left operand is less than the value of right operand, if yes then condition becomes true. (A < B) is true.
>= Checks if the value of left operand is greater than or equal to the value of right operand, if yes then condition becomes true. (A >= B) is not true.
<= Checks if the value of left operand is less than or equal to the value of right operand, if yes then condition becomes true. (A <= B) is true.

Boolean Operators

The following table shows all the Boolean operators supported by F# language. Assume variable A holds true and variable B holds false, then −

Show Example

Operator Description Example
&& Called Boolean AND operator. If both the operands are non-zero, then condition becomes true. (A && B) is false.
|| Called Boolean OR Operator. If any of the two operands is non-zero, then condition becomes true. (A || B) is true.
not Called Boolean NOT Operator. Use to reverses the logical state of its operand. If a condition is true then Logical NOT operator will make false. not (A && B) is true.

Bitwise Operators

Bitwise operators work on bits and perform bit-by-bit operation. The truth tables for &&& (bitwise AND), ||| (bitwise OR), and ^^^ (bitwise exclusive OR) are as follows −

Show Example

p q p &&& q p ||| q p ^^^ q
0 0 0 0 0
0 1 0 1 1
1 1 1 1 0
1 0 0 1 1

Assume if A = 60; and B = 13; now in binary format they will be as follows −

A = 0011 1100

B = 0000 1101

A&&&B = 0000 1100

A|||B = 0011 1101

A^^^B = 0011 0001

~~~A = 1100 0011

The Bitwise operators supported by F# language are listed in the following table. Assume variable A holds 60 and variable B holds 13, then −

Operator Description Example
&&& Binary AND Operator copies a bit to the result if it exists in both operands. (A &&& B) will give 12, which is 0000 1100
||| Binary OR Operator copies a bit if it exists in either operand. (A ||| B) will give 61, which is 0011 1101
^^^ Binary XOR Operator copies the bit if it is set in one operand but not both. (A ^^^ B) will give 49, which is 0011 0001
~~~ Binary Ones Complement Operator is unary and has the effect of 'flipping' bits. (~~~A) will give -61, which is 1100 0011 in 2's complement form.
<<< Binary Left Shift Operator. The left operands value is moved left by the number of bits specified by the right operand. A <<< 2 will give 240 which is 1111 0000
>>> Binary Right Shift Operator. The left operands value is moved right by the number of bits specified by the right operand. A >>> 2 will give 15 which is 0000 1111

Operators Precedence

The following table shows the order of precedence of operators and other expression keywords in the F# language, from lowest precedence to the highest precedence.

Show Example

Operator Associativity
as Right
when Right
| (pipe) Left
; Right
let Non associative
function, fun, match, try Non associative
if Non associative
Right
:= Right
, Non associative
or, || Left
&, && Left
< op, >op, =, |op, &op Left
&&& , |||, ^^^, ~~~, <<<, >>> Left
^ op Right
:: Right
:?>, :? Non associative
- op, +op, (binary) Left
* op, /op, %op Left
** op Right
f x (function application) Left
| (pattern match) Right
prefix operators (+op, -op, %, %%, &, &&, !op, ~op) Left
. Left
f(x) Left
f<types> Left

Decision making structures require that the programmer specify one or more conditions to be evaluated or tested by the program. It should be along with a statement or statements to be executed if the condition is determined to be true, and optionally, other statements to be executed if the condition is determined to be false.

Following is the general form of a typical decision making structure found in most of the programming languages −

F# programming language provides the following types of decision making statements.

Statement Description
if /then statement An if/then statement consists of a Boolean expression followed by one or more statements.
if/then/ else statement An if/then statement can be followed by an optional else statement, which executes when the Boolean expression is false.
if/then/elif/else statement An if/then/elif/else statement allows you to have multiple else branches.
nested if statements You can use one if or else if statement inside another if or else if statement(s).

Programming languages provide various control structures that allow for more complicated execution paths.

A loop statement allows us to execute a statement or group of statements multiple times and following is the general form of a loop statement in most of the programming languages −

F# provides the following types of loops to handle the looping requirements.

Loop Type Description
for… to and for… downto expressions The for...to expression is used to iterate in a loop over a range of values of a loop variable. The for… downto expression reduces the value of loop variable.
for … in expression This form of for loop is used to iterate over collections of items i.e., loops over collections and sequences
While…do loop Repeats a statement or group of statements while a given condition is true. It tests the condition before executing the loop body.
nested loops You can use one or more loop inside any other for or while loop.

In F#, functions work like data types. You can declare and use a function in the same way like any other variable.

Since functions can be used like any other variables, you can −

  • Create a function, with a name and associate that name with a type.
  • Assign it a value.
  • Perform some calculation on that value.
  • Pass it as a parameter to another function or sub-routine.
  • Return a function as the result of another function.

Defining a Function

Functions are defined by using the let keyword. A function definition has the following syntax −

let [inline] function-name parameter-list [ : return-type ]
= function-body

Where,

  • function-name is an identifier that represents the function.

  • parameter-list gives the list of parameters separated by spaces. You can also specify an explicit type for each parameter and if not specified compiler tends to deduce it from the function body (like variables).

  • function-body consists of an expression, or a compound expression consisting of a number of expressions. The final expression in the function body is the return value.

  • return-type is a colon followed by a type and is optional. If the return type is not specified, then the compiler determines it from the final expression in the function body.

Parameters of a Function

You list the names of parameters right after the function name. You can specify the type of a parameter. The type of the parameter should follow the name of the parameter separated by a colon.

If no parameter type is specified, it is inferred by the compiler.

For example −

let doubleIt (x : int) = 2 * x

Calling a Function

A function is called by specifying the function name followed by a space and then any arguments separated by spaces.

For example −

let vol = cylinderVolume 3.0 5.0

The following programs illustrate the concepts.

Example 1

The following program calculates the volume of a cylinder when the radius and length are given as parameters

// the function calculates the volume of
// a cylinder with radius and length as parameters

let cylinderVolume radius length : float =

   // function body
   let pi = 3.14159
   length * pi * radius * radius

let vol = cylinderVolume 3.0 5.0
printfn " Volume: %g " vol

When you compile and execute the program, it yields the following output −

Volume: 141.372

Example 2

The following program returns the larger value of two given parameters −

// the function returns the larger value between two
// arguments

let max num1 num2 : int32 =
   // function body
   if(num1>num2)then
      num1
   else
      num2

let res = max 39 52
printfn " Max Value: %d " res

When you compile and execute the program, it yields the following output −

Max Value: 52

Example 3

let doubleIt (x : int) = 2 * x
printfn "Double 19: %d" ( doubleIt(19))

When you compile and execute the program, it yields the following output −

Double 19: 38

Recursive Functions

Recursive functions are functions that call themselves.

You define a recursive using the let rec keyword combination.

Syntax for defining a recursive function is −

//Recursive function definition
let rec function-name parameter-list = recursive-function-body

For example −

let rec fib n = if n < 2 then 1 else fib (n - 1) + fib (n - 2)

Example 1

The following program returns Fibonacci 1 to 10 −

let rec fib n = if n < 2 then 1 else fib (n - 1) + fib (n - 2)
for i = 1 to 10 do
   printfn "Fibonacci %d: %d" i (fib i)

When you compile and execute the program, it yields the following output −

Fibonacci 1: 1
Fibonacci 2: 2
Fibonacci 3: 3
Fibonacci 4: 5
Fibonacci 5: 8
Fibonacci 6: 13
Fibonacci 7: 21
Fibonacci 8: 34
Fibonacci 9: 55
Fibonacci 10: 89

Example 2

The following program returns factorial 8 −

open System
let rec fact x =
   if x < 1 then 1
   else x * fact (x - 1)
Console.WriteLine(fact 8)

When you compile and execute the program, it yields the following output −

40320

Arrow Notations in F#

F# reports about data type in functions and values, using a chained arrow notation. Let us take an example of a function that takes one int input, and returns a string. In arrow notation, it is written as −

int -> string

Data types are read from left to right.

Let us take another hypothetical function that takes two int data inputs and returns a string.

let mydivfunction x y = (x / y).ToString();;

F# reports the data type using chained arrow notation as −

val mydivfunction : x:int -> y:int -> string

The return type is represented by the rightmost data type in chained arrow notation.

Some more examples −

Notation Meaning
float → float → float The function takes two float inputs, returns another float.
int → string → float The function takes an int and a string input, returns a float.

Lambda Expressions

A lambda expression is an unnamed function.

Let us take an example of two functions −

let applyFunction ( f: int -> int -> int) x y = f x y
let mul x y = x * y
let res = applyFunction mul 5 7
printfn "%d" res

When you compile and execute the program, it yields the following output −

35

Now in the above example, if instead of defining the function mul, we could have used lambda expressions as −

let applyFunction ( f: int -> int -> int) x y = f x y
let res = applyFunction (fun x y -> x * y ) 5 7
printfn "%d" res

When you compile and execute the program, it yields the following output −

35

Function Composition and Pipelining

In F#, one function can be composed from other functions.

The following example shows the composition of a function named f, from two functions function1 and function2 −

let function1 x = x + 1
let function2 x = x * 5

let f = function1 >> function2
let res = f 10
printfn "%d" res

When you compile and execute the program, it yields the following output −

55

F# also provides a feature called pipelining of functions. Pipelining allows function calls to be chained together as successive operations.

The following example shows that −

let function1 x = x + 1
let function2 x = x * 5

let res = 10 |> function1 |> function2
printfn "%d" res

When you compile and execute the program, it yields the following output −

55

In F#, the string type represents immutable text as a sequence of Unicode characters.

String Literals

String literals are delimited by the quotation mark (") character.

Some special characters are there for special uses like newline, tab, etc. They are encoded using backslash (\) character. The backslash character and the related character make the escape sequence. The following table shows the escape sequence supported by F#.

Character Escape sequence
Backspace \b
Newline \n
Carriage return \r
Tab \t
Backslash \\
Quotation mark \"
Apostrophe \'
Unicode character \uXXXX or \UXXXXXXXX (where X indicates a hexadecimal digit)

Ways of lgnoring the Escape Sequence

The following two ways makes the compiler ignore the escape sequence −

  • Using the @ symbol.
  • Enclosing the string in triple quotes.

When a string literal is preceded by the @ symbol, it is called a verbatim string. In that way, all escape sequences in the string are ignored, except that two quotation mark characters are interpreted as one quotation mark character.

When a string is enclosed by triple quotes, then also all escape sequences are ignored, including double quotation mark characters.

Example

The following example demonstrates this technique showing how to work with XML or other structures that include embedded quotation marks −

// Using a verbatim string
let xmldata = @"<book author=""Lewis, C.S"" title=""Narnia"">"
printfn "%s" xmldata

When you compile and execute the program, it yields the following output −

<book author="Lewis, C.S" title="Narnia">

Basic Operators on Strings

The following table shows the basic operations on strings −

Value Description
collect : (char → string) → string → string Creates a new string whose characters are the results of applying a specified function to each of the characters of the input string and concatenating the resulting strings.
concat : string → seq<string> → string Returns a new string made by concatenating the given strings with a separator.
exists : (char → bool) → string → bool Tests if any character of the string satisfies the given predicate.
forall : (char → bool) → string → bool Tests if all characters in the string satisfy the given predicate.
init : int → (int → string) → string Creates a new string whose characters are the results of applying a specified function to each index and concatenating the resulting strings.
iter : (char → unit) → string → unit Applies a specified function to each character in the string.
iteri : (int → char → unit) → string → unit Applies a specified function to the index of each character in the string and the character itself.
length : string → int Returns the length of the string.
map : (char → char) → string → string Creates a new string whose characters are the results of applying a specified function to each of the characters of the input string.
mapi : (int → char → char) → string → string Creates a new string whose characters are the results of applying a specified function to each character and index of the input string.
replicate : int → string → string Returns a string by concatenating a specified number of instances of a string.

The following examples demonstrate the uses of some of the above functionalities −

Example 1

The String.collect function builds a new string whose characters are the results of applying a specified function to each of the characters of the input string and concatenating the resulting strings.

let collectTesting inputS =
   String.collect (fun c -> sprintf "%c " c) inputS
printfn "%s" (collectTesting "Happy New Year!")

When you compile and execute the program, it yields the following output −

H a p p y N e w Y e a r !

Example 2

The String.concat function concatenates a given sequence of strings with a separator and returns a new string.

let strings = [ "Tutorials Point"; "Coding Ground"; "Absolute Classes" ]
let ourProducts = String.concat "\n" strings
printfn "%s" ourProducts

When you compile and execute the program, it yields the following output −

Tutorials Point
Coding Ground
Absolute Classes

Example 3

The String.replicate method returns a string by concatenating a specified number of instances of a string.

printfn "%s" <| String.replicate 10 "*! "

When you compile and execute the program, it yields the following output −

*! *! *! *! *! *! *! *! *! *!

The option type in F# is used in calculations when there may or may not exist a value for a variable or function. Option types are used for representing optional values in calculations. They can have two possible values − Some(x) or None.

For example, a function performing a division will return a value in normal situation, but will throw exceptions in case of a zero denominator. Using options here will help to indicate whether the function has succeeded or failed.

An option has an underlying type and can hold a value of that type, or it might not have a value.

Using Options

Let us take the example of division function. The following program explains this −

Let us write a function div, and send two arguments to it 20 and 5 −

let div x y = x / y
let res = div 20 5
printfn "Result: %d" res

When you compile and execute the program, it yields the following output −

Result: 4

If the second argument is zero, then the program throws an exception −

let div x y = x / y
let res = div 20 0
printfn "Result: %d" res

When you compile and execute the program, it yields the following output −

Unhandled Exception:
System.DivideByZeroException: Division by zero

In such cases, we can use option types to return Some (value) when the operation is successful or None if the operation fails.

The following example demonstrates the use of options −

Example

let div x y =
   match y with
   | 0 -> None
   | _ -> Some(x/y)

let res : int option = div 20 4
printfn "Result: %A " res

When you compile and execute the program, it yields the following output −

Result: Some 5

Option Properties and Methods

The option type supports the following properties and methods −

Property or method Type Description
None 'T option A static property that enables you to create an option value that has the None value.
IsNone bool Returns true if the option has the None value.
IsSome bool Returns true if the option has a value that is not None.
Some 'T option A static member that creates an option that has a value that is not None.
Value 'T Returns the underlying value, or throws a NullReferenceException if the value is None.

Example 1

let checkPositive (a : int) =
   if a > 0 then
      Some(a)
   else
      None

let res : int option = checkPositive(-31)
printfn "Result: %A " res

When you compile and execute the program, it yields the following output −

Result: <null>

Example 2

let div x y =
   match y with
   | 0 -> None
   | _ -> Some(x/y)

let res : int option = div 20 4
printfn "Result: %A " res
printfn "Result: %A " res.Value

When you compile and execute the program, it yields the following output −

Result: Some 5
Result: 5

Example 3

let isHundred = function
   | Some(100) -> true
   | Some(_) | None -> false

printfn "%A" (isHundred (Some(45)))
printfn "%A" (isHundred (Some(100)))
printfn "%A" (isHundred None)

When you compile and execute the program, it yields the following output −

false
true
false

UNE tupleest une collection de valeurs séparées par des virgules. Ils sont utilisés pour créer des structures de données ad hoc, qui regroupent les valeurs associées.

Par exemple, («Zara Ali», «Hyderabad», 10) est un 3-tuple avec deux valeurs de chaîne et une valeur int, il a le type (string * string * int).

Les tuples peuvent être des paires, des triplets, etc., de types identiques ou différents.

Quelques exemples sont fournis ici -

// Tuple of two integers.
( 4, 5 )

// Triple of strings.
( "one", "two", "three" )

// Tuple of unknown types.
( a, b )

// Tuple that has mixed types.
( "Absolute Classes", 1, 2.0 )

// Tuple of integer expressions.
( a * 4, b + 7)

Exemple

Ce programme a une fonction qui prend un tuple de quatre valeurs flottantes et renvoie la moyenne -

let averageFour (a, b, c, d) =
   let sum = a + b + c + d
   sum / 4.0

let avg:float = averageFour (4.0, 5.1, 8.0, 12.0)
printfn "Avg of four numbers: %f" avg

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Avg of four numbers: 7.275000

Accès aux membres individuels de Tuple

Les membres individuels d'un tuple peuvent être évalués et imprimés à l'aide de la correspondance de motifs.

L'exemple suivant illustre le concept -

Exemple

let display tuple1 =
   match tuple1 with
   | (a, b, c) -> printfn "Detail Info: %A %A %A" a b c

display ("Zara Ali", "Hyderabad", 10 )

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Detail Info: "Zara Ali" "Hyderabad" 10

F # a deux fonctions intégrées, fst et snd, qui renvoient les premier et deuxième éléments dans un 2-tuple.

L'exemple suivant illustre le concept -

Exemple

printfn "First member: %A" (fst(23, 30))
printfn "Second member: %A" (snd(23, 30))

printfn "First member: %A" (fst("Hello", "World!"))
printfn "Second member: %A" (snd("Hello", "World!"))

let nameTuple = ("Zara", "Ali")

printfn "First Name: %A" (fst nameTuple)
printfn "Second Name: %A" (snd nameTuple)

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

First member: 23
Second member: 30
First member: "Hello"
Second member: "World!"
First Name: "Zara"
Second Name: "Ali"

UNE recordest similaire à un tuple, mais il contient des champs nommés. Par exemple,

type website =
   { title : string;
      url : string }

Définition de l'enregistrement

Un enregistrement est défini comme un type à l'aide de type mot-clé et les champs de l'enregistrement sont définis comme une liste séparée par des points-virgules.

La syntaxe pour définir un enregistrement est -

type recordName =
   { [ fieldName : dataType ] + }

Créer un enregistrement

Vous pouvez créer un enregistrement en spécifiant les champs de l'enregistrement. Par exemple, créons un enregistrement de site Web nommé page d'accueil -

let homepage = { Title = "TutorialsPoint"; Url = "www.tutorialspoint.com" }

Les exemples suivants expliquent les concepts -

Exemple 1

Ce programme définit un type d'enregistrement nommé site Web. Ensuite, il crée des enregistrements de type site Web et imprime les enregistrements.

(* defining a record type named website *)
type website =
   { Title : string;
      Url : string }

(* creating some records *)
let homepage = { Title = "TutorialsPoint"; Url = "www.tutorialspoint.com" }
let cpage = { Title = "Learn C"; Url = "www.tutorialspoint.com/cprogramming/index.htm" }
let fsharppage = { Title = "Learn F#"; Url = "www.tutorialspoint.com/fsharp/index.htm" }
let csharppage = { Title = "Learn C#"; Url = "www.tutorialspoint.com/csharp/index.htm" }

(*printing records *)
(printfn "Home Page: Title: %A \n \t URL: %A") homepage.Title homepage.Url
(printfn "C Page: Title: %A \n \t URL: %A") cpage.Title cpage.Url
(printfn "F# Page: Title: %A \n \t URL: %A") fsharppage.Title fsharppage.Url
(printfn "C# Page: Title: %A \n \t URL: %A") csharppage.Title csharppage.Url

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Home Page: Title: "TutorialsPoint"
       URL: "www.tutorialspoint.com"
C Page: Title: "Learn C"
      URL: "www.tutorialspoint.com/cprogramming/index.htm"
F# Page: Title: "Learn F#"
      URL: "www.tutorialspoint.com/fsharp/index.htm"
C# Page: Title: "Learn C#"
      URL: "www.tutorialspoint.com/csharp/index.htm"

Exemple 2

type student =
   { Name : string;
      ID : int;
      RegistrationText : string;
      IsRegistered : bool }

let getStudent name id =
   { Name = name; ID = id; RegistrationText = null; IsRegistered = false }

let registerStudent st =
   { st with
      RegistrationText = "Registered";
      IsRegistered = true }

let printStudent msg st =
   printfn "%s: %A" msg st

let main() =
   let preRegisteredStudent = getStudent "Zara" 10
   let postRegisteredStudent = registerStudent preRegisteredStudent

   printStudent "Before Registration: " preRegisteredStudent
   printStudent "After Registration: " postRegisteredStudent

main()

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Before Registration: : {Name = "Zara";
   ID = 10;
   RegistrationText = null;
   IsRegistered = false;}
After Registration: : {Name = "Zara";
   ID = 10;
   RegistrationText = "Registered";
   IsRegistered = true;}

En F #, une liste est une série ordonnée et immuable d'éléments du même type. Il est dans une certaine mesure équivalent à une structure de données de liste chaînée.

Le module F #, Microsoft.FSharp.Collections.List,a les opérations communes sur les listes. Cependant F # importe ce module automatiquement et le rend accessible à toutes les applications F #.

Créer et initialiser une liste

Voici les différentes manières de créer des listes -

  • Utilisation de la liste literals.

  • En utilisant cons (: :) opérateur.

  • En utilisant le List.init méthode du module Liste.

  • En utilisant certains syntactic constructs appelé List Comprehensions.

Lister les littéraux

Dans cette méthode, vous spécifiez simplement une séquence de valeurs délimitée par des points-virgules entre crochets. Par exemple -

let list1 = [1; 2; 3; 4; 5; 6; 7; 8; 9; 10]

Les inconvénients (: :) Opérateur

Avec cette méthode, vous pouvez ajouter des valeurs en ajoutant ou cons-ingà une liste existante en utilisant l'opérateur ::. Par exemple -

let list2 = 1::2::3::4::5::6::7::8::9::10::[];;

[] désigne une liste vide.

List init, méthode

La méthode List.init du module List est souvent utilisée pour créer des listes. Cette méthode a le type -

val init : int -> (int -> 'T) -> 'T list

Le premier argument est la longueur souhaitée de la nouvelle liste et le second argument est une fonction d'initialisation, qui génère des éléments dans la liste.

Par exemple,

let list5 = List.init 5 (fun index -> (index, index * index, index * index * index))

Ici, la fonction index génère la liste.

Compréhensions de listes

Les compréhensions de listes sont des constructions syntaxiques spéciales utilisées pour générer des listes.

La syntaxe de compréhension de liste F # se présente sous deux formes: les plages et les générateurs.

Les plages ont les constructions - [début .. fin] et [début .. étape .. fin]

Par exemple,

let list3 = [1 .. 10]

Les générateurs ont la construction - [for x in collection do ... yield expr]

Par exemple,

let list6 = [ for a in 1 .. 10 do yield (a * a) ]

Comme le yield mot-clé pousse une seule valeur dans une liste, le mot-clé, yield!, pousse une collection de valeurs dans la liste.

La fonction suivante illustre les méthodes ci-dessus -

Exemple

(* using list literals *)
let list1 = [1; 2; 3; 4; 5; 6; 7; 8; 9; 10]
printfn "The list: %A" list1

(*using cons operator *)
let list2 = 1 :: 2 :: 3 :: []
printfn "The list: %A" list2

(* using range constructs*)
let list3 = [1 .. 10]
printfn "The list: %A" list3

(* using range constructs *)
let list4 = ['a' .. 'm']
printfn "The list: %A" list4

(* using init method *)
let list5 = List.init 5 (fun index -> (index, index * index, index * index * index))
printfn "The list: %A" list5

(* using yield operator *)
let list6 = [ for a in 1 .. 10 do yield (a * a) ]
printfn "The list: %A" list6

(* using yield operator *)
let list7 = [ for a in 1 .. 100 do if a % 3 = 0 && a % 5 = 0 then yield a]
printfn "The list: %A" list7

(* using yield! operator *)
let list8 = [for a in 1 .. 3 do yield! [ a .. a + 3 ] ]
printfn "The list: %A" list8

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

The list: [1; 2; 3; 4; 5; 6; 7; 8; 9; 10]
The list: [1; 2; 3]
The list: [1; 2; 3; 4; 5; 6; 7; 8; 9; 10]
The list: ['a'; 'b'; 'c'; 'd'; 'e'; 'f'; 'g'; 'h'; 'i'; 'j'; 'k'; 'l'; 'm']
The list: [(0, 0, 0); (1, 1, 1); (2, 4, 8); (3, 9, 27); (4, 16, 64)]
The list: [1; 4; 9; 16; 25; 36; 49; 64; 81; 100]
The list: [15; 30; 45; 60; 75; 90]
The list: [1; 2; 3; 4; 2; 3; 4; 5; 3; 4; 5; 6]

Propriétés du type de données de liste

Le tableau suivant présente diverses propriétés du type de données de liste -

Propriété Type La description
Tête 'T Le premier élément.
Vide 'T liste Une propriété statique qui renvoie une liste vide du type approprié.
Est vide booléen true si la liste ne contient aucun élément.
Article 'T L'élément à l'index spécifié (base zéro).
Longueur int Le nombre d'éléments.
Queue 'T liste La liste sans le premier élément.

L'exemple suivant montre l'utilisation de ces propriétés -

Exemple

let list1 = [ 2; 4; 6; 8; 10; 12; 14; 16 ]

// Use of Properties
printfn "list1.IsEmpty is %b" (list1.IsEmpty)
printfn "list1.Length is %d" (list1.Length)
printfn "list1.Head is %d" (list1.Head)
printfn "list1.Tail.Head is %d" (list1.Tail.Head)
printfn "list1.Tail.Tail.Head is %d" (list1.Tail.Tail.Head)
printfn "list1.Item(1) is %d" (list1.Item(1))

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

list1.IsEmpty is false
list1.Length is 8
list1.Head is 2
list1.Tail.Head is 4
list1.Tail.Tail.Head is 6
list1.Item(1) is 4

Opérateurs de base sur la liste

Le tableau suivant montre les opérations de base sur le type de données de liste -

Valeur La description
ajouter: 'T liste →' T liste → 'T liste Renvoie une nouvelle liste qui contient les éléments de la première liste suivis des éléments de la seconde.
moyenne: 'T liste → ^ T Renvoie la moyenne des éléments de la liste.
averageBy: ('T → ^ U) →' T liste → ^ U Renvoie la moyenne des éléments générés en appliquant la fonction à chaque élément de la liste.
choisissez: ('T →' U option) → 'T list →' U list Applique la fonction donnée à chaque élément de la liste. Renvoie la liste composée des résultats pour chaque élément où la fonction retourneSome.
collecter: ('T →' U liste) → 'T liste →' U liste Pour chaque élément de la liste, applique la fonction donnée. Concatène tous les résultats et renvoie la liste combinée.
concat: seq <'T liste> →' T liste Renvoie une nouvelle liste qui contient les éléments de chacune des listes dans l'ordre.
vide: 'T liste Renvoie une liste vide du type donné.
existe: ('T → bool) →' T liste → bool Teste si un élément de la liste satisfait le prédicat donné.
existe2: ('T1 →' T2 → booléen) → 'T1 liste →' T2 liste → booléen Teste si une paire d'éléments correspondants des listes satisfait le prédicat donné.
filtre: ('T → booléen) →' T liste → 'T liste Renvoie une nouvelle collection contenant uniquement les éléments de la collection pour lesquels le prédicat donné retourne true.
trouver: ('T → booléen) →' T liste → 'T Renvoie le premier élément pour lequel la fonction donnée renvoie true.
findIndex: ('T → booléen) →' T liste → int Renvoie l'index du premier élément de la liste qui satisfait le prédicat donné.
fold: ('State →' T → 'State) →' State → 'T list →' State Applique une fonction à chaque élément de la collection, en enfilant un argument d'accumulateur dans le calcul. Cette fonction prend le deuxième argument et lui applique la fonction ainsi que le premier élément de la liste. Ensuite, il transmet ce résultat à la fonction avec le deuxième élément, et ainsi de suite. Enfin, il renvoie le résultat final. Si la fonction d'entrée est f et les éléments sont i0 ... iN, alors cette fonction calcule f (... (fs i0) i1 ...) iN.
fold2: ('State →' T1 → 'T2 →' State) → 'State →' T1 list → 'T2 list →' State Applique une fonction aux éléments correspondants de deux collections, en enfilant un argument d'accumulateur dans le calcul. Les collections doivent avoir des tailles identiques. Si la fonction d'entrée est f et les éléments sont i0 ... iN et j0 ... jN, alors cette fonction calcule f (... (fs i0 j0) ...) iN jN.
foldBack: ('T →' State → 'State) →' T list → 'State →' State Applique une fonction à chaque élément de la collection, en enfilant un argument d'accumulateur dans le calcul. Si la fonction d'entrée estf et les éléments sont i0 ... iN alors calcule f i0 (... (f iN s)).
foldBack2: ('T1 →' T2 → 'State →' State) → 'T1 list →' T2 list → 'State →' State Applique une fonction aux éléments correspondants de deux collections, en enfilant un argument d'accumulateur dans le calcul. Les collections doivent avoir des tailles identiques. Si la fonction d'entrée est f et les éléments sont i0 ... iN et j0 ... jN, alors cette fonction calcule f i0 j0 (... (f iN jN s)).
forall: ('T → bool) →' T liste → bool Teste si tous les éléments de la collection satisfont au prédicat donné.
forall2: ('T1 →' T2 → booléen) → 'T1 list →' T2 list → bool Teste si tous les éléments correspondants de la collection satisfont le prédicat donné par paires.
head: 'T liste →' T Renvoie le premier élément de la liste.
init: int → (int → 'T) →' T liste Crée une liste en appelant le générateur donné sur chaque index.
isEmpty: 'T liste → booléen Retour true si la liste ne contient aucun élément, false autrement.
iter: ('T → unité) →' T liste → unité Applique la fonction donnée à chaque élément de la collection.
iter2: ('T1 →' T2 → unité) → 'Liste T1 →' Liste T2 → unité Applique la fonction donnée à deux collections simultanément. Les collections doivent avoir une taille identique.
iteri: (int → 'T → unité) →' T liste → unité Applique la fonction donnée à chaque élément de la collection. L'entier passé à la fonction indique l'index de l'élément.
iteri2: (int → 'T1 →' T2 → unité) → 'Liste T1 →' Liste T2 → unité Applique la fonction donnée à deux collections simultanément. Les collections doivent avoir une taille identique. L'entier passé à la fonction indique l'index de l'élément.
longueur: 'T liste → int Renvoie la longueur de la liste.
carte: ('T →' U) → 'T liste →' U liste Crée une nouvelle collection dont les éléments sont les résultats de l'application de la fonction donnée à chacun des éléments de la collection.
map2: ('T1 →' T2 → 'U) →' Liste T1 → 'Liste T2 →' Liste U Crée une nouvelle collection dont les éléments sont les résultats de l'application de la fonction donnée aux éléments correspondants des deux collections par paires.
map3: ('T1 →' T2 → 'T3 →' U) → 'Liste T1 →' Liste T2 → 'Liste T3 →' Liste U Crée une nouvelle collection dont les éléments sont le résultat de l'application simultanée de la fonction donnée aux éléments correspondants des trois collections.
mapi: (int → 'T →' U) → 'T liste →' U liste Crée une nouvelle collection dont les éléments sont les résultats de l'application de la fonction donnée à chacun des éléments de la collection. L'index entier passé à la fonction indique l'index (à partir de 0) de l'élément en cours de transformation.
mapi2: (int → 'T1 →' T2 → 'U) →' Liste T1 → 'Liste T2 →' Liste U Comme List.mapi, mais en mappant les éléments correspondants à partir de deux listes de longueur égale.
max: 'T liste →' T Renvoie le plus grand de tous les éléments de la liste, comparé à l'aide d'Operators.max.
maxBy: ('T →' U) → 'T liste →' T Renvoie le plus grand de tous les éléments de la liste, comparé à l'aide d'Operators.max sur le résultat de la fonction.
min: 'T liste →' T Renvoie le plus bas de tous les éléments de la liste, comparé à l'aide de Operators.min.
minBy: ('T →' U) → 'T liste →' T Renvoie le plus bas de tous les éléments de la liste, comparé à l'aide d'Operators.min sur le résultat de la fonction
nième: 'T liste → int →' T Indexe dans la liste. Le premier élément a l'index 0.
ofArray: 'T [] →' T liste Crée une liste à partir du tableau donné.
ofSeq: seq <'T> →' T liste Crée une nouvelle liste à partir de l'objet énumérable donné.
partition: ('T → booléen) →' T liste * 'T liste Divise la collection en deux collections, contenant les éléments pour lesquels le prédicat donné retourne true et false respectivement.
permute: (int → int) → 'T liste →' T liste Renvoie une liste avec tous les éléments permutés selon la permutation spécifiée.
choisir: ('T →' U option) → 'T liste →' U Applique la fonction donnée aux éléments successifs, retournant le premier résultat où la fonction retourne Some pour une certaine valeur.
réduire: ('T →' T → 'T) →' T liste → 'T Applique une fonction à chaque élément de la collection, en enfilant un argument d'accumulateur dans le calcul. Cette fonction applique la fonction spécifiée aux deux premiers éléments de la liste. Il transmet ensuite ce résultat dans la fonction avec le troisième élément, et ainsi de suite. Enfin, il renvoie le résultat final. Si la fonction d'entrée est f et les éléments sont i0 ... iN, alors cette fonction calcule f (... (f i0 i1) i2 ...) iN.
reductionBack: ('T →' T → 'T) →' T liste → 'T Applique une fonction à chaque élément de la collection, en enfilant un argument d'accumulateur dans le calcul. Si la fonction d'entrée estf et que les éléments sont i0 ... iN, alors cette fonction calcule f i0 (... (f iN-1 iN)).
répliquer: (int → 'T →' T liste) Crée une liste en appelant le générateur donné sur chaque index.
rev: 'T liste →' T liste Renvoie une nouvelle liste avec les éléments dans l'ordre inverse.
scan: ('State →' T → 'State) →' State → 'T list →' State list Applique une fonction à chaque élément de la collection, en enfilant un argument d'accumulateur dans le calcul. Cette fonction prend le deuxième argument et lui applique la fonction spécifiée ainsi que le premier élément de la liste. Ensuite, il passe ce résultat dans la fonction avec le deuxième élément et ainsi de suite. Enfin, il renvoie la liste des résultats intermédiaires et le résultat final.
scanBack: ('T →' State → 'State) →' T list → 'State →' State list Comme foldBack, mais renvoie les résultats intermédiaires et finaux
trier: 'T liste →' T liste Trie la liste donnée en utilisant Operators.compare.
sortBy: ('T →' Clé) → 'T liste →' T liste Trie la liste donnée en utilisant les clés données par la projection donnée. Les clés sont comparées à l'aide d'Operators.compare.
sortWith: ('T →' T → int) → 'T liste →' T liste Trie la liste donnée en utilisant la fonction de comparaison donnée.
somme: ^ T liste → ^ T Renvoie la somme des éléments de la liste.
sumBy: ('T → ^ U) →' T liste → ^ U Renvoie la somme des résultats générés en appliquant la fonction à chaque élément de la liste.
queue: 'T liste →' T liste Renvoie la liste d'entrée sans le premier élément.
toArray: 'T liste →' T [] Crée un tableau à partir de la liste donnée.
toSeq: 'T liste → seq <' T> Affiche la liste donnée sous forme de séquence.
tryFind: ('T → booléen) →' T liste → 'T option Renvoie le premier élément pour lequel la fonction donnée renvoie true. RevenirNone si aucun élément de ce type n'existe.
tryFindIndex: ('T → booléen) →' T liste → option int Renvoie l'index du premier élément de la liste qui satisfait le prédicat donné. RevenirNone si aucun élément de ce type n'existe.
tryPick: (option 'T →' U) → 'Liste T →' option U Applique la fonction donnée aux éléments successifs, retournant le premier résultat où la fonction retourne Somepour une certaine valeur. Si aucun élément de ce type n'existe, retournezNone.
décompressez: ('T1 *' T2) liste → 'T1 liste *' T2 liste Divise une liste de paires en deux listes.
décompressez3: ('T1 *' T2 * 'T3) liste →' T1 liste * 'T2 liste *' T3 liste Divise une liste de triplets en trois listes.
zip: 'Liste T1 →' Liste T2 → ('T1 *' T2) liste Combine les deux listes en une liste de paires. Les deux listes doivent avoir des longueurs égales.
zip3: 'Liste T1 →' Liste T2 → 'Liste T3 → (' T1 * 'T2 *' T3) liste Combine les trois listes en une liste de triplets. Les listes doivent avoir des longueurs égales.

Les exemples suivants illustrent les utilisations des fonctionnalités ci-dessus -

Exemple 1

Ce programme montre l'inversion d'une liste récursivement -

let list1 = [ 2; 4; 6; 8; 10; 12; 14; 16 ]
printfn "The original list: %A" list1

let reverse lt =
   let rec loop acc = function
      | [] -> acc
      | hd :: tl -> loop (hd :: acc) tl
   loop [] lt

printfn "The reversed list: %A" (reverse list1)

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

The original list: [2; 4; 6; 8; 10; 12; 14; 16]
The reversed list: [16; 14; 12; 10; 8; 6; 4; 2]

Cependant, vous pouvez utiliser le rev fonction du module dans le même but -

let list1 = [ 2; 4; 6; 8; 10; 12; 14; 16 ]
printfn "The original list: %A" list1
printfn "The reversed list: %A" (List.rev list1)

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

The original list: [2; 4; 6; 8; 10; 12; 14; 16]
The reversed list: [16; 14; 12; 10; 8; 6; 4; 2]

Exemple 2

Ce programme montre le filtrage d'une liste à l'aide du List.filter méthode -

let list1 = [1; 2; 3; 4; 5; 6; 7; 8; 9; 10]
printfn "The list: %A" list1
let list2 = list1 |> List.filter (fun x -> x % 2 = 0);;
printfn "The Filtered list: %A" list2

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

The list: [1; 2; 3; 4; 5; 6; 7; 8; 9; 10]
The Filtered list: [2; 4; 6; 8; 10]

Exemple 3

le List.map méthode mappe une liste d'un type à un autre -

let list1 = [1; 2; 3; 4; 5; 6; 7; 8; 9; 10]
printfn "The list: %A" list1
let list2 = list1 |> List.map (fun x -> (x * x).ToString());;
printfn "The Mapped list: %A" list2

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

The list: [1; 2; 3; 4; 5; 6; 7; 8; 9; 10]
The Mapped list: ["1"; "4"; "9"; "16"; "25"; "36"; "49"; "64"; "81"; "100"]

Exemple 4

le List.append méthode et l'opérateur @ ajoute une liste à une autre -

let list1 = [1; 2; 3; 4; 5 ]
let list2 = [6; 7; 8; 9; 10]
let list3 = List.append list1 list2

printfn "The first list: %A" list1
printfn "The second list: %A" list2
printfn "The appened list: %A" list3

let lt1 = ['a'; 'b';'c' ]
let lt2 = ['e'; 'f';'g' ]
let lt3 = lt1 @ lt2

printfn "The first list: %A" lt1
printfn "The second list: %A" lt2
printfn "The appened list: %A" lt3

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

The first list: [1; 2; 3; 4; 5]
The second list: [6; 7; 8; 9; 10]
The appened list: [1; 2; 3; 4; 5; 6; 7; 8; 9; 10]
The first list: ['a'; 'b'; 'c']
The second list: ['e'; 'f'; 'g']
The appened list: ['a'; 'b'; 'c'; 'e'; 'f'; 'g']

Exemple 5

le List.sortméthode trie une liste. leList.sum méthode donne la somme des éléments de la liste et la List.average méthode donne la moyenne des éléments de la liste -

let list1 = [9.0; 0.0; 2.0; -4.5; 11.2; 8.0; -10.0]
printfn "The list: %A" list1

let list2 = List.sort list1
printfn "The sorted list: %A" list2

let s = List.sum list1
let avg = List.average list1
printfn "The sum: %f" s
printfn "The average: %f" avg

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

The list: [9.0; 0.0; 2.0; -4.5; 11.2; 8.0; -10.0]
The sorted list: [-10.0; -4.5; 0.0; 2.0; 8.0; 9.0; 11.2]
The sum: 15.700000
The average: 2.242857

Une opération de «repli» applique une fonction à chaque élément d'une liste, agrège le résultat de la fonction dans une variable d'accumulateur et renvoie l'accumulateur comme résultat de l'opération de repli.

Exemple 6

le List.fold applique une fonction à chaque élément de gauche à droite, tandis que List.foldBack applique une fonction à chaque élément de droite à gauche.

let sumList list = List.fold (fun acc elem -> acc + elem) 0 list
printfn "Sum of the elements of list %A is %d." [ 1 .. 10 ] (sumList [ 1 .. 10 ])

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Sum of the elements of list [1; 2; 3; 4; 5; 6; 7; 8; 9; 10] is 55.

Les séquences, comme les listes, représentent également une collection ordonnée de valeurs. Cependant, les éléments d'une séquence ou d'une expression de séquence sont calculés lorsque cela est nécessaire. Ils ne sont pas calculés à la fois, et pour cette raison, ils sont utilisés pour représenter des structures de données infinies.

Définition de séquences

Les séquences sont définies à l'aide de la syntaxe suivante -

seq { expr }

Par exemple,

let seq1 = seq { 1 .. 10 }

Création d'expressions de séquences et de séquences

Comme pour les listes, vous pouvez créer des séquences à l'aide de plages et de compréhensions.

Les expressions de séquence sont les expressions que vous pouvez écrire pour créer des séquences. Celles-ci peuvent être faites -

  • En spécifiant la plage.
  • En spécifiant la plage avec incrémentation ou décrémentation.
  • En utilisant le yield mot-clé pour produire des valeurs qui font partie de la séquence.
  • En utilisant l'opérateur →.

Les exemples suivants illustrent le concept -

Exemple 1

(* Sequences *)
let seq1 = seq { 1 .. 10 }

(* ascending order and increment*)
printfn "The Sequence: %A" seq1
let seq2 = seq { 1 .. 5 .. 50 }

(* descending order and decrement*)
printfn "The Sequence: %A" seq2
let seq3 = seq {50 .. -5 .. 0}
printfn "The Sequence: %A" seq3

(* using yield *)
let seq4 = seq { for a in 1 .. 10 do yield a, a*a, a*a*a }
printfn "The Sequence: %A" seq4

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

The Sequence: seq [1; 2; 3; 4; ...]
The Sequence: seq [1; 6; 11; 16; ...]
The Sequence: seq [50; 45; 40; 35; ...]
The Sequence: seq [(1, 1, 1); (2, 4, 8); (3, 9, 27); (4, 16, 64); ...]

Exemple 2

Le programme suivant imprime les nombres premiers de 1 à 50 -

(* Recursive isprime function. *)
let isprime n =
   let rec check i =
      i > n/2 || (n % i <> 0 && check (i + 1))
   check 2

let primeIn50 = seq { for n in 1..50 do if isprime n then yield n }
for x in primeIn50 do
   printfn "%d" x

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

1
2
3
5
7
11
13
17
19
23
29
31
37
41
43
47

Opérations de base sur la séquence

Le tableau suivant présente les opérations de base sur le type de données de séquence -

Valeur La description
ajouter: seq <'T> → seq <' T> → seq <'T> Entoure les deux énumérations données en une seule énumération concaténée.
moyenne: seq <^ T> → ^ T Renvoie la moyenne des éléments de la séquence.
averageBy: ('T → ^ U) → seq <' T> → ^ U Renvoie la moyenne des résultats générés en appliquant la fonction à chaque élément de la séquence.
cache: seq <'T> → seq <' T> Renvoie une séquence qui correspond à une version mise en cache de la séquence d'entrée.
cast: IEnumerable → seq <'T> Enveloppe un système de typage lâche. Séquence de collections sous forme de séquence typée.
choisissez: (option 'T →' U) → seq <'T> → seq <' U> Applique la fonction donnée à chaque élément de la liste. Renvoie la liste composée des résultats pour chaque élément où la fonction retourneSome.
collect: ('T →' Collection) → seq <'T> → seq <' U> Applique la fonction donnée à chaque élément de la séquence et concatène tous les résultats.
compareWith: ('T →' T → int) → seq <'T> → seq <' T> → int Compare deux séquences à l'aide de la fonction de comparaison donnée, élément par élément.
concat: seq <'Collection> → seq <' T> Combine l'énumération d'énumérations donnée en une seule énumération concaténée.
countBy: (touche 'T →') → seq <'T> → seq <' Key * int> Applique une fonction de génération de clé à chaque élément d'une séquence et renvoie une séquence produisant des clés uniques et leur nombre d'occurrences dans la séquence d'origine.
délai: (unité → seq <'T>) → seq <' T> Renvoie une séquence construite à partir de la spécification différée donnée d'une séquence.
distinct: seq <'T> → seq <' T> Renvoie une séquence qui ne contient aucune entrée en double selon des comparaisons génériques de hachage et d'égalité sur les entrées. Si un élément apparaît plusieurs fois dans la séquence, les dernières occurrences sont supprimées.
distinctBy: (touche 'T →') → seq <'T> → seq <' T> Renvoie une séquence qui ne contient aucune entrée en double selon les comparaisons génériques de hachage et d'égalité sur les clés renvoyées par la fonction de génération de clé donnée. Si un élément apparaît plusieurs fois dans la séquence, les dernières occurrences sont supprimées.
vide: seq <'T> Crée une séquence vide.
exactementOne: seq <'T> →' T Renvoie le seul élément de la séquence.
existe: ('T → bool) → seq <' T> → bool Teste si un élément de la séquence satisfait le prédicat donné.
existe2: ('T1 →' T2 → booléen) → seq <'T1> → seq <' T2> → bool Teste si une paire d'éléments correspondants des séquences d'entrée satisfait le prédicat donné.
filtre: ('T → bool) → seq <' T> → seq <'T> Renvoie une nouvelle collection contenant uniquement les éléments de la collection pour lesquels le prédicat donné retourne true.
trouver: ('T → bool) → seq <' T> → 'T Renvoie le premier élément pour lequel la fonction donnée renvoie true.
findIndex: ('T → booléen) → seq <' T> → int Renvoie l'index du premier élément pour lequel la fonction donnée renvoie true.
fold: ('State →' T → 'State) →' State → seq <'T> →' State Applique une fonction à chaque élément de la collection, en enfilant un argument d'accumulateur dans le calcul. Si la fonction d'entrée est f et les éléments sont i0 ... iN, alors cette fonction calcule f (... (fs i0) ...) iN.
forall: ('T → bool) → seq <' T> → bool Teste si tous les éléments de la séquence satisfont au prédicat donné.
forall2: ('T1 →' T2 → booléen) → seq <'T1> → seq <' T2> → bool Teste toutes les paires d'éléments tirées des deux séquences satisfont au prédicat donné. Si une séquence est plus courte que l'autre, les éléments restants de la séquence plus longue sont ignorés.
groupBy: (Touche 'T →') → seq <'T> → seq <' Touche * seq <'T >> Applique une fonction de génération de clé à chaque élément d'une séquence et produit une séquence de clés uniques. Chaque clé unique contient également une séquence de tous les éléments correspondant à cette clé.
tête: seq <'T> →' T Renvoie le premier élément de la séquence.
init: int → (int → 'T) → seq <' T> Génère une nouvelle séquence qui, lorsqu'elle est itérée, retourne des éléments successifs en appelant la fonction donnée, jusqu'au nombre donné. Les résultats de l'appel de la fonction ne sont pas enregistrés, c'est-à-dire que la fonction est réappliquée si nécessaire pour régénérer les éléments. La fonction reçoit l'index de l'élément en cours de génération.
initInfinite: (int → 'T) → seq <' T> Génère une nouvelle séquence qui, une fois itérée, retournera des éléments successifs en appelant la fonction donnée. Les résultats de l'appel de la fonction ne sont pas enregistrés, c'est-à-dire que la fonction sera réappliquée si nécessaire pour régénérer les éléments. La fonction reçoit l'index de l'élément en cours de génération.
isEmpty: seq <'T> → booléen Teste si une séquence contient des éléments.
iter: ('T → unité) → seq <' T> → unité Applique la fonction donnée à chaque élément de la collection.
iter2: ('T1 →' T2 → unité) → seq <'T1> → seq <' T2> → unité Applique la fonction donnée à deux collections simultanément. Si une séquence est plus courte que l'autre, les éléments restants de la séquence plus longue sont ignorés.
iteri: (int → 'T → unité) → seq <' T> → unité Applique la fonction donnée à chaque élément de la collection. L'entier passé à la fonction indique l'index de l'élément.
dernier: seq <'T> →' T Renvoie le dernier élément de la séquence.
longueur: seq <'T> → int Renvoie la longueur de la séquence.
carte: ('T →' U) → seq <'T> → seq <' U> Crée une nouvelle collection dont les éléments sont les résultats de l'application de la fonction donnée à chacun des éléments de la collection. La fonction donnée sera appliquée lorsque les éléments sont demandés à l'aide de la méthode MoveNext sur les énumérateurs récupérés à partir de l'objet.
map2: ('T1 →' T2 → 'U) → seq <' T1> → seq <'T2> → seq <' U> Crée une nouvelle collection dont les éléments sont les résultats de l'application de la fonction donnée aux paires d'éléments correspondants des deux séquences. Si une séquence d'entrée est plus courte que l'autre, les éléments restants de la séquence plus longue sont ignorés.
mapi: (int → 'T →' U) → seq <'T> → seq <' U> Crée une nouvelle collection dont les éléments sont les résultats de l'application de la fonction donnée à chacun des éléments de la collection. L'index entier passé à la fonction indique l'index (à partir de 0) de l'élément en cours de transformation.
max: seq <'T> →' T Renvoie le plus grand de tous les éléments de la séquence, comparé à l'aide d'Operators.max.
maxBy: ('T →' U) → seq <'T> →' T Renvoie le plus grand de tous les éléments de la séquence, comparé à l'aide d'Operators.max sur le résultat de la fonction.
min: seq <'T> →' T Renvoie le plus bas de tous les éléments de la séquence, comparé à l'aide de Operators.min.
minBy: ('T →' U) → seq <'T> →' T Renvoie le plus bas de tous les éléments de la séquence, comparé à l'aide de Operators.min sur le résultat de la fonction.
nième: int → seq <'T> →' T Calcule le nième élément de la collection.
ofArray: 'T tableau → seq <' T> Affiche le tableau donné sous forme de séquence.
ofList: 'T liste → seq <' T> Affiche la liste donnée sous forme de séquence.
par paire: seq <'T> → seq <' T * 'T> Renvoie une séquence de chaque élément de la séquence d'entrée et de son prédécesseur, à l'exception du premier élément qui n'est renvoyé que comme prédécesseur du deuxième élément.
choisir: (option 'T →' U) → seq <'T> →' U Applique la fonction donnée aux éléments successifs, en retournant la première valeur où la fonction renvoie un Some valeur.
lecture seule: seq <'T> → seq <' T> Crée un nouvel objet séquence qui délègue à l'objet séquence donné. Cela garantit que la séquence d'origine ne peut pas être redécouverte et mutée par une conversion de type. Par exemple, si un tableau est donné, la séquence retournée renverra les éléments du tableau, mais vous ne pouvez pas convertir l'objet séquence retourné en tableau.
réduire: ('T →' T → 'T) → seq <' T> → 'T Applique une fonction à chaque élément de la séquence, en enfilant un argument d'accumulateur dans le calcul. Commencez par appliquer la fonction aux deux premiers éléments. Puis introduisez ce résultat dans la fonction avec le troisième élément et ainsi de suite. Renvoie le résultat final.
scan: ('State →' T → 'State) →' State → seq <'T> → seq <' State> Comme Seq.fold, mais calcule à la demande et renvoie la séquence des résultats intermédiaires et finaux.
singleton: 'T → seq <' T> Renvoie une séquence qui ne produit qu'un seul élément.
sauter: int → seq <'T> → seq <' T> Renvoie une séquence qui ignore un nombre spécifié d'éléments de la séquence sous-jacente, puis renvoie les éléments restants de la séquence.
skipWhile: ('T → booléen) → seq <' T> → seq <'T> Renvoie une séquence qui, lorsqu'elle est itérée, ignore les éléments de la séquence sous-jacente tandis que le prédicat donné retourne true, puis donne les éléments restants de la séquence.
trier: seq <'T> → seq <' T> Donne une séquence ordonnée par clés.
sortBy: (Touche 'T →') → seq <'T> → seq <' T> Applique une fonction de génération de clé à chaque élément d'une séquence et produit une séquence ordonnée par clés. Les clés sont comparées à l'aide de la comparaison générique implémentée par Operators.compare.
somme: seq <^ T> → ^ T Renvoie la somme des éléments de la séquence.
sommeBy Renvoie la somme des résultats générés en appliquant la fonction à chaque élément de la séquence.
prendre: int → seq <'T> → seq <' T> Renvoie les premiers éléments de la séquence jusqu'à un nombre spécifié.
takeWhile: ('T → booléen) → seq <' T> → seq <'T> Renvoie une séquence qui, lorsqu'elle est itérée, produit des éléments de la séquence sous-jacente tandis que le prédicat donné renvoie true, puis ne renvoie aucun autre élément.
toArray: seq <'T> →' T [] Crée un tableau à partir de la collection donnée.
toList: seq <'T> →' T liste Crée une liste à partir de la collection donnée.
tronquer: int → seq <'T> → seq <' T> Renvoie une séquence qui, lorsqu'elle est énumérée, ne renvoie pas plus d'un nombre spécifié d'éléments.
tryFind: ('T → bool) → seq <' T> → 'T option Renvoie le premier élément pour lequel la fonction donnée renvoie true, ou None si aucun élément de ce type n'existe.
tryFindIndex: ('T → bool) → seq <' T> → option int Renvoie l'index du premier élément de la séquence qui satisfait le prédicat donné, ou None si aucun élément de ce type n'existe.
tryPick: (option 'T →' U) → seq <'T> →' option U Applique la fonction donnée aux éléments successifs, en retournant la première valeur où la fonction renvoie un Some valeur.
déplier: (option 'Etat →' T * 'Etat) →' Etat → seq <'T> Renvoie une séquence contenant les éléments générés par le calcul donné.
où: ('T → bool) → seq <' T> → seq <'T> Renvoie une nouvelle collection contenant uniquement les éléments de la collection pour lesquels le prédicat donné retourne true. Un synonyme de Seq.filter.
fenêtré: int → seq <'T> → seq <' T []> Renvoie une séquence qui produit des fenêtres glissantes contenant des éléments tirés de la séquence d'entrée. Chaque fenêtre est renvoyée sous la forme d'un nouveau tableau.
zip: seq <'T1> → seq <' T2> → seq <'T1 *' T2> Combine les deux séquences en une liste de paires. Les deux séquences n'ont pas besoin d'avoir des longueurs égales - lorsqu'une séquence est épuisée, tous les éléments restants dans l'autre séquence sont ignorés.
zip3: seq <'T1> → seq <' T2> → seq <'T3> → seq <' T1 * 'T2 *' T3> Combine les trois séquences en une liste de triplets. Les séquences n'ont pas besoin d'avoir la même longueur - lorsqu'une séquence est épuisée, tous les éléments restants dans les autres séquences sont ignorés.

Les exemples suivants illustrent les utilisations de certaines des fonctionnalités ci-dessus -

Exemple 1

Ce programme crée une séquence vide et la remplit plus tard -

(* Creating sequences *)
let emptySeq = Seq.empty
let seq1 = Seq.singleton 20

printfn"The singleton sequence:"
printfn "%A " seq1
printfn"The init sequence:"

let seq2 = Seq.init 5 (fun n -> n * 3)
Seq.iter (fun i -> printf "%d " i) seq2
printfn""

(* converting an array to sequence by using cast *)
printfn"The array sequence 1:"
let seq3 = [| 1 .. 10 |] :> seq<int>
Seq.iter (fun i -> printf "%d " i) seq3
printfn""

(* converting an array to sequence by using Seq.ofArray *)
printfn"The array sequence 2:"
let seq4 = [| 2..2.. 20 |] |> Seq.ofArray
Seq.iter (fun i -> printf "%d " i) seq4
printfn""

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

The singleton sequence:
seq [20]
The init sequence:
0 3 6 9 12
The array sequence 1:
1 2 3 4 5 6 7 8 9 10
The array sequence 2:
2 4 6 8 10 12 14 16 18 20

Veuillez noter que -

  • La méthode Seq.empty crée une séquence vide.

  • La méthode Seq.singleton crée une séquence d'un seul élément spécifié.

  • La méthode Seq.init crée une séquence pour laquelle les éléments sont créés à l'aide d'une fonction donnée.

  • Les méthodes Seq.ofArray et Seq.ofList <'T> créent des séquences à partir de tableaux et de listes.

  • La méthode Seq.iter permet d'itérer dans une séquence.

Exemple 2

La méthode Seq.unfold génère une séquence à partir d'une fonction de calcul qui prend un état et le transforme pour produire chaque élément suivant de la séquence.

La fonction suivante produit les 20 premiers nombres naturels -

let seq1 = Seq.unfold (fun state -> if (state > 20) then None else Some(state, state + 1)) 0
printfn "The sequence seq1 contains numbers from 0 to 20."
for x in seq1 do printf "%d " x
printfn" "

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

The sequence seq1 contains numbers from 0 to 20.
0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20

Exemple 3

La méthode Seq.truncate crée une séquence à partir d'une autre séquence, mais limite la séquence à un nombre spécifié d'éléments.

La méthode Seq.take crée une nouvelle séquence qui contient un nombre spécifié d'éléments à partir du début d'une séquence.

let mySeq = seq { for i in 1 .. 10 -> 3*i }
let truncatedSeq = Seq.truncate 5 mySeq
let takeSeq = Seq.take 5 mySeq

printfn"The original sequence"
Seq.iter (fun i -> printf "%d " i) mySeq
printfn""

printfn"The truncated sequence"
Seq.iter (fun i -> printf "%d " i) truncatedSeq
printfn""

printfn"The take sequence"
Seq.iter (fun i -> printf "%d " i) takeSeq
printfn""

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

The original sequence
3 6 9 12 15 18 21 24 27 30
The truncated sequence
3 6 9 12 15
The take sequence
3 6 9 12 15

Un ensemble en F # est une structure de données qui agit comme une collection d'éléments sans préserver l'ordre dans lequel les éléments sont insérés. Les ensembles ne permettent pas d'insérer des entrées en double dans la collection.

Créer des ensembles

Les ensembles peuvent être créés des manières suivantes -

  • En créant un ensemble vide à l'aide de Set.empty et en ajoutant des éléments à l'aide de la fonction add.
  • Conversion de séquences et de listes en ensembles.

Le programme suivant montre les techniques -

(* creating sets *)
let set1 = Set.empty.Add(3).Add(5).Add(7). Add(9)
printfn"The new set: %A" set1

let weekdays = Set.ofList ["mon"; "tues"; "wed"; "thurs"; "fri"]
printfn "The list set: %A" weekdays

let set2 = Set.ofSeq [ 1 .. 2.. 10 ]
printfn "The sequence set: %A" set2

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

The new set: set [3; 5; 7; 9]
The list set: set ["fri"; "mon"; "thurs"; "tues"; "wed"]
The sequence set: set [1; 3; 5; 7; 9]

Opérations de base sur les ensembles

Le tableau suivant présente les opérations de base sur les postes -

Valeur La description
ajouter: 'T → Set <' T> → Set <'T> Renvoie un nouvel ensemble avec un élément ajouté à l'ensemble. Aucune exception n'est déclenchée si l'ensemble contient déjà l'élément donné.
contient: 'T → Set <' T> → booléen Évalue à true si l'élément donné est dans l'ensemble donné.
count: Set <'T> → int Renvoie le nombre d'éléments de l'ensemble.
différence: Set <'T> → Set <' T> → Set <'T> Renvoie un nouvel ensemble avec les éléments du second ensemble supprimés du premier.
vide: définir <'T> L'ensemble vide pour le type spécifié.
existe: ('T → bool) → Set <' T> → bool Teste si un élément de la collection satisfait le prédicat donné. Si la fonction d'entrée est prédicat et que les éléments sont i0 ... iN, cette fonction calcule le prédicat i0 ou ... ou le prédicat iN.
filtre: ('T → bool) → Set <' T> → Set <'T> Renvoie une nouvelle collection contenant uniquement les éléments de la collection pour lesquels le prédicat donné retourne true.
fold: ('State →' T → 'State) →' State → Set <'T> →' State Applique la fonction d'accumulation donnée à tous les éléments de l'ensemble.
foldBack: ('T →' State → 'State) → Set <' T> → 'State →' State Applique la fonction d'accumulation donnée à tous les éléments de l'ensemble.
forall: ('T → bool) → Set <' T> → bool Teste si tous les éléments de la collection satisfont au prédicat donné. Si la fonction d'entrée est p et les éléments sont i0 ... iN, alors cette fonction calcule p i0 && ... && p iN.
intersecter: Set <'T> → Set <' T> → Set <'T> Calcule l'intersection des deux ensembles.
intersectMany: seq <Set <'T >> → Set <' T> Calcule l'intersection d'une séquence d'ensembles. La séquence doit être non vide.
isEmpty: Set <'T> → booléen Retour true si l'ensemble est vide.
isProperSubset: Set <'T> → Set <' T> → booléen Évalue à true si tous les éléments du premier ensemble sont dans le second et qu'au moins un élément du second n'est pas dans le premier.
isProperSuperset: Set <'T> → Set <' T> → booléen Évalue à true si tous les éléments du second ensemble sont dans le premier et qu'au moins un élément du premier n'est pas dans le second.
isSubset: Set <'T> → Set <' T> → booléen Évalue à true si tous les éléments du premier ensemble sont dans le second.
isSuperset: Set <'T> → Set <' T> → booléen Évalue à true si tous les éléments du deuxième ensemble sont dans le premier.
iter: ('T → unité) → Set <' T> → unité Applique la fonction donnée à chaque élément de l'ensemble, dans l'ordre selon la fonction de comparaison.
map: ('T →' U) → Set <'T> → Set <' U> Renvoie une nouvelle collection contenant les résultats de l'application de la fonction donnée à chaque élément de l'ensemble d'entrée.
maxElement: Définir <'T> →' T Renvoie l'élément le plus élevé de l'ensemble en fonction de l'ordre utilisé pour l'ensemble.
minElement: Définir <'T> →' T Renvoie l'élément le plus bas de l'ensemble en fonction de l'ordre utilisé pour l'ensemble.
ofArray: 'T tableau → Définir <' T> Crée un ensemble contenant les mêmes éléments que le tableau donné.
ofList: 'T liste → Définir <' T> Crée un ensemble contenant les mêmes éléments que la liste donnée.
ofSeq: seq <'T> → Set <' T> Crée une nouvelle collection à partir de l'objet énumérable donné.
partition: ('T → booléen) → Set <' T> → Set <'T> * Set <' T> Divise l'ensemble en deux ensembles contenant les éléments pour lesquels le prédicat donné renvoie respectivement true et false.
supprimer: 'T → Set <' T> → Set <'T> Renvoie un nouvel ensemble avec l'élément donné supprimé. Aucune exception n'est déclenchée si l'ensemble ne contient pas l'élément donné.
singleton: 'T → Set <' T> L'ensemble contenant l'élément donné.
toArray: Set <'T> →' T tableau Crée un tableau qui contient les éléments de l'ensemble dans l'ordre.
toList: Set <'T> →' T liste Crée une liste qui contient les éléments de l'ensemble dans l'ordre.
toSeq: Set <'T> → seq <' T> Renvoie une vue ordonnée de la collection en tant qu'objet énumérable.
union: Set <'T> → Set <' T> → Set <'T> Calcule l'union des deux ensembles.
unionMany: seq <Set <'T >> → Set <' T> Calcule l'union d'une séquence d'ensembles.

L'exemple suivant montre les utilisations de certaines des fonctionnalités ci-dessus -

Exemple

let a = Set.ofSeq [ 1 ..2.. 20 ]
let b = Set.ofSeq [ 1 ..3 .. 20 ]
let c = Set.intersect a b
let d = Set.union a b
let e = Set.difference a b

printfn "Set a: "
Set.iter (fun x -> printf "%O " x) a
printfn""

printfn "Set b: "
Set.iter (fun x -> printf "%O " x) b
printfn""

printfn "Set c = set intersect of a and b : "
Set.iter (fun x -> printf "%O " x) c
printfn""

printfn "Set d = set union of a and b : "
Set.iter (fun x -> printf "%O " x) d
printfn""

printfn "Set e = set difference of a and b : "
Set.iter (fun x -> printf "%O " x) e
printfn""

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Set a:
1 3 5 7 9 11 13 15 17 19
Set b:
1 4 7 10 13 16 19
Set c = set intersect of a and b :
1 7 13 19
Set d = set union of a and b :
1 3 4 5 7 9 10 11 13 15 16 17 19
Set e = set difference of a and b :
3 5 9 11 15 17

En F #, une carte est un type spécial d'ensemble qui associe les valeurs à la clé. Une carte est créée de la même manière que les ensembles sont créés.

Créer des cartes

Les cartes sont créées en créant une carte vide à l'aide de Map.empty et en ajoutant des éléments à l'aide de la fonction Ajouter. L'exemple suivant illustre ceci -

Exemple

(* Create an empty Map *)
let students =
   Map.empty. (* Creating an empty Map *)
      Add("Zara Ali", "1501").
      Add("Rishita Gupta", "1502").
      Add("Robin Sahoo", "1503").
      Add("Gillian Megan", "1504");;
printfn "Map - students: %A" students

(* Convert a list to Map *)
let capitals =
   [ "Argentina", "Buenos Aires";
      "France ", "Paris";
      "Chili", "Santiago";
      "Malaysia", " Kuala Lumpur";
      "Switzerland", "Bern" ]
   |> Map.ofList;;
printfn "Map capitals : %A" capitals

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Map - students: map
[("Gillian Megan", "1504"); ("Rishita Gupta", "1502"); ("Robin Sahoo", "1503
");
("Zara Ali", "1501")]
Map capitals : map
[("Argentina", "Buenos Aires"); ("Chili", "Santiago"); ("France ", "Paris");
("Malaysia", " Kuala Lumpur"); ("Switzerland", "Bern")]

Vous pouvez accéder à des éléments individuels de la carte à l'aide de la touche.

Exemple

(* Create an empty Map *)
let students =
   Map.empty. (* Creating an empty Map *)
      Add("Zara Ali", "1501").
      Add("Rishita Gupta", "1502").
      Add("Robin Sahoo", "1503").
      Add("Gillian Megan", "1504");;
printfn "Map - students: %A" students

(*Accessing an element using key *)
printfn "%A" students.["Zara Ali"]

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Map - students: map
[("Gillian Megan", "1504"); ("Rishita Gupta", "1502"); ("Robin Sahoo", "1503
");
("Zara Ali", "1501")]
"1501"

Opérations de base sur les cartes

Ajouter un nom de module

Le tableau suivant présente les opérations de base sur les cartes -

Membre La description
Ajouter Renvoie une nouvelle carte avec la liaison ajoutée à la carte donnée.
Contient la clé Teste si un élément est dans le domaine de la carte.
Compter Le nombre de liaisons dans la carte.
Est vide Renvoie true s'il n'y a pas de liaisons dans la carte.
Article Recherchez un élément sur la carte. Déclenche KeyNotFoundException si aucune liaison n'existe dans la mappe.
Retirer Supprime un élément du domaine de la carte. Aucune exception n'est déclenchée si l'élément n'est pas présent.
EssayerTrouver Rechercher un élément dans la carte, renvoyer un Some valeur si l'élément est dans le domaine de la carte et None si non.

L'exemple suivant montre les utilisations de certaines des fonctionnalités ci-dessus -

Exemple

(* Create an empty Map *)
let students =
   Map.empty. (* Creating an empty Map *)
      Add("Zara Ali", "1501").
      Add("Rishita Gupta", "1502").
      Add("Robin Sahoo", "1503").
      Add("Gillian Megan", "1504").
      Add("Shraddha Dubey", "1505").
      Add("Novonil Sarker", "1506").
      Add("Joan Paul", "1507");;
printfn "Map - students: %A" students
printfn "Map - number of students: %d" students.Count

(* finding the registration number of a student*)
let found = students.TryFind "Rishita Gupta"
match found with
| Some x -> printfn "Found %s." x
| None -> printfn "Did not find the specified value."

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Map - students: map
[("Gillian Megan", "1504"); ("Joan Paul", "1507"); ("Novonil Sarker", "1506"
);
("Rishita Gupta", "1502"); ("Robin Sahoo", "1503");
("Shraddha Dubey", "1505"); ("Zara Ali", "1501")]
Map - number of students: 7
Found 1502.

Les unions ou unions discriminées vous permettent de créer des structures de données complexes représentant un ensemble de choix bien défini. Par exemple, vous devez créer une implémentation d'une variable de choix , qui a deux valeurs yes et no. À l'aide de l'outil Unions, vous pouvez concevoir cela.

Syntaxe

Les unions discriminées sont définies à l'aide de la syntaxe suivante -

type type-name =
   | case-identifier1 [of [ fieldname1 : ] type1 [ * [ fieldname2 : ] 
type2 ...]
   | case-identifier2 [of [fieldname3 : ]type3 [ * [ fieldname4 : ]type4 ...]
...

Notre implémentation simple de, choice, ressemblera à ce qui suit -

type choice =
   | Yes
   | No

L'exemple suivant utilise le choix de type -

type choice =
   | Yes
   | No

let x = Yes (* creates an instance of choice *)
let y = No (* creates another instance of choice *)
let main() =
   printfn "x: %A" x
   printfn "y: %A" y
main()

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

x: Yes
y: No

Exemple 1

L'exemple suivant montre l'implémentation des états de tension qui mettent un bit sur haut ou bas -

type VoltageState =
   | High
   | Low

let toggleSwitch = function (* pattern matching input *)
   | High -> Low
   | Low -> High

let main() =
   let on = High
   let off = Low
   let change = toggleSwitch off

   printfn "Switch on state: %A" on
   printfn "Switch off state: %A" off
   printfn "Toggle off: %A" change
   printfn "Toggle the Changed state: %A" (toggleSwitch change)

main()

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Switch on state: High
Switch off state: Low
Toggle off: High
Toggle the Changed state: Low

Exemple 2

type Shape =
   // here we store the radius of a circle
   | Circle of float

   // here we store the side length.
   | Square of float

   // here we store the height and width.
   | Rectangle of float * float

let pi = 3.141592654

let area myShape =
   match myShape with
   | Circle radius -> pi * radius * radius
   | Square s -> s * s
   | Rectangle (h, w) -> h * w

let radius = 12.0
let myCircle = Circle(radius)
printfn "Area of circle with radius %g: %g" radius (area myCircle)

let side = 15.0
let mySquare = Square(side)
printfn "Area of square that has side %g: %g" side (area mySquare)

let height, width = 5.0, 8.0
let myRectangle = Rectangle(height, width)
printfn "Area of rectangle with height %g and width %g is %g" height width (area myRectangle)

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Area of circle with radius 12: 452.389
Area of square that has side 15: 225
Area of rectangle with height 5 and width 8 is 40

Les variables en F # sont immutable,ce qui signifie qu'une fois qu'une variable est liée à une valeur, elle ne peut pas être modifiée. Ils sont en fait compilés en tant que propriétés statiques en lecture seule.

L'exemple suivant illustre cela.

Exemple

let x = 10
let y = 20
let z = x + y

printfn "x: %i" x
printfn "y: %i" y
printfn "z: %i" z

let x = 15
let y = 20
let z = x + y

printfn "x: %i" x
printfn "y: %i" y
printfn "z: %i" z

Lorsque vous compilez et exécutez le programme, il affiche le message d'erreur suivant -

Duplicate definition of value 'x'
Duplicate definition of value 'Y'
Duplicate definition of value 'Z'

Variables mutables

Parfois, vous devez modifier les valeurs stockées dans une variable. Pour spécifier qu'il pourrait y avoir un changement dans la valeur d'une variable déclarée et assignée dans une partie ultérieure d'un programme, F # fournit lemutablemot-clé. Vous pouvez déclarer et affecter des variables mutables à l'aide de ce mot clé, dont vous modifierez les valeurs.

le mutable Le mot clé vous permet de déclarer et d'attribuer des valeurs dans une variable mutable.

Vous pouvez attribuer une valeur initiale à une variable mutable en utilisant le letmot-clé. Cependant, pour lui attribuer une nouvelle valeur ultérieure, vous devez utiliser le<- opérateur.

Par exemple,

let mutable x = 10
x <- 15

L'exemple suivant clarifiera le concept -

Exemple

let mutable x = 10
let y = 20
let mutable z = x + y

printfn "Original Values:"
printfn "x: %i" x
printfn "y: %i" y
printfn "z: %i" z

printfn "Let us change the value of x"
printfn "Value of z will change too."

x <- 15
z <- x + y

printfn "New Values:"
printfn "x: %i" x
printfn "y: %i" y
printfn "z: %i" z

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Original Values:
x: 10
y: 20
z: 30
Let us change the value of x
Value of z will change too.
New Values:
x: 15
y: 20
z: 35

Utilisations des données mutables

Des données mutables sont souvent requises et utilisées dans le traitement des données, en particulier avec la structure des données d'enregistrement. L'exemple suivant illustre ceci -

open System

type studentData =
   { ID : int;
      mutable IsRegistered : bool;
      mutable RegisteredText : string; }

let getStudent id =
   { ID = id;
      IsRegistered = false;
      RegisteredText = null; }

let registerStudents (students : studentData list) =
   students |> List.iter(fun st ->
      st.IsRegistered <- true
      st.RegisteredText <- sprintf "Registered %s" (DateTime.Now.ToString("hh:mm:ss"))

      Threading.Thread.Sleep(1000) (* Putting thread to sleep for 1 second to simulate processing overhead. *))

let printData (students : studentData list) =
   students |> List.iter (fun x -> printfn "%A" x)

let main() =
   let students = List.init 3 getStudent

   printfn "Before Process:"
   printData students

   printfn "After process:"
   registerStudents students
   printData students

   Console.ReadKey(true) |> ignore

main()

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Before Process:
{ID = 0;
IsRegistered = false;
RegisteredText = null;}
{ID = 1;
IsRegistered = false;
RegisteredText = null;}
{ID = 2;
IsRegistered = false;
RegisteredText = null;}
After process:
{ID = 0;
IsRegistered = true;
RegisteredText = "Registered 05:39:15";}
{ID = 1;
IsRegistered = true;
RegisteredText = "Registered 05:39:16";}
{ID = 2;
IsRegistered = true;
RegisteredText = "Registered 05:39:17";}

Les tableaux sont des collections de taille fixe, de base zéro et modifiables d'éléments de données consécutifs qui sont tous du même type.

Création de tableaux

Vous pouvez créer des tableaux en utilisant différentes syntaxes et méthodes ou en utilisant les fonctions du module Array. Dans cette section, nous aborderons la création de tableaux sans utiliser les fonctions du module.

Il existe trois manières syntaxiques de créer des tableaux sans fonctions -

  • En listant les valeurs consécutives entre [| et |] et séparés par des points-virgules.
  • En mettant chaque élément sur une ligne distincte, auquel cas le séparateur point-virgule est facultatif.
  • En utilisant des expressions de séquence.

Vous pouvez accéder aux éléments du tableau en utilisant un opérateur point (.) Et des crochets ([et]).

L'exemple suivant illustre la création de tableaux -

//using semicolon separator
let array1 = [| 1; 2; 3; 4; 5; 6 |]
for i in 0 .. array1.Length - 1 do
   printf "%d " array1.[i]
printfn" "

// without semicolon separator
let array2 =
   [|
      1
      2
      3
      4
      5
   |]
for i in 0 .. array2.Length - 1 do
   printf "%d " array2.[i]
printfn" "

//using sequence
let array3 = [| for i in 1 .. 10 -> i * i |]
for i in 0 .. array3.Length - 1 do
   printf "%d " array3.[i]
printfn" "

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

1 2 3 4 5 6
1 2 3 4 5
1 4 9 16 25 36 49 64 81 100

Opérations de base sur les tableaux

Le module de bibliothèque Microsoft.FSharp.Collections.Array prend en charge les opérations sur les tableaux unidimensionnels.

Le tableau suivant présente les opérations de base sur les baies -

Valeur La description
ajouter: 'T [] →' T [] → 'T [] Crée un tableau qui contient les éléments d'un tableau suivis des éléments d'un autre tableau.
moyenne: ^ T [] → ^ T Renvoie la moyenne des éléments d'un tableau.
averageBy: ('T → ^ U) →' T [] → ^ U Renvoie la moyenne des éléments générés en appliquant une fonction à chaque élément d'un tableau.
blit: 'T [] → int →' T [] → int → int → unité Lit une plage d'éléments d'un tableau et les écrit dans un autre.
choisissez: ('T → U option) →' T [] → 'U [] Applique une fonction fournie à chaque élément d'un tableau. Renvoie un tableau qui contient les résultats x pour chaque élément pour lequel la fonction renvoie Some (x).
recueillir: ('T →' U []) → T [] → 'U [] Applique la fonction fournie à chaque élément d'un tableau, concatène les résultats et renvoie le tableau combiné.
concat: seq <'T []> →' T [] Crée un tableau qui contient les éléments de chacune des séquences de tableaux fournies.
copie: 'T →' T [] Crée un tableau qui contient les éléments du tableau fourni.
créer: int → 'T →' T [] Crée un tableau dont les éléments sont tous initialement la valeur fournie.
vide: 'T [] Renvoie un tableau vide du type donné.
existe: ('T → bool) →' T [] → bool Teste si un élément d'un tableau satisfait le prédicat fourni.
existe2: ('T1 →' T2 → booléen) → 'T1 [] →' T2 [] → booléen Teste si une paire d'éléments correspondants de deux tableaux satisfait la condition fournie.
fill: 'T [] → int → int →' T → unité Remplit une plage d'éléments d'un tableau avec la valeur fournie.
filtre: ('T → booléen) →' T [] → 'T [] Renvoie une collection qui contient uniquement les éléments du tableau fourni pour lesquels la condition fournie renvoie true.
trouver: ('T → booléen) →' T [] → 'T Renvoie le premier élément pour lequel la fonction fournie renvoie true. Déclenche KeyNotFoundException si aucun élément de ce type n'existe.
findIndex: ('T → booléen) →' T [] → int Renvoie l'index du premier élément d'un tableau qui satisfait la condition fournie. Déclenche KeyNotFoundException si aucun des éléments ne satisfait la condition.
fold: ('State →' T → 'State) →' State → 'T [] →' State Applique une fonction à chaque élément d'un tableau, en enfilant un argument d'accumulateur dans le calcul. Si la fonction d'entrée est f et les éléments du tableau sont i0 ... iN, cette fonction calcule f (... (fs i0) ...) iN.
fold2: ('État →' T1 → 'T2 →' État) → 'État →' T1 [] → 'T2 [] →' État Applique une fonction à des paires d'éléments de deux tableaux fournis, de gauche à droite, en passant un argument d'accumulateur dans le calcul. Les deux tableaux d'entrée doivent avoir les mêmes longueurs; sinon, ArgumentException est déclenchée.
foldBack: ('T →' Etat → 'Etat) →' T [] → 'Etat →' Etat Applique une fonction à chaque élément d'un tableau, en enfilant un argument d'accumulateur dans le calcul. Si la fonction d'entrée est f et les éléments du tableau sont i0 ... iN, cette fonction calcule f i0 (... (f iN s)).
foldBack2: ('T1 →' T2 → 'État →' État) → 'T1 [] →' T2 [] → 'État →' État Applique une fonction à des paires d'éléments de deux tableaux fournis, de droite à gauche, en passant un argument d'accumulateur dans le calcul. Les deux tableaux d'entrée doivent avoir les mêmes longueurs; sinon, ArgumentException est déclenchée.
forall: ('T → bool) →' T [] → bool Teste si tous les éléments d'un tableau satisfont à la condition fournie.
forall2: ('T1 →' T2 → booléen) → 'T1 [] →' T2 [] → bool Teste si tous les éléments correspondants de deux tableaux fournis satisfont à une condition fournie.
obtenir: 'T [] → int →' T Obtient un élément d'un tableau.
init: int → (int → 'T) →' T [] Utilise une fonction fournie pour créer un tableau de la dimension fournie.
isEmpty: 'T [] → booléen Teste si un tableau contient des éléments.
iter: ('T → unité) →' T [] → unité Applique la fonction fournie à chaque élément d'un tableau.
iter2: ('T1 →' T2 → unité) → 'T1 [] →' T2 [] → unité) Applique la fonction fournie à une paire d'éléments à partir d'index correspondants dans deux tableaux. Les deux tableaux doivent avoir les mêmes longueurs; sinon, ArgumentException est déclenchée.
iteri: (int → 'T → unité) →' T [] → unité Applique la fonction fournie à chaque élément d'un tableau. L'entier passé à la fonction indique l'index de l'élément.
iteri2: (int → 'T1 →' T2 → unité) → 'T1 [] →' T2 [] → unité Applique la fonction fournie à une paire d'éléments à partir d'index correspondants dans deux tableaux, en passant également l'index des éléments. Les deux tableaux doivent avoir les mêmes longueurs; sinon, une ArgumentException est déclenchée.
longueur: 'T [] → int Renvoie la longueur d'un tableau. La propriété Length fait la même chose.
carte: ('T →' U) → 'T [] →' U [] Crée un tableau dont les éléments sont les résultats de l'application de la fonction fournie à chacun des éléments d'un tableau fourni.
map2: ('T1 →' T2 → 'U) →' T1 [] → 'T2 [] →' U [] Crée un tableau dont les éléments sont les résultats de l'application de la fonction fournie aux éléments correspondants de deux tableaux fournis. Les deux tableaux d'entrée doivent avoir les mêmes longueurs; sinon, ArgumentException est déclenchée.
mapi: (int → 'T →' U) → 'T [] →' U [] Crée un tableau dont les éléments sont les résultats de l'application de la fonction fournie à chacun des éléments d'un tableau fourni. Un index entier passé à la fonction indique l'index de l'élément en cours de transformation.
mapi2: (int → 'T1 →' T2 → 'U) →' T1 [] → 'T2 [] →' U [] Crée un tableau dont les éléments sont les résultats de l'application de la fonction fournie aux éléments correspondants des deux collections par paires, en passant également l'index des éléments. Les deux tableaux d'entrée doivent avoir les mêmes longueurs; sinon, ArgumentException est déclenchée.
max: 'T [] →' T Renvoie le plus grand de tous les éléments d'un tableau. Operators.max est utilisé pour comparer les éléments.
maxBy: ('T →' U) → 'T [] →' T Renvoie le plus grand de tous les éléments d'un tableau, comparé via Operators.max sur le résultat de la fonction.
min: ('T [] →' T Renvoie le plus petit de tous les éléments d'un tableau. Operators.min est utilisé pour comparer les éléments.
minBy: ('T →' U) → 'T [] →' T Renvoie le plus petit de tous les éléments d'un tableau. Operators.min est utilisé pour comparer les éléments.
ofList: 'T liste →' T [] Crée un tableau à partir de la liste fournie.
ofSeq: seq <'T> →' T [] Crée un tableau à partir de l'objet énumérable fourni.
partition: ('T → booléen) →' T [] → 'T [] *' T [] Divise un tableau en deux tableaux, l'un contenant les éléments pour lesquels la condition fournie renvoie true, et l'autre contenant ceux pour lesquels il retourne false.
permute: (int → int) → 'T [] →' T [] Permute les éléments d'un tableau selon la permutation spécifiée.
choisir: (option 'T →' U) → 'T [] →' U Applique la fonction fournie aux éléments successifs d'un tableau fourni, renvoyant le premier résultat où la fonction renvoie Some (x) pour certains x. Si la fonction ne renvoie jamais Some (x), KeyNotFoundException est déclenchée.
réduire: ('T →' T → 'T) →' T [] → 'T Applique une fonction à chaque élément d'un tableau, en enfilant un argument d'accumulateur dans le calcul. Si la fonction d'entrée est f et les éléments du tableau sont i0 ... iN, cette fonction calcule f (... (f i0 i1) ...) iN. Si le tableau a une taille zéro, ArgumentException est déclenchée.
réduireBack: ('T →' T → 'T) →' T [] → 'T Applique une fonction à chaque élément d'un tableau, en enfilant un argument d'accumulateur dans le calcul. Si la fonction d'entrée est f et les éléments sont i0 ... iN, cette fonction calcule f i0 (... (f iN-1 iN)). Si le tableau a une taille zéro, ArgumentException est déclenchée.
rev: 'T [] →' T [] Inverse l'ordre des éléments dans un tableau fourni.
scan: ('State →' T → 'State) →' State → 'T [] →' State []) Se comporte comme un pli, mais renvoie les résultats intermédiaires avec les résultats finaux.
scanBack: ('T →' État → 'État) →' T [] → 'État →' État [] Se comporte comme foldBack, mais renvoie les résultats intermédiaires avec les résultats finaux.
set: 'T [] → int →' T → unité Définit un élément d'un tableau.
trier: 'T [] →' T [] Trie les éléments d'un tableau et renvoie un nouveau tableau. Operators.compare est utilisé pour comparer les éléments.
sortBy: (Clé 'T →') → 'T [] →' T [] Trie les éléments d'un tableau en utilisant la fonction fournie pour transformer les éléments dans le type sur lequel l'opération de tri est basée et renvoie un nouveau tableau. Operators.compare est utilisé pour comparer les éléments.
sortInPlace: 'T [] → unité Trie les éléments d'un tableau en modifiant le tableau en place, à l'aide de la fonction de comparaison fournie. Operators.compare est utilisé pour comparer les éléments.
sortInPlaceBy: (touche 'T →') → 'T [] → unité Trie les éléments d'un tableau en modifiant le tableau en place, en utilisant la projection fournie pour les clés. Operators.compare est utilisé pour comparer les éléments.
sortInPlaceWith: ('T →' T → int) → 'T [] → unité Trie les éléments d'un tableau à l'aide de la fonction de comparaison fournie pour modifier le tableau en place.
sortWith: ('T →' T → int) → 'T [] →' T [] Trie les éléments d'un tableau à l'aide de la fonction de comparaison fournie et renvoie un nouveau tableau.
sub: 'T [] → int → int →' T [] Crée un tableau qui contient la sous-plage fournie, qui est spécifiée par l'index de départ et la longueur.
somme: 'T [] → ^ T Renvoie la somme des éléments du tableau.
sommeBy: ('T → ^ U) →' T [] → ^ U Renvoie la somme des résultats générés en appliquant une fonction à chaque élément d'un tableau.
toList: 'T [] →' T liste Convertit le tableau fourni en liste.
toSeq: 'T [] → seq <' T> Affiche le tableau fourni sous forme de séquence.
tryFind: ('T → bool) →' T [] → 'T option Renvoie le premier élément du tableau fourni pour lequel la fonction fournie renvoie true. RetourNone si aucun élément de ce type n'existe.
tryFindIndex: ('T → bool) →' T [] → option int Renvoie l'index du premier élément d'un tableau qui satisfait la condition fournie.
tryPick: (option 'T →' U) → option 'T [] →' U Applique la fonction fournie aux éléments successifs du tableau fourni et renvoie le premier résultat où la fonction renvoie Some (x) pour certains x. Si la fonction ne renvoie jamais Some (x),None est retourné.
décompresser: ('T1 *' T2) [] → 'T1 [] *' T2 [] Divise un tableau de paires de tuple en un tuple de deux tableaux.
unzip3: ('T1 *' T2 * 'T3) [] →' T1 [] * 'T2 [] *' T3 [] Divise un tableau de tuples de trois éléments en un tuple de trois tableaux.
zeroCreate: int → 'T [] Crée un tableau dont les éléments sont initialement définis sur la valeur par défaut Unchecked.defaultof <'T>.
zip: 'T1 [] →' T2 [] → ('T1 *' T2) [] Combine deux tableaux en un tableau de tuples qui ont deux éléments. Les deux tableaux doivent avoir des longueurs égales; sinon, ArgumentException est déclenchée.
zip3: 'T1 [] →' T2 [] → 'T3 [] → (' T1 * 'T2 * 113' T3) [] Combine trois tableaux en un tableau de tuples qui ont trois éléments. Les trois tableaux doivent avoir des longueurs égales; sinon, ArgumentException est déclenchée.

Dans la section suivante, nous verrons les utilisations de certaines de ces fonctionnalités.

Création de tableaux à l'aide de fonctions

Le module Array fournit plusieurs fonctions qui créent un tableau à partir de zéro.

  • le Array.empty crée un nouveau tableau vide.

  • le Array.create La fonction crée un tableau d'une taille spécifiée et définit tous les éléments sur des valeurs données.

  • le Array.init function crée un tableau, avec une dimension et une fonction pour générer les éléments.

  • le Array.zeroCreate La fonction crée un tableau dans lequel tous les éléments sont initialisés à la valeur zéro.

  • le Array.copy crée un nouveau tableau contenant des éléments copiés à partir d'un tableau existant.

  • le Array.sub La fonction génère un nouveau tableau à partir d'une sous-plage d'un tableau.

  • le Array.append crée un nouveau tableau en combinant deux tableaux existants.

  • le Array.choose La fonction sélectionne les éléments d'un tableau à inclure dans un nouveau tableau.

  • le Array.collect function exécute une fonction spécifiée sur chaque élément de tableau d'un tableau existant, puis collecte les éléments générés par la fonction et les combine dans un nouveau tableau.

  • le Array.concat function prend une séquence de tableaux et les combine en un seul tableau.

  • le Array.filter La fonction prend une fonction de condition booléenne et génère un nouveau tableau qui contient uniquement les éléments du tableau d'entrée pour lesquels la condition est vraie.

  • le Array.rev La fonction génère un nouveau tableau en inversant l'ordre d'un tableau existant.

Les exemples suivants illustrent ces fonctions -

Exemple 1

(* using create and set *)
let array1 = Array.create 10 ""
for i in 0 .. array1.Length - 1 do
   Array.set array1 i (i.ToString())
for i in 0 .. array1.Length - 1 do
   printf "%s " (Array.get array1 i)
printfn " "

(* empty array *)
let array2 = Array.empty
printfn "Length of empty array: %d" array2.Length

let array3 = Array.create 10 7.0
printfn "Float Array: %A" array3

(* using the init and zeroCreate *)
let array4 = Array.init 10 (fun index -> index * index)
printfn "Array of squares: %A" array4

let array5 : float array = Array.zeroCreate 10
let (myZeroArray : float array) = Array.zeroCreate 10
printfn "Float Array: %A" array5

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

0 1 2 3 4 5 6 7 8 9
Length of empty array: 0
Float Array: [|7.0; 7.0; 7.0; 7.0; 7.0; 7.0; 7.0; 7.0; 7.0; 7.0|]
Array of squares: [|0; 1; 4; 9; 16; 25; 36; 49; 64; 81|]
Float Array: [|0.0; 0.0; 0.0; 0.0; 0.0; 0.0; 0.0; 0.0; 0.0; 0.0|]

Exemple 2

(* creating subarray from element 5 *)
(* containing 15 elements thereon *)

let array1 = [| 0 .. 50 |]
let array2 = Array.sub array1 5 15
printfn "Sub Array:"
printfn "%A" array2

(* appending two arrays *)
let array3 = [| 1; 2; 3; 4|]
let array4 = [| 5 .. 9 |]
printfn "Appended Array:"
let array5 = Array.append array3 array4
printfn "%A" array5

(* using the Choose function *)
let array6 = [| 1 .. 20 |]
let array7 = Array.choose (fun elem -> if elem % 3 = 0 then
                                             Some(float (elem))
                                          else
                                             None) array6
printfn "Array with Chosen elements:"
printfn "%A" array7

(*using the Collect function *)
let array8 = [| 2 .. 5 |]
let array9 = Array.collect (fun elem -> [| 0 .. elem - 1 |]) array8
printfn "Array with collected elements:"
printfn "%A" array9

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Sub Array:
[|5; 6; 7; 8; 9; 10; 11; 12; 13; 14; 15; 16; 17; 18; 19|]
Appended Array:
[|1; 2; 3; 4; 5; 6; 7; 8; 9|]
Array with Chosen elements:
[|3.0; 6.0; 9.0; 12.0; 15.0; 18.0|]
Array with collected elements:
[|0; 1; 0; 1; 2; 0; 1; 2; 3; 0; 1; 2; 3; 4|]

Recherche de tableaux

le Array.find function prend une fonction booléenne et retourne le premier élément pour lequel la fonction renvoie true, sinon déclenche une KeyNotFoundException.

le Array.findIndex La fonction fonctionne de la même manière sauf qu'elle renvoie l'index de l'élément au lieu de l'élément lui-même.

L'exemple suivant illustre cela.

Microsoft fournit cet exemple de programme intéressant, qui trouve le premier élément dans la plage d'un nombre donné qui est à la fois un carré parfait et un cube parfait -

let array1 = [| 2 .. 100 |]
let delta = 1.0e-10
let isPerfectSquare (x:int) =
   let y = sqrt (float x)
   abs(y - round y) < delta

let isPerfectCube (x:int) =
   let y = System.Math.Pow(float x, 1.0/3.0)
   abs(y - round y) < delta

let element = Array.find (fun elem -> isPerfectSquare elem && isPerfectCube elem) array1

let index = Array.findIndex (fun elem -> isPerfectSquare elem && isPerfectCube elem) array1

printfn "The first element that is both a square and a cube is %d and its index is %d." element index

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

The first element that is both a square and a cube is 64 and its index is 62.

le List<'T> class représente une liste d'objets fortement typés accessibles par index.

C'est un équivalent mutable de la classe List. Il est similaire aux tableaux, car il est accessible par un index, cependant, contrairement aux tableaux, les listes peuvent être redimensionnées. Par conséquent, vous n'avez pas besoin de spécifier une taille lors de la déclaration.

Créer une liste mutable

Les listes sont créées à l'aide du newmot-clé et appelant le constructeur de la liste. L'exemple suivant illustre ceci -

(* Creating a List *)
open System.Collections.Generic

let booksList = new List<string>()
booksList.Add("Gone with the Wind")
booksList.Add("Atlas Shrugged")
booksList.Add("Fountainhead")
booksList.Add("Thornbirds")
booksList.Add("Rebecca")
booksList.Add("Narnia")

booksList |> Seq.iteri (fun index item -> printfn "%i: %s" index booksList.[index])

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

0: Gone with the Wind
1: Atlas Shrugged
2: Fountainhead
3: Thornbirds
4: Rebecca
5: Narnia

La classe List (T)

La classe List (T) représente une liste d'objets fortement typés accessibles par index. Il fournit des méthodes pour rechercher, trier et manipuler des listes.

Les tableaux suivants fournissent les propriétés, les constructeurs et les méthodes de la classe List (T) -

Propriétés

Propriété La description
Capacité Obtient ou définit le nombre total d'éléments que la structure de données interne peut contenir sans redimensionnement.
Compter Obtient le nombre d'éléments contenus dans la liste (T).
Article Obtient ou définit l'élément à l'index spécifié.

Constructeurs

Constructeur La description
Liste (T) () Initialise une nouvelle instance de la classe List (T) qui est vide et a la capacité initiale par défaut.
Liste (T) (IEnumerable (T)) Initialise une nouvelle instance de la classe List (T) qui contient des éléments copiés à partir de la collection spécifiée et a une capacité suffisante pour accueillir le nombre d'éléments copiés.
Liste (T) (Int32) Initialise une nouvelle instance de la classe List (T) qui est vide et a la capacité initiale spécifiée.

Méthode

Méthodes La description
Ajouter Ajoute un objet à la fin de la liste (T).
AddRange Ajoute les éléments de la collection spécifiée à la fin de la liste (T).
AsReadOnly Renvoie un wrapper IList (T) en lecture seule pour la collection actuelle.
Recherche binaire (T) Recherche dans toute la liste triée (T) un élément à l'aide du comparateur par défaut et renvoie l'index de base zéro de l'élément.
Recherche binaire (T, IComparer (T)) Recherche dans toute la liste triée (T) un élément à l'aide du comparateur spécifié et retourne l'index de base zéro de l'élément.
Recherche binaire (Int32, Int32, T, IComparer (T)) Recherche un élément dans une plage d'éléments dans la liste triée (T) à l'aide du comparateur spécifié et retourne l'index de base zéro de l'élément.
Clair Supprime tous les éléments de la liste (T).
Contient Détermine si un élément est dans la liste (T).
ConvertAll (TOutput) Convertit les éléments de la liste actuelle (T) en un autre type et renvoie une liste contenant les éléments convertis.
Copier vers (T []) Copie la liste entière (T) dans un tableau unidimensionnel compatible, en commençant au début du tableau cible.
Copier vers (T [], Int32) Copie la liste entière (T) dans un tableau unidimensionnel compatible, en commençant à l'index spécifié du tableau cible.
CopyTo (Int32, T [], Int32, Int32) Copie une plage d'éléments de la liste (T) dans un tableau unidimensionnel compatible, en commençant à l'index spécifié du tableau cible.
Equals (objet) Détermine si l'objet spécifié est égal à l'objet actuel. (Hérité de Object.)
Existe Détermine si la liste (T) contient des éléments qui correspondent aux conditions définies par le prédicat spécifié.
Finaliser Permet à un objet d'essayer de libérer des ressources et d'effectuer d'autres opérations de nettoyage avant qu'il ne soit récupéré par le garbage collection (hérité de Object).
Trouver Recherche un élément qui correspond aux conditions définies par le prédicat spécifié et renvoie la première occurrence dans toute la liste (T).
Trouver tout Récupère tous les éléments qui correspondent aux conditions définies par le prédicat spécifié.
FindIndex (Prédicat (T)) Recherche un élément qui correspond aux conditions définies par le prédicat spécifié et renvoie l'index de base zéro de la première occurrence dans l'ensemble de List (T).
FindIndex (Int32, Prédicat (T)) Recherche un élément qui correspond aux conditions définies par le prédicat spécifié et retourne l'index de base zéro de la première occurrence dans la plage d'éléments de la liste (T) qui s'étend de l'index spécifié au dernier élément.
FindIndex (Int32, Int32, Prédicat (T)) Recherche un élément qui correspond aux conditions définies par le prédicat spécifié et renvoie l'index de base zéro de la première occurrence dans la plage d'éléments de la liste (T) qui commence à l'index spécifié et contient le nombre d'éléments spécifié.
FindLast Recherche un élément qui correspond aux conditions définies par le prédicat spécifié et renvoie la dernière occurrence dans toute la liste (T).
FindLastIndex (Prédicat (T)) Recherche un élément qui correspond aux conditions définies par le prédicat spécifié et renvoie l'index de base zéro de la dernière occurrence dans l'ensemble de List (T).
FindLastIndex (Int32, Prédicat (T)) Recherche un élément qui correspond aux conditions définies par le prédicat spécifié et retourne l'index de base zéro de la dernière occurrence dans la plage d'éléments de la liste (T) qui s'étend du premier élément à l'index spécifié.
FindLastIndex (Int32, Int32, Prédicat (T)) Recherche un élément qui correspond aux conditions définies par le prédicat spécifié et renvoie l'index de base zéro de la dernière occurrence dans la plage d'éléments de la liste (T) qui contient le nombre d'éléments spécifié et se termine à l'index spécifié.
Pour chaque Exécute l'action spécifiée sur chaque élément de la liste (T).
GetEnumerator Renvoie un énumérateur qui parcourt la liste (T).
GetHashCode Sert de fonction de hachage par défaut. (Hérité de Object.)
GetRange Crée une copie superficielle d'une plage d'éléments dans la liste source (T).
GetType Obtient le Type de l'instance actuelle. (Hérité de Object.)
IndexOf (T) Recherche l'objet spécifié et renvoie l'index de base zéro de la première occurrence dans l'ensemble de List (T).
IndexOf (T, Int32) Recherche l'objet spécifié et retourne l'index de base zéro de la première occurrence dans la plage d'éléments de la liste (T) qui s'étend de l'index spécifié au dernier élément.
IndexOf (T, Int32, Int32) Recherche l'objet spécifié et retourne l'index de base zéro de la première occurrence dans la plage d'éléments de la liste (T) qui commence à l'index spécifié et contient le nombre d'éléments spécifié.
Insérer Insère un élément dans la liste (T) à l'index spécifié.
InsertRange Insère les éléments d'une collection dans la liste (T) à l'index spécifié.
LastIndexOf (T) Recherche l'objet spécifié et renvoie l'index de base zéro de la dernière occurrence dans l'ensemble de la liste (T).
LastIndexOf (T, Int32) Recherche l'objet spécifié et retourne l'index de base zéro de la dernière occurrence dans la plage d'éléments de la liste (T) qui s'étend du premier élément à l'index spécifié.
LastIndexOf (T, Int32, Int32) Recherche l'objet spécifié et retourne l'index de base zéro de la dernière occurrence dans la plage d'éléments de la liste (T) qui contient le nombre d'éléments spécifié et se termine à l'index spécifié.
MemberwiseClone Crée une copie superficielle de l'objet actuel. (Hérité de Object.)
Retirer Supprime la première occurrence d'un objet spécifique de la liste (T).
Enlever tout Supprime tous les éléments qui correspondent aux conditions définies par le prédicat spécifié.
RemoveAt Supprime l'élément à l'index spécifié de la liste (T).
RemoveRange Supprime une plage d'éléments de la liste (T).
Inverser() Inverse l'ordre des éléments dans toute la liste (T).
Inverser (Int32, Int32) Inverse l'ordre des éléments dans la plage spécifiée.
Trier() Trie les éléments de la liste entière (T) à l'aide du comparateur par défaut.
Trier (Comparaison (T)) Trie les éléments de la liste entière (T) à l'aide du système spécifié. Comparaison (T).
Trier (IComparer (T)) Trie les éléments de la liste entière (T) à l'aide du comparateur spécifié.
Trier (Int32, Int32, IComparer (T)) Trie les éléments d'une plage d'éléments dans List (T) à l'aide du comparateur spécifié.
ToArray Copie les éléments de la liste (T) dans un nouveau tableau.
ToString Retourne une chaîne qui représente l'objet actuel. (Hérité de Object.)
TrimExcess Définit la capacité sur le nombre réel d'éléments dans la liste (T), si ce nombre est inférieur à une valeur seuil.
TrueForAll Détermine si chaque élément de la liste (T) correspond aux conditions définies par le prédicat spécifié.

Exemple

(* Creating a List *)
open System.Collections.Generic

let booksList = new List<string>()
booksList.Add("Gone with the Wind")
booksList.Add("Atlas Shrugged")
booksList.Add("Fountainhead")
booksList.Add("Thornbirds")
booksList.Add("Rebecca")
booksList.Add("Narnia")

printfn"Total %d books" booksList.Count
booksList |> Seq.iteri (fun index item -> printfn "%i: %s" index booksList.[index])
booksList.Insert(2, "Roots")

printfn("after inserting at index 2")
printfn"Total %d books" booksList.Count

booksList |> Seq.iteri (fun index item -> printfn "%i: %s" index booksList.[index])
booksList.RemoveAt(3)

printfn("after removing from index 3")
printfn"Total %d books" booksList.Count

booksList |> Seq.iteri (fun index item -> printfn "%i: %s" index booksList.[index])

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Total 6 books
0: Gone with the Wind
1: Atlas Shrugged
2: Fountainhead
3: Thornbirds
4: Rebecca
5: Narnia
after inserting at index 2
Total 7 books
0: Gone with the Wind
1: Atlas Shrugged
2: Roots
3: Fountainhead
4: Thornbirds
5: Rebecca
6: Narnia
after removing from index 3
Total 6 books
0: Gone with the Wind
1: Atlas Shrugged
2: Roots
3: Thornbirds
4: Rebecca
5: Narnia

le Dictionary<'TKey, 'TValue> class est l'analogue mutable de la structure de données de la carte F # et contient plusieurs des mêmes fonctions.

Récapitulant à partir du chapitre Carte en F #, une carte est un type spécial d'ensemble qui associe les valeurs à la clé.

Création d'un dictionnaire mutable

Les dictionnaires mutables sont créés à l'aide du newmot-clé et appelant le constructeur de la liste. L'exemple suivant illustre ceci -

open System.Collections.Generic
let dict = new Dictionary<string, string>()
dict.Add("1501", "Zara Ali")
dict.Add("1502","Rishita Gupta")
dict.Add("1503","Robin Sahoo")
dict.Add("1504","Gillian Megan")
printfn "Dictionary - students: %A" dict

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Dictionary - students: seq
[[1501, Zara Ali]; [1502, Rishita Gupta]; [1503, Robin Sahoo];
[1504, Gillian Megan]]

La classe Dictionnaire (TKey, TValue)

La classe Dictionary (TKey, TValue) représente une collection de clés et de valeurs.

Les tableaux suivants fournissent les propriétés, les constructeurs et les méthodes de la classe List (T) -

Propriétés

Propriété La description
Comparer Obtient le IEqualityComparer (T) qui est utilisé pour déterminer l'égalité des clés pour le dictionnaire.
Compter Obtient le nombre de paires clé / valeur contenues dans le dictionnaire (TKey, TValue).
Article Obtient ou définit la valeur associée à la clé spécifiée.
Clés Obtient une collection contenant les clés du dictionnaire (TKey, TValue).
Valeurs Obtient une collection contenant les valeurs du Dictionary (TKey, TValue).

Constructeurs

Constructeurs La description
Dictionnaire (TKey, TValue) () Initialise une nouvelle instance du Dictionary(TKey, TValue) classe qui est vide, a la capacité initiale par défaut et utilise le comparateur d'égalité par défaut pour le type de clé.
Dictionnaire (TKey, TValue) (IDictionary (TKey, TValue)) Initialise une nouvelle instance du Dictionary(TKey, TValue) classe qui contient des éléments copiés à partir du spécifié IDictionary(TKey, TValue) et utilise le comparateur d'égalité par défaut pour le type de clé.
Dictionnaire (TKey, TValue) (IEqualityComparer (TKey)) Initialise une nouvelle instance du Dictionary(TKey, TValue) classe qui est vide, a la capacité initiale par défaut et utilise le IEqualityComparer(T).
Dictionnaire (TKey, TValue) (Int32) Initialise une nouvelle instance du Dictionary(TKey, TValue) classe qui est vide, a la capacité initiale spécifiée et utilise le comparateur d'égalité par défaut pour le type de clé.
Dictionnaire (TKey, TValue) (IDictionary (TKey, TValue), IEqualityComparer (TKey)) Initialise une nouvelle instance du Dictionary(TKey, TValue) classe qui contient des éléments copiés à partir du spécifié IDictionary(TKey, TValue) et utilise le spécifié IEqualityComparer(T).
Dictionnaire (TKey, TValue) (Int32, IEqualityComparer (TKey)) Initialise une nouvelle instance du Dictionary(TKey, TValue) classe qui est vide, a la capacité initiale spécifiée et utilise le IEqualityComparer(T).
Dictionnaire (TKey, TValue) (SerializationInfo, StreamingContext) Initialise une nouvelle instance du ictionary(TKey, TValue) classe avec des données sérialisées.

Méthodes

Méthode La description
Ajouter Ajoute la clé et la valeur spécifiées au dictionnaire.
Clair Supprime toutes les clés et valeurs du dictionnaire (TKey, TValue).
Contient la clé Détermine si le dictionnaire (TKey, TValue) contient la clé spécifiée.
ContientValeur Détermine si le dictionnaire (TKey, TValue) contient une valeur spécifique.
Equals (objet) Détermine si l'objet spécifié est égal à l'objet actuel. (Hérité de Object.)
Finaliser Permet à un objet d'essayer de libérer des ressources et d'effectuer d'autres opérations de nettoyage avant qu'il ne soit récupéré par le garbage collection. (Hérité de Object.)
GetEnumerator Renvoie un énumérateur qui itère dans le dictionnaire (TKey, TValue).
GetHashCode Sert de fonction de hachage par défaut. (Hérité de Object.)
GetObjectData Implémente l'interface System.Runtime.Serialization.ISerializable et retourne les données nécessaires pour sérialiser l'instance Dictionary (TKey, TValue).
GetType Obtient le Type de l'instance actuelle. (Hérité de Object.)
MemberwiseClone Crée une copie superficielle de l'objet actuel. (Hérité de Object.)
Désérialisation Implémente l'interface System.Runtime.Serialization.ISerializable et déclenche l'événement de désérialisation lorsque la désérialisation est terminée.
Retirer Supprime la valeur avec la clé spécifiée du dictionnaire (TKey, TValue).
ToString Retourne une chaîne qui représente l'objet actuel. (Hérité de Object.)
TryGetValue Obtient la valeur associée à la clé spécifiée.

Exemple

open System.Collections.Generic
let dict = new Dictionary<string, string>()

dict.Add("1501", "Zara Ali")
dict.Add("1502","Rishita Gupta")
dict.Add("1503","Robin Sahoo")
dict.Add("1504","Gillian Megan")

printfn "Dictionary - students: %A" dict
printfn "Total Number of Students: %d" dict.Count
printfn "The keys: %A" dict.Keys
printf"The Values: %A" dict.Values

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Dictionary - students: seq
[[1501, Zara Ali]; [1502, Rishita Gupta]; [1503, Robin Sahoo];
[1504, Gillian Megan]]
Total Number of Students: 4
The keys: seq ["1501"; "1502"; "1503"; "1504"]
The Values: seq ["Zara Ali"; "Rishita Gupta"; "Robin Sahoo"; "Gillian Megan"]

Entrée de base La sortie comprend -

  • Lecture et écriture dans la console.
  • Lecture et écriture dans un fichier.

Module Core.Printf

Nous avons utilisé les fonctions printf et printfn pour écrire dans la console. Dans cette section, nous examinerons les détails de laPrintf module de F #.

Outre les fonctions ci-dessus, le module Core.Printf de F # dispose de diverses autres méthodes d'impression et de formatage en utilisant des marqueurs% comme espaces réservés. Le tableau suivant montre les méthodes avec une brève description -

Valeur La description
bprintf: StringBuilder → BuilderFormat <'T> →' T Imprime sur un StringBuilder.
eprintf: TextWriterFormat <'T> →' T Imprime la sortie formatée sur stderr.
eprintfn: TextWriterFormat <'T> →' T Imprime la sortie formatée sur stderr, en ajoutant une nouvelle ligne.
failwithf: StringFormat <'T,' Résultat> → 'T Imprime dans un tampon de chaîne et déclenche une exception avec le résultat donné.
fprintf: TextWriter → TextWriterFormat <'T> →' T Imprime à un rédacteur de texte.
fprintfn: TextWriter → TextWriterFormat <'T> →' T Imprime sur un rédacteur de texte, en ajoutant une nouvelle ligne.
kbprintf: (unité → 'Résultat) → StringBuilder → BuilderFormat <' T, 'Résultat> →' T Comme bprintf, mais appelle la fonction spécifiée pour générer le résultat.
kfprintf: (unité → 'Résultat) → TextWriter → TextWriterFormat <' T, 'Résultat> →' T Comme fprintf, mais appelle la fonction spécifiée pour générer le résultat.
kprintf: (chaîne → 'Résultat) → StringFormat <' T, 'Résultat> →' T Comme printf, mais appelle la fonction spécifiée pour générer le résultat. Par exemple, ceux-ci laissent la force d'impression un flush après que toutes les sorties ont été entrées sur le canal, mais pas avant.
ksprintf: (chaîne → 'Résultat) → StringFormat <' T, 'Résultat> →' T Comme sprintf, mais appelle la fonction spécifiée pour générer le résultat.
printf: TextWriterFormat <'T> →' T Imprime la sortie formatée sur stdout.
printfn: TextWriterFormat <'T> →' T Imprime la sortie formatée sur stdout, en ajoutant une nouvelle ligne.
sprintf: StringFormat <'T> →' T Imprime dans une chaîne à l'aide d'un tampon de chaîne interne et renvoie le résultat sous forme de chaîne.

Spécifications de format

Les spécifications de format sont utilisées pour formater l'entrée ou la sortie, selon les besoins des programmeurs.

Ce sont des chaînes avec des marqueurs% indiquant les espaces réservés de format.

La syntaxe d'un espace réservé de format est -

%[flags][width][.precision][type]

le type est interprété comme -

Type La description
% b Formats un bool, formaté comme true ou false.
% c Formate un caractère.
% s Formats un string, formaté comme son contenu, sans interpréter les caractères d'échappement.
% d,% i Formate tout type entier de base mis en forme comme un entier décimal, signé si le type entier de base est signé.
% u Formate tout type d'entier de base mis en forme comme un entier décimal non signé.
%X Formate tout type d'entier de base mis en forme comme un entier hexadécimal non signé, en utilisant les lettres minuscules de a à f.
%X Formate tout type d'entier de base mis en forme comme un entier hexadécimal non signé, en utilisant les lettres majuscules A à F.
% o Formate tout type d'entier de base formaté comme un entier octal non signé.
% e,% E,% f,% F,% g,% G Formate tout type de base à virgule flottante (float, float32) formaté en utilisant des spécifications de format à virgule flottante de style C.
% e,% E Formate une valeur signée de la forme [-] d.dddde [signe] ddd où d est un seul chiffre décimal, dddd est un ou plusieurs chiffres décimaux, ddd est exactement trois chiffres décimaux et le signe est + ou -.
%F Formate une valeur signée de la forme [-] dddd.dddd, où dddd est un ou plusieurs chiffres décimaux. Le nombre de chiffres avant la virgule décimale dépend de la magnitude du nombre et le nombre de chiffres après la virgule décimale dépend de la précision demandée.
% g,% G Formate une valeur signée imprimée au format f ou e, selon ce qui est le plus compact pour la valeur et la précision données.
% M Formate une valeur décimale.
% O Formate toute valeur, imprimée en encadrant l'objet et en utilisant son ToString méthode.
% A,% + A Formate toute valeur, imprimée avec les paramètres de mise en page par défaut. Utilisez% + A pour imprimer la structure des syndicats discriminés avec des représentations internes et privées.
%une

Un spécificateur de format général nécessite deux arguments. Le premier argument est une fonction qui accepte deux arguments: d'abord, un paramètre de contexte du type approprié pour la fonction de mise en forme donnée (par exemple, un TextWriter), et deuxièmement, une valeur à imprimer et qui produit ou renvoie le texte approprié.

Le deuxième argument est la valeur particulière à imprimer.

% t Un spécificateur de format général nécessite un argument: une fonction qui accepte un paramètre de contexte du type approprié pour la fonction de mise en forme donnée (aTextWriter) et qui génère ou renvoie le texte approprié. Les types entiers de base sontbyte, sbyte, int16, uint16, int32, uint32, int64, uint64, nativeint, et unativeint. Les types à virgule flottante de base sont float et float32.

le widthest un paramètre facultatif. C'est un entier qui indique la largeur minimale du résultat. Par exemple,% 5d imprime un entier avec au moins des espaces de 5 caractères.

Valide flags sont décrits dans le tableau suivant -

Valeur La description
0 Spécifie d'ajouter des zéros au lieu d'espaces pour créer la largeur requise.
- Spécifie de justifier à gauche le résultat dans la largeur spécifiée.
+ Spécifie d'ajouter un caractère + si le nombre est positif (pour correspondre au signe - pour les nombres négatifs).
' ' (espace) Spécifie d'ajouter un espace supplémentaire si le nombre est positif (pour correspondre au signe - pour les nombres négatifs).
# Invalide.

Exemple

printf "Hello "
printf "World"
printfn ""
printfn "Hello "
printfn "World"
printf "Hi, I'm %s and I'm a %s" "Rohit" "Medical Student"

printfn "d: %f" 212.098f
printfn "e: %f" 504.768f

printfn "x: %g" 212.098f
printfn "y: %g" 504.768f

printfn "x: %e" 212.098f
printfn "y: %e" 504.768f
printfn "True: %b" true

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Hello World
Hello
World
Hi, I'm Rohit and I'm a Medical Studentd: 212.098000
e: 504.768000
x: 212.098
y: 504.768
x: 2.120980e+002
y: 5.047680e+002
True: true

La classe console

Cette classe fait partie du framework .NET. Il représente les flux d'entrée, de sortie et d'erreur standard pour les applications de console.

Il fournit diverses méthodes de lecture et d'écriture dans la console. Le tableau suivant montre les méthodes -

Méthode La description
Bip() Joue le son d'un bip via le haut-parleur de la console.
Bip (Int32, Int32) Reproduit le son d'un bip d'une fréquence et d'une durée spécifiées via le haut-parleur de la console.
Clair Efface la mémoire tampon de la console et la fenêtre de console correspondante des informations d'affichage.
MoveBufferArea (Int32, Int32, Int32, Int32, Int32, Int32) Copie une zone source spécifiée de la mémoire tampon d'écran vers une zone de destination spécifiée.
MoveBufferArea (Int32, Int32, Int32, Int32, Int32, Int32, Char, ConsoleColor, ConsoleColor) Copie une zone source spécifiée de la mémoire tampon d'écran vers une zone de destination spécifiée.
OpenStandardError () Acquiert le flux d'erreur standard.
OpenStandardError (Int32) Acquiert le flux d'erreur standard, qui est défini sur une taille de tampon spécifiée.
OpenStandardInput () Acquiert le flux d'entrée standard.
OpenStandardInput (Int32) Acquiert le flux d'entrée standard, qui est défini sur une taille de tampon spécifiée.
OpenStandardOutput () Acquiert le flux de sortie standard.
OpenStandardOutput (Int32) Acquiert le flux de sortie standard, qui est défini sur une taille de tampon spécifiée.
Lis Lit le caractère suivant du flux d'entrée standard.
ReadKey () Obtient le caractère suivant ou la touche de fonction pressée par l'utilisateur. La touche enfoncée s'affiche dans la fenêtre de la console.
ReadKey (booléen) Obtient le caractère suivant ou la touche de fonction pressée par l'utilisateur. La touche enfoncée est éventuellement affichée dans la fenêtre de la console.
ReadLine Lit la ligne suivante de caractères du flux d'entrée standard.
Réinitialiser la couleur Définit les couleurs de la console de premier plan et d'arrière-plan sur leurs valeurs par défaut.
SetBufferSize Définit la hauteur et la largeur de la zone tampon de l'écran sur les valeurs spécifiées.
SetCursorPosition Définit la position du curseur.
SetError Définit la propriété Error sur l' objet TextWriter spécifié .
S'installer Définit la propriété In sur l' objet TextReader spécifié .
Partir planifier Définit la propriété Out sur l' objet TextWriter spécifié .
SetWindowPosition Définit la position de la fenêtre de la console par rapport au tampon d'écran.
SetWindowSize Définit la hauteur et la largeur de la fenêtre de la console sur les valeurs spécifiées.
Écrire (booléen) Écrit la représentation textuelle de la valeur booléenne spécifiée dans le flux de sortie standard.
Ecrire (Char) Écrit la valeur de caractère Unicode spécifiée dans le flux de sortie standard.
Écrire (Char []) Écrit le tableau spécifié de caractères Unicode dans le flux de sortie standard.
Ecrire (décimal) Écrit la représentation textuelle de la valeur Decimal spécifiée dans le flux de sortie standard.
Écrire (double) Écrit la représentation textuelle de la valeur à virgule flottante double précision spécifiée dans le flux de sortie standard.
Ecrire (Int32) Écrit la représentation textuelle de la valeur entière signée 32 bits spécifiée dans le flux de sortie standard.
Ecrire (Int64) Écrit la représentation textuelle de la valeur entière signée 64 bits spécifiée dans le flux de sortie standard.
Ecrire (objet) Écrit la représentation textuelle de l'objet spécifié dans le flux de sortie standard.
Ecrire (unique) Écrit la représentation textuelle de la valeur à virgule flottante simple précision spécifiée dans le flux de sortie standard.
Écrire (chaîne) Écrit la valeur de chaîne spécifiée dans le flux de sortie standard.
Ecrire (UInt32) Écrit la représentation textuelle de la valeur entière non signée 32 bits spécifiée dans le flux de sortie standard.
Ecrire (UInt64) Écrit la représentation textuelle de la valeur entière non signée 64 bits spécifiée dans le flux de sortie standard.
Écrire (chaîne, objet) Écrit la représentation textuelle de l'objet spécifié dans le flux de sortie standard à l'aide des informations de format spécifiées.
Écriture (chaîne, objet []) Écrit la représentation textuelle du tableau d'objets spécifié dans le flux de sortie standard à l'aide des informations de format spécifiées.
Écriture (Char [], Int32, Int32) Écrit le sous-tableau spécifié de caractères Unicode dans le flux de sortie standard.
Écrire (chaîne, objet, objet) Écrit la représentation textuelle des objets spécifiés dans le flux de sortie standard à l'aide des informations de format spécifiées.
Écriture (chaîne, objet, objet, objet) Écrit la représentation textuelle des objets spécifiés dans le flux de sortie standard à l'aide des informations de format spécifiées.
Écriture (chaîne, objet, objet, objet, objet) Écrit la représentation textuelle des objets spécifiés et de la liste de paramètres de longueur variable dans le flux de sortie standard à l'aide des informations de format spécifiées.
WriteLine () Écrit le terminateur de ligne actuel dans le flux de sortie standard.
WriteLine (booléen) Écrit la représentation textuelle de la valeur booléenne spécifiée, suivie de la terminaison de ligne actuelle, dans le flux de sortie standard.
WriteLine (Char) Écrit la valeur du caractère Unicode spécifié, suivi de la fin de ligne actuelle, dans le flux de sortie standard.
WriteLine (Char []) Écrit le tableau spécifié de caractères Unicode, suivi du terminateur de ligne actuel, dans le flux de sortie standard.
WriteLine (décimal) Écrit la représentation textuelle de la valeur Decimal spécifiée, suivie du terminateur de ligne actuel, dans le flux de sortie standard.
WriteLine (Double) Écrit la représentation textuelle de la valeur à virgule flottante double précision spécifiée, suivie de la terminaison de ligne actuelle, dans le flux de sortie standard.
WriteLine (Int32) Écrit la représentation textuelle de la valeur entière signée 32 bits spécifiée, suivie de la terminaison de ligne actuelle, dans le flux de sortie standard.
WriteLine (Int64) Écrit la représentation textuelle de la valeur entière signée 64 bits spécifiée, suivie de la terminaison de ligne actuelle, dans le flux de sortie standard.
WriteLine (objet) Écrit la représentation textuelle de l'objet spécifié, suivie de la terminaison de ligne actuelle, dans le flux de sortie standard.
WriteLine (unique) Écrit la représentation textuelle de la valeur à virgule flottante simple précision spécifiée, suivie de la terminaison de ligne actuelle, dans le flux de sortie standard.
WriteLine (chaîne) Écrit la valeur de chaîne spécifiée, suivie du terminateur de ligne actuel, dans le flux de sortie standard.
WriteLine (UInt32) Écrit la représentation textuelle de la valeur entière non signée 32 bits spécifiée, suivie de la terminaison de ligne actuelle, dans le flux de sortie standard.
WriteLine (UInt64) Écrit la représentation textuelle de la valeur entière non signée 64 bits spécifiée, suivie de la terminaison de ligne actuelle, dans le flux de sortie standard.
WriteLine (chaîne, objet) Écrit la représentation textuelle de l'objet spécifié, suivie de la terminaison de ligne actuelle, dans le flux de sortie standard à l'aide des informations de format spécifiées.
WriteLine (chaîne, objet []) Écrit la représentation textuelle du tableau d'objets spécifié, suivie de la terminaison de ligne actuelle, dans le flux de sortie standard à l'aide des informations de format spécifiées.
WriteLine (Char [], Int32, Int32) Écrit le sous-tableau spécifié de caractères Unicode, suivi de la terminaison de ligne actuelle, dans le flux de sortie standard.
WriteLine (chaîne, objet, objet) Écrit la représentation textuelle des objets spécifiés, suivie de la terminaison de ligne actuelle, dans le flux de sortie standard à l'aide des informations de format spécifiées.
WriteLine (chaîne, objet, objet, objet) Écrit la représentation textuelle des objets spécifiés, suivie de la terminaison de ligne actuelle, dans le flux de sortie standard à l'aide des informations de format spécifiées.
WriteLine (chaîne, objet, objet, objet, objet) Écrit la représentation textuelle des objets spécifiés et de la liste de paramètres de longueur variable, suivie de la terminaison de ligne actuelle, dans le flux de sortie standard à l'aide des informations de format spécifiées.

L'exemple suivant montre comment lire depuis la console et y écrire -

Exemple

open System
let main() =
   Console.Write("What's your name? ")
   let name = Console.ReadLine()
   Console.Write("Hello, {0}\n", name)
   Console.WriteLine(System.String.Format("Big Greetings from {0} and {1}", "TutorialsPoint", "Absoulte Classes"))
   Console.WriteLine(System.String.Format("|{0:yyyy-MMM-dd}|", System.DateTime.Now))
main()

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

What's your name? Kabir
Hello, Kabir
Big Greetings from TutorialsPoint and Absoulte Classes
|2015-Jan-05|

L'espace de noms System.IO

L'espace de noms System.IO contient une variété de classes utiles pour effectuer des E / S de base.

Il contient des types ou classes qui permettent la lecture et l'écriture dans des fichiers et des flux de données et des types qui fournissent une prise en charge de base des fichiers et des répertoires.

Classes utiles pour travailler avec le système de fichiers -

  • La classe System.IO.File est utilisée pour créer, ajouter et supprimer des fichiers.
  • La classe System.IO.Directory est utilisée pour créer, déplacer et supprimer des répertoires.
  • La classe System.IO.Path effectue des opérations sur les chaînes, qui représentent les chemins d'accès aux fichiers.
  • La classe System.IO.FileSystemWatcher permet aux utilisateurs d'écouter un répertoire pour les modifications.

Classes utiles pour travailler avec les flux (séquence d'octets) -

  • La classe System.IO.StreamReader est utilisée pour lire les caractères d'un flux.
  • La classe System.IO.StreamWriter est utilisée pour écrire des caractères dans un flux.
  • La classe System.IO.MemoryStream crée un flux d'octets en mémoire.

Le tableau suivant montre toutes les classes fournies dans l'espace de noms avec une brève description -

Classe La description
BinaryReader Lit les types de données primitifs sous forme de valeurs binaires dans un codage spécifique.
BinaryWriter Écrit des types primitifs en binaire dans un flux et prend en charge l'écriture de chaînes dans un codage spécifique.
BufferedStream Ajoute une couche tampon pour lire et écrire des opérations sur un autre flux.
Annuaire Expose des méthodes statiques pour créer, déplacer et énumérer dans les répertoires et sous-répertoires.
DirectoryInfo Expose des méthodes d'instance pour créer, déplacer et énumérer dans les répertoires et sous-répertoires.
DirectoryNotFoundException L'exception qui est levée lorsqu'une partie d'un fichier ou d'un répertoire est introuvable.
DriveInfo Donne accès aux informations sur un lecteur.
DriveNotFoundException Exception levée lors de la tentative d'accès à un lecteur ou à un partage qui n'est pas disponible.
EndOfStreamException L'exception qui est levée lors de la lecture est tentée après la fin d'un flux.
ErrorEventArgs Fournit des données pour l'événement FileSystemWatcher.Error.
Fichier Fournit des méthodes statiques pour la création, la copie, la suppression, le déplacement et l'ouverture d'un seul fichier, et facilite la création d'objets FileStream.
FileFormatException L'exception qui est levée lorsqu'un fichier d'entrée ou un flux de données censé se conformer à une certaine spécification de format de fichier est mal formé.
FileInfo Fournit des propriétés et des méthodes d'instance pour la création, la copie, la suppression, le déplacement et l'ouverture de fichiers et facilite la création d'objets FileStream.
FileLoadException Exception levée lorsqu'un assembly géré est trouvé mais ne peut pas être chargé.
FileNotFoundException Exception levée lorsqu'une tentative d'accès à un fichier qui n'existe pas sur le disque échoue.
FileStream Expose un Stream autour d'un fichier, prenant en charge les opérations de lecture et d'écriture synchrones et asynchrones.
FileSystemEventArgs Fournit des données pour les événements de l'annuaire - Modifié, créé, supprimé.
FileSystemInfo Fournit la classe de base pour les objets FileInfo et DirectoryInfo.
FileSystemWatcher Écoute les notifications de modification du système de fichiers et déclenche des événements lorsqu'un répertoire ou un fichier dans un répertoire change.
InternalBufferOverflowException L'exception levée lorsque le tampon interne déborde.
InvalidDataException Exception levée lorsqu'un flux de données est dans un format non valide.
IODescriptionAttribute Définit la description que les concepteurs visuels peuvent afficher lorsqu'ils référencent un événement, une extension ou une propriété.
IOException L'exception qui est levée lorsqu'une erreur d'E / S se produit.
MemoryStream Crée un flux dont le stockage de sauvegarde est la mémoire.
Chemin Effectue des opérations sur les instances String qui contiennent des informations de chemin de fichier ou de répertoire. Ces opérations sont effectuées de manière multiplateforme.
PathTooLongException Exception levée lorsqu'un chemin d'accès ou un nom de fichier est plus long que la longueur maximale définie par le système.
PipeException Lancé lorsqu'une erreur se produit dans un canal nommé.
RenamedEventArgs Fournit des données pour l'événement Renamed.
Courant Fournit une vue générique d'une séquence d'octets. Ceci est une classe abstraite.
StreamReader Implémente un TextReader qui lit les caractères d'un flux d'octets dans un codage particulier.
StreamWriter Implémente un TextWriter pour écrire des caractères dans un flux dans un encodage particulier. Pour parcourir le code source du .NET Framework pour ce type, consultez la source de référence.
StringReader Implémente un TextReader qui lit à partir d'une chaîne.
StringWriter Implémente un TextWriter pour écrire des informations dans une chaîne. Les informations sont stockées dans un StringBuilder sous-jacent.
TextReader Représente un lecteur capable de lire une série séquentielle de caractères.
TextWriter Représente un rédacteur capable d'écrire une série séquentielle de caractères. Cette classe est abstraite.
UnmanagedMemoryAccessor Fournit un accès aléatoire aux blocs de mémoire non gérés à partir du code géré.
UnmanagedMemoryStream Fournit l'accès aux blocs de mémoire non gérés à partir du code managé.
WindowsRuntimeStorageExtensions Contient des méthodes d'extension pour les interfaces IStorageFile et IStorageFolder dans Windows Runtime lors du développement d'applications Windows Store.
WindowsRuntimeStreamExtensions Contient des méthodes d'extension pour la conversion entre les flux dans Windows Runtime et les flux gérés dans les applications .NET pour Windows Store.

Exemple

L'exemple suivant crée un fichier appelé test.txt, y écrit un message, lit le texte du fichier et l'imprime sur la console.

Note - La quantité de code nécessaire pour ce faire est étonnamment moindre!

open System.IO // Name spaces can be opened just as modules
File.WriteAllText("test.txt", "Hello There\n Welcome to:\n Tutorials Point")
let msg = File.ReadAllText("test.txt")
printfn "%s" msg

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Hello There
Welcome to:
Tutorials Point

Les génériques vous permettent de retarder la spécification du type de données des éléments de programmation dans une classe ou une méthode, jusqu'à ce qu'il soit réellement utilisé dans le programme. En d'autres termes, les génériques vous permettent d'écrire une classe ou une méthode qui peut fonctionner avec n'importe quel type de données.

Vous écrivez les spécifications de la classe ou de la méthode, avec des paramètres de substitution pour les types de données. Lorsque le compilateur rencontre un constructeur pour la classe ou un appel de fonction pour la méthode, il génère du code pour gérer le type de données spécifique.

En F #, les valeurs de fonction, les méthodes, les propriétés et les types d'agrégats tels que les classes, les enregistrements et les unions discriminées peuvent être génériques.

Les constructions génériques contiennent au moins un paramètre de type. Les fonctions et types génériques vous permettent d'écrire du code qui fonctionne avec une variété de types sans répéter le code pour chaque type.

Syntaxe

La syntaxe pour écrire une construction générique est la suivante -

// Explicitly generic function.
let function-name<type-parameters> parameter-list =
   function-body

// Explicitly generic method.
[ static ] member object-identifer.method-name<type-parameters> parameter-list [ return-type ] =
   method-body

// Explicitly generic class, record, interface, structure,
// or discriminated union.
type type-name<type-parameters> type-definition

Exemples

(* Generic Function *)
let printFunc<'T> x y =
   printfn "%A, %A" x y

printFunc<float> 10.0 20.0

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

10.0, 20.0

Vous pouvez également rendre une fonction générique en utilisant la syntaxe des guillemets simples -

(* Generic Function *)
let printFunction (x: 'a) (y: 'a) =
   printfn "%A %A" x y

printFunction 10.0 20.0

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

10.0 20.0

Veuillez noter que lorsque vous utilisez des fonctions ou des méthodes génériques, vous n'aurez peut-être pas à spécifier les arguments de type. Cependant, en cas d'ambiguïté, vous pouvez fournir des arguments de type entre crochets comme nous l'avons fait dans le premier exemple.

Si vous avez plus d'un type, vous séparez les arguments de type par des virgules.

Classe générique

Comme les fonctions génériques, vous pouvez également écrire des classes génériques. L'exemple suivant illustre ceci -

type genericClass<'a> (x: 'a) =
   do printfn "%A" x

let gr = new genericClass<string>("zara")
let gs = genericClass( seq { for i in 1 .. 10 -> (i, i*i) } )

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

"zara"
seq [(1, 1); (2, 4); (3, 9); (4, 16); ...]

Un délégué est une variable de type référence qui contient la référence à une méthode. La référence peut être modifiée lors de l'exécution. Les délégués F # sont similaires aux pointeurs vers des fonctions, en C ou C ++.

Déclarer les délégués

La déclaration de délégué détermine les méthodes qui peuvent être référencées par le délégué. Un délégué peut faire référence à une méthode, qui a la même signature que celle du délégué.

La syntaxe de la déclaration de délégué est -

type delegate-typename = delegate of type1 -> type2

Par exemple, considérez les délégués -

// Delegate1 works with tuple arguments.
type Delegate1 = delegate of (int * int) -> int
// Delegate2 works with curried arguments.
type Delegate2 = delegate of int * int -> int

Les deux délégués peuvent être utilisés pour référencer n'importe quelle méthode qui a deux paramètres int et retourne une variable de type int .

Dans la syntaxe -

  • type1 représente le (s) type (s) d'argument.

  • type2 représente le type de retour.

Veuillez noter -

  • Les types d'arguments sont automatiquement curry.

  • Les délégués peuvent être attachés aux valeurs de fonction et aux méthodes statiques ou d'instance.

  • Les valeurs de la fonction F # peuvent être transmises directement en tant qu'arguments aux constructeurs délégués.

  • Pour une méthode statique, le délégué est appelé à l'aide du nom de la classe et de la méthode. Pour une méthode d'instance, le nom de l'instance d'objet et de la méthode est utilisé.

  • La méthode Invoke sur le type délégué appelle la fonction encapsulée.

  • En outre, les délégués peuvent être passés en tant que valeurs de fonction en référençant le nom de la méthode Invoke sans les parenthèses.

L'exemple suivant illustre le concept -

Exemple

type Myclass() =
   static member add(a : int, b : int) =
      a + b
   static member sub (a : int) (b : int) =
      a - b
   member x.Add(a : int, b : int) =
      a + b
   member x.Sub(a : int) (b : int) =
      a - b

// Delegate1 works with tuple arguments.
type Delegate1 = delegate of (int * int) -> int
// Delegate2 works with curried arguments.
type Delegate2 = delegate of int * int -> int

let InvokeDelegate1 (dlg : Delegate1) (a : int) (b: int) =
   dlg.Invoke(a, b)
let InvokeDelegate2 (dlg : Delegate2) (a : int) (b: int) =
   dlg.Invoke(a, b)

// For static methods, use the class name, the dot operator, and the
// name of the static method.
let del1 : Delegate1 = new Delegate1( Myclass.add )
let del2 : Delegate2 = new Delegate2( Myclass.sub )

let mc = Myclass()
// For instance methods, use the instance value name, the dot operator, and the instance method name.

let del3 : Delegate1 = new Delegate1( mc.Add )
let del4 : Delegate2 = new Delegate2( mc.Sub )

for (a, b) in [ (400, 200); (100, 45) ] do
   printfn "%d + %d = %d" a b (InvokeDelegate1 del1 a b)
   printfn "%d - %d = %d" a b (InvokeDelegate2 del2 a b)
   printfn "%d + %d = %d" a b (InvokeDelegate1 del3 a b)
   printfn "%d - %d = %d" a b (InvokeDelegate2 del4 a b)

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

400 + 200 = 600
400 - 200 = 200
400 + 200 = 600
400 - 200 = 200
100 + 45 = 145
100 - 45 = 55
100 + 45 = 145
100 - 45 = 55

Une énumération est un ensemble de constantes entières nommées.

En fa #, enumerations, aussi connu sous le nom enums,sont des types intégraux où les étiquettes sont affectées à un sous-ensemble des valeurs. Vous pouvez les utiliser à la place de littéraux pour rendre le code plus lisible et maintenable.

Déclaration d'énumérations

La syntaxe générale pour déclarer une énumération est -

type enum-name =
   | value1 = integer-literal1
   | value2 = integer-literal2
...

L'exemple suivant montre l'utilisation d'énumérations -

Exemple

// Declaration of an enumeration.
type Days =
   | Sun = 0
   | Mon = 1
   | Tues = 2
   | Wed = 3
   | Thurs = 4
   | Fri = 5
   | Sat = 6

// Use of an enumeration.
let weekend1 : Days = Days.Sat
let weekend2 : Days = Days.Sun
let weekDay1 : Days = Days.Mon

printfn "Monday: %A" weekDay1
printfn "Saturday: %A" weekend1
printfn "Sunday: %A" weekend2

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Monday: Mon
Saturday: Sat
Sunday: Sun

L'appariement de modèles vous permet de «comparer des données avec une ou plusieurs structures logiques, de décomposer les données en parties constituantes ou d'extraire des informations à partir de données de diverses manières».

En d'autres termes, il fournit un moyen plus flexible et plus puissant de tester les données par rapport à une série de conditions et d'effectuer certains calculs en fonction de la condition remplie.

Conceptuellement, c'est comme une série d'énoncés si… alors.

Syntaxe

En termes généraux, la correspondance de modèles suit cette syntaxe en F # -

match expr with
| pat1 - result1
| pat2 -> result2
| pat3 when expr2 -> result3
| _ -> defaultResult

Où,

  • Chacun | symbole définit une condition.
  • Le symbole -> signifie "si la condition est vraie, renvoie cette valeur ...".
  • Le symbole _ fournit le modèle par défaut, ce qui signifie qu'il correspond à toutes les autres choses comme un caractère générique.

Exemple 1

L'exemple suivant calcule les nombres de Fibonacci à l'aide de la syntaxe de correspondance de modèle -

let rec fib n =
   match n with
   | 0 -> 0
   | 1 -> 1
   | _ -> fib (n - 1) + fib (n - 2)
for i = 1 to 10 do
   printfn "Fibonacci %d: %d" i (fib i)

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Fibonacci 1: 1
Fibonacci 2: 1
Fibonacci 3: 2
Fibonacci 4: 3
Fibonacci 5: 5
Fibonacci 6: 8
Fibonacci 7: 13
Fibonacci 8: 21
Fibonacci 9: 34
Fibonacci 10: 55

Vous pouvez également enchaîner plusieurs conditions, qui renvoient la même valeur. Par exemple -

Exemple 2

let printSeason month =
   match month with
   | "December" | "January" | "February" -> printfn "Winter"
   | "March" | "April" -> printfn "Spring"
   | "May" | "June" -> printfn "Summer"
   | "July" | "August" -> printfn "Rainy"
   | "September" | "October" | "November" -> printfn "Autumn"
   | _ -> printfn "Season depends on month!"

printSeason "February"
printSeason "April"
printSeason "November"
printSeason "July"

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Winter
Spring
Autumn
Rainy

Fonctions de correspondance de modèles

F # vous permet d'écrire des fonctions de correspondance de motifs en utilisant function mot-clé -

let getRate = function
   | "potato" -> 10.00
   | "brinjal" -> 20.50
   | "cauliflower" -> 21.00
   | "cabbage" -> 8.75
   | "carrot" -> 15.00
   | _ -> nan (* nan is a special value meaning "not a number" *)

printfn "%g"(getRate "potato")
printfn "%g"(getRate "brinjal")
printfn "%g"(getRate "cauliflower")
printfn "%g"(getRate "cabbage")
printfn "%g"(getRate "carrot")

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

10
20.5
21
8.75
15

Ajout de filtres ou de gardes aux modèles

Vous pouvez ajouter des filtres ou des gardes aux motifs à l'aide de when mot-clé.

Exemple 1

let sign = function
   | 0 -> 0
   | x when x < 0 -> -1
   | x when x > 0 -> 1

printfn "%d" (sign -20)
printfn "%d" (sign 20)
printfn "%d" (sign 0)

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

-1
1
0

Exemple 2

let compareInt x =
   match x with
   | (var1, var2) when var1 > var2 -> printfn "%d is greater than %d" var1 var2
   | (var1, var2) when var1 < var2 -> printfn "%d is less than %d" var1 var2
   | (var1, var2) -> printfn "%d equals %d" var1 var2

compareInt (11,25)
compareInt (72, 10)
compareInt (0, 0)

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

11 is less than 25
72 is greater than 10
0 equals 0

Correspondance de motifs avec des tuples

L'exemple suivant illustre la correspondance de modèle avec les tuples -

let greeting (name, subject) =
   match (name, subject) with
   | ("Zara", _) -> "Hello, Zara"
   | (name, "English") -> "Hello, " + name + " from the department of English"
   | (name, _) when subject.StartsWith("Comp") -> "Hello, " + name + " from the department of Computer Sc."
   | (_, "Accounts and Finance") -> "Welcome to the department of Accounts and Finance!"
   | _ -> "You are not registered into the system"

printfn "%s" (greeting ("Zara", "English"))
printfn "%s" (greeting ("Raman", "Computer Science"))
printfn "%s" (greeting ("Ravi", "Mathematics"))

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Hello, Zara
Hello, Raman from the department of Computer Sc.
You are not registered into the system

Correspondance de modèles avec des enregistrements

L'exemple suivant illustre la mise en correspondance de modèles avec des enregistrements -

type Point = { x: float; y: float }
let evaluatePoint (point: Point) =
   match point with
   | { x = 0.0; y = 0.0 } -> printfn "Point is at the origin."
   | { x = xVal; y = 0.0 } -> printfn "Point is on the x-axis. Value is %f." xVal
   | { x = 0.0; y = yVal } -> printfn "Point is on the y-axis. Value is %f." yVal
   | { x = xVal; y = yVal } -> printfn "Point is at (%f, %f)." xVal yVal

evaluatePoint { x = 0.0; y = 0.0 }
evaluatePoint { x = 10.0; y = 0.0 }
evaluatePoint { x = 0.0; y = 10.0 }
evaluatePoint { x = 10.0; y = 10.0 }

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Point is at the origin.
Point is on the x-axis. Value is 10.000000.
Point is on the y-axis. Value is 10.000000.
Point is at (10.000000, 10.000000).

Une exception est un problème qui survient lors de l'exécution d'un programme. Une exception F # est une réponse à une circonstance exceptionnelle qui survient lors de l'exécution d'un programme, telle qu'une tentative de division par zéro.

Les exceptions fournissent un moyen de transférer le contrôle d'une partie d'un programme à une autre. La gestion des exceptions F # fournit les constructions suivantes -

Construction La description
augmenter expr Déclenche l'exception donnée.
échec avec expr Augmente le System.Exception exception.
essayez expr avec des règles Attrape les expressions correspondant aux règles du modèle.
essayer expr enfin expr Exécution le finally expression à la fois lorsque le calcul est réussi et lorsqu'une exception est déclenchée.
| :? ArgumentException Une règle correspondant au type d'exception .NET donné.
| :? ArgumentException comme e Une règle correspondant au type d'exception .NET donné, liant le nom e à la valeur de l'objet d'exception.
| Échec (msg) → expr Une règle correspondant à l'exception F # portant des données donnée.
| exn → expr Une règle correspondant à n'importe quelle exception, liant le nom exn à la valeur de l'objet d'exception.
| exn quand expr → expr Une règle correspondant à l'exception sous la condition donnée, liant le nom exn à la valeur de l'objet d'exception.

Commençons par la syntaxe de base de la gestion des exceptions.

Syntaxe

La syntaxe de base du bloc de gestion des exceptions F # est -

exception exception-type of argument-type

Où,

  • exception-type est le nom d'un nouveau type d'exception F #.

  • argument-type représente le type d'un argument qui peut être fourni lorsque vous déclenchez une exception de ce type.

  • Plusieurs arguments peuvent être spécifiés à l'aide d'un type tuple pour le type d'argument.

le try...with expression est utilisée pour la gestion des exceptions dans le langage F #.

La syntaxe de l'essai… avec expression est -

try
   expression1
with
   | pattern1 -> expression2
   | pattern2 -> expression3
...

le try...finally expression vous permet d'exécuter du code de nettoyage même si un bloc de code lève une exception.

La syntaxe de l'expression try… finally est -

try
   expression1
finally
   expression2

le raiseLa fonction est utilisée pour indiquer qu'une erreur ou une condition exceptionnelle s'est produite. Il capture également les informations sur l'erreur dans un objet d'exception.

La syntaxe de la fonction rise est -

raise (expression)

le failwith La fonction génère une exception F #.

La syntaxe de la fonction failwith est -

failwith error-message-string

le invalidArg La fonction génère une exception d'argument.

invalidArg parameter-name error-message-string

Exemple de gestion des exceptions

Exemple 1

Le programme suivant montre la gestion des exceptions de base avec un simple essai… avec bloc -

let divisionprog x y =
   try
      Some (x / y)
   with
      | :? System.DivideByZeroException -> printfn "Division by zero!"; None

let result1 = divisionprog 100 0

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Division by zero!

Exemple 2

F # fournit un exceptiontype pour déclarer des exceptions. Vous pouvez utiliser un type d'exception directement dans les filtres d'untry...with expression.

L'exemple suivant illustre ceci -

exception Error1 of string
// Using a tuple type as the argument type.
exception Error2 of string * int

let myfunction x y =
   try
      if x = y then raise (Error1("Equal Number Error"))
      else raise (Error2("Error Not detected", 100))
   with
      | Error1(str) -> printfn "Error1 %s" str
      | Error2(str, i) -> printfn "Error2 %s %d" str i
myfunction 20 10
myfunction 5 5

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Error2 Error Not detected 100
Error1 Equal Number Error

Exemple 3

L'exemple suivant illustre la gestion des exceptions imbriquées -

exception InnerError of string
exception OuterError of string

let func1 x y =
   try
      try
         if x = y then raise (InnerError("inner error"))
         else raise (OuterError("outer error"))
      with
         | InnerError(str) -> printfn "Error:%s" str
   finally
      printfn "From the finally block."

let func2 x y =
   try
      func1 x y
   with
      | OuterError(str) -> printfn "Error: %s" str

func2 100 150
func2 100 100
func2 100 120

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

From the finally block.
Error: outer error
Error:inner error
From the finally block.
From the finally block.
Error: outer error

Exemple 4

La fonction suivante montre le failwith fonction -

let divisionFunc x y =
   if (y = 0) then failwith "Divisor cannot be zero."
   else
      x / y

let trydivisionFunc x y =
   try
      divisionFunc x y
   with
      | Failure(msg) -> printfn "%s" msg; 0

let result1 = trydivisionFunc 100 0
let result2 = trydivisionFunc 100 4
printfn "%A" result1
printfn "%A" result2

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Divisor cannot be zero.
0
25

Exemple 5

le invalidArgLa fonction génère une exception d'argument. Le programme suivant le démontre -

let days = [| "Sunday"; "Monday"; "Tuesday"; "Wednesday"; "Thursday"; "Friday"; "Saturday" |]
let findDay day =
   if (day > 7 || day < 1)
      then invalidArg "day" (sprintf "You have entered %d." day)
   days.[day - 1]

printfn "%s" (findDay 1)
printfn "%s" (findDay 5)
printfn "%s" (findDay 9)

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Sunday
Thursday
Unhandled Exception:
System.ArgumentException: You have entered 9.
…

Certaines autres informations sur le fichier et la variable à l'origine de l'erreur dans le système seront également affichées, en fonction du système.

Les classes sont des types qui représentent des objets pouvant avoir des propriétés, des méthodes et des événements. «Ils sont utilisés pour modéliser des actions, des processus et toute entité conceptuelle dans les applications».

Syntaxe

La syntaxe pour définir un type de classe est la suivante -

// Class definition:
type [access-modifier] type-name [type-params] [access-modifier] ( parameter-list ) [ as identifier ] =
   [ class ]
      [ inherit base-type-name(base-constructor-args) ]
      [ let-bindings ]
      [ do-bindings ]
      member-list
      ...
   [ end ]

// Mutually recursive class definitions:
type [access-modifier] type-name1 ...
and [access-modifier] type-name2 ...
...

Où,

  • le type-nameest tout identifiant valide. Le modificateur d'accès par défaut pour cela estpublic.

  • le type-params décrit les paramètres de type générique facultatifs.

  • le parameter-listdécrit les paramètres du constructeur. Le modificateur d'accès par défaut pour le constructeur principal estpublic.

  • le identifier utilisé avec l'option as mot-clé donne un nom à la variable d'instance, ou self-identifier, qui peut être utilisé dans la définition de type pour faire référence à l'instance du type.

  • le inherit Le mot-clé vous permet de spécifier la classe de base d'une classe.

  • le let les liaisons vous permettent de déclarer des champs ou des valeurs de fonction locales à la classe.

  • le do-bindings La section comprend le code à exécuter lors de la construction de l'objet.

  • le member-list se compose de constructeurs supplémentaires, de déclarations d'instances et de méthodes statiques, de déclarations d'interface, de liaisons abstraites et de déclarations de propriétés et d'événements.

  • Les mots clés class et end qui marquent le début et la fin de la définition sont facultatifs.

Constructeur d'une classe

Le constructeur est un code qui crée une instance du type de classe.

En F #, les constructeurs fonctionnent un peu différemment des autres langages .Net. Dans la définition de classe, les arguments du constructeur principal sont décrits comme une liste de paramètres.

Le corps du constructeur se compose du let et do fixations.

Vous pouvez ajouter des constructeurs supplémentaires en utilisant le nouveau mot clé pour ajouter un membre -

new (argument-list) = constructor-body

L'exemple suivant illustre le concept -

Exemple

Le programme suivant crée une classe de ligne avec un constructeur qui calcule la longueur de la ligne pendant qu'un objet de la classe est créé -

type Line = class
   val X1 : float
   val Y1 : float
   val X2 : float
   val Y2 : float

   new (x1, y1, x2, y2) as this =
      { X1 = x1; Y1 = y1; X2 = x2; Y2 = y2;}
      then
         printfn " Creating Line: {(%g, %g), (%g, %g)}\nLength: %g"
            this.X1 this.Y1 this.X2 this.Y2 this.Length

   member x.Length =
      let sqr x = x * x
      sqrt(sqr(x.X1 - x.X2) + sqr(x.Y1 - x.Y2) )
end
let aLine = new Line(1.0, 1.0, 4.0, 5.0)

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Creating Line: {(1, 1), (4, 5)}
Length: 5

Laisser les liaisons

Les liaisons let dans une définition de classe vous permettent de définir des champs privés et des fonctions privées pour les classes F #.

type Greetings(name) as gr =
   let data = name
   do
      gr.PrintMessage()
   member this.PrintMessage() =
      printf "Hello %s\n" data
let gtr = new Greetings("Zara")

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Hello Zara

Veuillez noter l'utilisation de l'auto-identifiant gr pour la classe Greetings .

Une structure en F # est un type de données de type valeur. Il vous aide à créer une seule variable, contenant des données associées de différents types de données. lestruct Le mot-clé est utilisé pour créer une structure.

Syntaxe

La syntaxe pour définir une structure est la suivante -

[ attributes ]
type [accessibility-modifier] type-name =
   struct
      type-definition-elements
   end
// or
[ attributes ]
[<StructAttribute>]
type [accessibility-modifier] type-name =
   type-definition-elements

Il existe deux syntaxes. La première syntaxe est principalement utilisée, car si vous utilisez lestruct et end mots clés, vous pouvez omettre StructAttribute attribut.

Les éléments de définition de structure fournissent -

  • Déclarations et définitions des membres.
  • Constructeurs et champs mutables et immuables.
  • Membres et implémentations d'interface.

Contrairement aux classes, les structures ne peuvent pas être héritées et ne peuvent pas contenir de liaisons let ou do. Depuis, les structures n'ont pas de liaisons let; vous devez déclarer les champs dans les structures en utilisant leval mot-clé.

Lorsque vous définissez un champ et son type à l'aide de valmot clé, vous ne pouvez pas initialiser la valeur du champ, au lieu de cela, ils sont initialisés à zéro ou à null. Donc, pour une structure ayant un constructeur implicite, leval les déclarations soient annotées avec le DefaultValue attribut.

Exemple

Le programme suivant crée une structure de ligne avec un constructeur. Le programme calcule la longueur d'une ligne en utilisant la structure -

type Line = struct
   val X1 : float
   val Y1 : float
   val X2 : float
   val Y2 : float

   new (x1, y1, x2, y2) =
      {X1 = x1; Y1 = y1; X2 = x2; Y2 = y2;}
end
let calcLength(a : Line)=
   let sqr a = a * a
   sqrt(sqr(a.X1 - a.X2) + sqr(a.Y1 - a.Y2) )

let aLine = new Line(1.0, 1.0, 4.0, 5.0)
let length = calcLength aLine
printfn "Length of the Line: %g " length

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Length of the Line: 5

Vous pouvez redéfinir ou surcharger la plupart des opérateurs intégrés disponibles dans F #. Ainsi, un programmeur peut également utiliser des opérateurs avec des types définis par l'utilisateur.

Les opérateurs sont des fonctions avec des noms spéciaux, entre crochets. Ils doivent être définis comme des membres de classe statiques. Comme toute autre fonction, un opérateur surchargé a un type de retour et une liste de paramètres.

L'exemple suivant montre un opérateur + sur les nombres complexes -

//overloading + operator
static member (+) (a : Complex, b: Complex) =
Complex(a.x + b.x, a.y + b.y)

La fonction ci-dessus implémente l'opérateur d'addition (+) pour une classe définie par l'utilisateur Complex. Il ajoute les attributs de deux objets et renvoie l'objet Complex résultant.

Mise en œuvre de la surcharge des opérateurs

Le programme suivant montre l'implémentation complète -

//implementing a complex class with +, and - operators
//overloaded
type Complex(x: float, y : float) =
   member this.x = x
   member this.y = y
   //overloading + operator
   static member (+) (a : Complex, b: Complex) =
      Complex(a.x + b.x, a.y + b.y)

   //overloading - operator
   static member (-) (a : Complex, b: Complex) =
      Complex(a.x - b.x, a.y - b.y)

   // overriding the ToString method
   override this.ToString() =
      this.x.ToString() + " " + this.y.ToString()

//Creating two complex numbers
let c1 = Complex(7.0, 5.0)
let c2 = Complex(4.2, 3.1)

// addition and subtraction using the
//overloaded operators
let c3 = c1 + c2
let c4 = c1 - c2

//printing the complex numbers
printfn "%s" (c1.ToString())
printfn "%s" (c2.ToString())
printfn "%s" (c3.ToString())
printfn "%s" (c4.ToString())

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

7 5
4.2 3.1
11.2 8.1
2.8 1.9

L'un des concepts les plus importants de la programmation orientée objet est celui de l'héritage. L'héritage nous permet de définir une classe en termes d'une autre classe, ce qui facilite la création et la maintenance d'une application. Cela offre également la possibilité de réutiliser la fonctionnalité de code et un temps de mise en œuvre rapide.

Lors de la création d'une classe, au lieu d'écrire des membres de données et des fonctions membres complètement nouveaux, le programmeur peut désigner que la nouvelle classe doit hériter des membres d'une classe existante. Cette classe existante est appelée la classe de base et la nouvelle classe est appelée la classe dérivée.

L'idée d'héritage met en œuvre la relation IS-A. Par exemple, le mammifère EST un animal, le chien IS-A mammifère donc le chien IS-A également et ainsi de suite.

Classe de base et sous-classe

Une sous-classe est dérivée d'une classe de base, qui est déjà définie. Une sous-classe hérite des membres de la classe de base, ainsi que de ses propres membres.

Une sous-classe est définie à l'aide du inherit mot-clé comme indiqué ci-dessous -

type MyDerived(...) =
   inherit MyBase(...)

En F #, une classe peut avoir au plus une classe de base directe. Si vous ne spécifiez pas de classe de base en utilisant leinherit mot-clé, la classe hérite implicitement de Object.

Veuillez noter -

  • Les méthodes et les membres de la classe de base sont disponibles pour les utilisateurs de la classe dérivée comme les membres directs de la classe dérivée.

  • Laissez les liaisons et les paramètres du constructeur sont privés pour une classe et, par conséquent, ne sont pas accessibles à partir des classes dérivées.

  • Le mot clé basefait référence à l'instance de classe de base. Il est utilisé comme l'auto-identifiant.

Exemple

type Person(name) =
   member x.Name = name
   member x.Greet() = printfn "Hi, I'm %s" x.Name

type Student(name, studentID : int) =
   inherit Person(name)
   let mutable _GPA = 0.0
   member x.StudentID = studentID
   member x.GPA
      with get() = _GPA
      and set value = _GPA <- value

type Teacher(name, expertise : string) =
   inherit Person(name)

   let mutable _salary = 0.0
   member x.Salary
      with get() = _salary
      and set value = _salary <- value
   member x.Expertise = expertise

//using the subclasses
let p = new Person("Mohan")
let st = new Student("Zara", 1234)
let tr = new Teacher("Mariam", "Java")

p.Greet()
st.Greet()
tr.Greet()

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Hi, I'm Mohan
Hi, I'm Zara
Hi, I'm Mariam

Remplacer les méthodes

Vous pouvez remplacer un comportement par défaut d'une méthode de classe de base et l'implémenter différemment dans la sous-classe ou la classe dérivée.

Les méthodes en F # ne sont pas remplaçables par défaut.

Pour remplacer des méthodes dans une classe dérivée, vous devez déclarer votre méthode comme substituable à l'aide du abstract et default mots-clés comme suit -

type Person(name) =
   member x.Name = name
   abstract Greet : unit -> unit
   default x.Greet() = printfn "Hi, I'm %s" x.Name

Désormais, la méthode Greet de la classe Person peut être remplacée dans les classes dérivées. L'exemple suivant illustre ceci -

Exemple

type Person(name) =
   member x.Name = name
   abstract Greet : unit -> unit
   default x.Greet() = printfn "Hi, I'm %s" x.Name

type Student(name, studentID : int) =
   inherit Person(name)

   let mutable _GPA = 0.0

   member x.StudentID = studentID
   member x.GPA
      with get() = _GPA
      and set value = _GPA <- value
   override x.Greet() = printfn "Student %s" x.Name

type Teacher(name, expertise : string) =
   inherit Person(name)
   let mutable _salary = 0.0
   member x.Salary
      with get() = _salary
      and set value = _salary <- value

   member x.Expertise = expertise
   override x.Greet() = printfn "Teacher %s." x.Name

//using the subclasses
let p = new Person("Mohan")
let st = new Student("Zara", 1234)
let tr = new Teacher("Mariam", "Java")

//default Greet
p.Greet()

//Overriden Greet
st.Greet()
tr.Greet()

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Hi, I'm Mohan
Student Zara
Teacher Mariam.

Classe abstraite

Parfois, vous devez fournir une implémentation incomplète d'un objet, qui ne doit pas être implémentée dans la réalité. Plus tard, un autre programmeur devrait créer des sous-classes de la classe abstraite pour une implémentation complète.

Par exemple, la classe Personne ne sera pas nécessaire dans un système de gestion scolaire. Cependant, la classe Étudiant ou Professeur sera nécessaire. Dans de tels cas, vous pouvez déclarer la classe Person en tant que classe abstraite.

le AbstractClass L'attribut indique au compilateur que la classe a des membres abstraits.

Vous ne pouvez pas créer une instance d'une classe abstraite car la classe n'est pas entièrement implémentée.

L'exemple suivant illustre ceci -

Exemple

[<AbstractClass>]
type Person(name) =
   member x.Name = name
   abstract Greet : unit -> unit

type Student(name, studentID : int) =
   inherit Person(name)
   let mutable _GPA = 0.0
   member x.StudentID = studentID
   member x.GPA
      with get() = _GPA
      and set value = _GPA <- value
   override x.Greet() = printfn "Student %s" x.Name

type Teacher(name, expertise : string) =
   inherit Person(name)
   let mutable _salary = 0.0
   member x.Salary
      with get() = _salary
      and set value = _salary <- value
   member x.Expertise = expertise
   override x.Greet() = printfn "Teacher %s." x.Name

let st = new Student("Zara", 1234)
let tr = new Teacher("Mariam", "Java")

//Overriden Greet
st.Greet()
tr.Greet()

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Student Zara
Teacher Mariam.

Les interfaces fournissent une manière abstraite d'écrire les détails d'implémentation d'une classe. C'est un modèle qui déclare les méthodes que la classe doit implémenter et exposer publiquement.

Syntaxe

Une interface spécifie les ensembles de membres associés que d'autres classes implémentent. Il a la syntaxe suivante -

// Interface declaration:
[ attributes ]
type interface-name =
   [ interface ]
      [ inherit base-interface-name ...]
      abstract member1 : [ argument-types1 -> ] return-type1
      abstract member2 : [ argument-types2 -> ] return-type2
      ...
   [ end ]
	
// Implementing, inside a class type definition:
interface interface-name with
   member self-identifier.member1 argument-list = method-body1
   member self-identifier.member2 argument-list = method-body2
// Implementing, by using an object expression:
[ attributes ]
let class-name (argument-list) =
   { new interface-name with
      member self-identifier.member1 argument-list = method-body1
      member self-identifier.member2 argument-list = method-body2
      [ base-interface-definitions ]
   }
member-list

Veuillez noter -

  • Dans une déclaration d'interface, les membres ne sont pas implémentés.

  • Les membres sont abstraits, déclarés par le abstractmot-clé. Cependant, vous pouvez fournir une implémentation par défaut en utilisant ledefault mot-clé.

  • Vous pouvez implémenter des interfaces en utilisant des expressions d'objet ou en utilisant des types de classe.

  • Dans l'implémentation de classe ou d'objet, vous devez fournir des corps de méthode pour les méthodes abstraites de l'interface.

  • Les mots clés interface et end, qui marquent le début et la fin de la définition, sont facultatifs.

Par exemple,

type IPerson =
   abstract Name : string
   abstract Enter : unit -> unit
   abstract Leave : unit -> unit

Appel des méthodes d'interface

Les méthodes d'interface sont appelées via l'interface, et non via l'instance de l'interface d'implémentation de classe ou de type. Pour appeler une méthode d'interface, vous transtypez vers le type d'interface en utilisant le:> opérateur (opérateur upcast).

Par exemple,

(s :> IPerson).Enter()
(s :> IPerson).Leave()

L'exemple suivant illustre le concept -

Exemple

type IPerson =
   abstract Name : string
   abstract Enter : unit -> unit
   abstract Leave : unit -> unit

type Student(name : string, id : int) =
   member this.ID = id
   interface IPerson with
      member this.Name = name
      member this.Enter() = printfn "Student entering premises!"
      member this.Leave() = printfn "Student leaving premises!"

type StuffMember(name : string, id : int, salary : float) =
   let mutable _salary = salary

   member this.Salary
      with get() = _salary
      and set(value) = _salary <- value

   interface IPerson with
      member this.Name = name
      member this.Enter() = printfn "Stuff member entering premises!"
      member this.Leave() = printfn "Stuff member leaving premises!"

let s = new Student("Zara", 1234)
let st = new StuffMember("Rohit", 34, 50000.0)

(s :> IPerson).Enter()
(s :> IPerson).Leave()
(st :> IPerson).Enter()
(st :> IPerson).Leave()

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Student entering premises!
Student leaving premises!
Stuff member entering premises!
Stuff member leaving premises!

Héritage d'interface

Les interfaces peuvent hériter d'une ou plusieurs interfaces de base.

L'exemple suivant montre le concept -

type Interface1 =
   abstract member doubleIt: int -> int

type Interface2 =
   abstract member tripleIt: int -> int

type Interface3 =
   inherit Interface1
   inherit Interface2
   abstract member printIt: int -> string

type multiplierClass() =
   interface Interface3 with
      member this.doubleIt(a) = 2 * a
      member this.tripleIt(a) = 3 * a
      member this.printIt(a) = a.ToString()

let ml = multiplierClass()
printfn "%d" ((ml:>Interface3).doubleIt(5))
printfn "%d" ((ml:>Interface3).tripleIt(5))
printfn "%s" ((ml:>Interface3).printIt(5))

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

10
15
5

Les événements permettent aux classes d'envoyer et de recevoir des messages entre elles.

Dans l'interface graphique, les événements sont des actions de l'utilisateur telles que la pression de touches, les clics, les mouvements de la souris, etc., ou certaines occurrences telles que les notifications générées par le système. Les applications doivent répondre aux événements lorsqu'ils se produisent. Par exemple, les interruptions. Les événements sont utilisés pour la communication inter-processus.

Les objets communiquent entre eux grâce à la transmission synchrone des messages.

Les événements sont associés à d'autres fonctions; registre d'objetscallback fonctions à un événement, et ces rappels sont exécutés lorsque (et si) l'événement est déclenché par un objet.

La classe d'événements et le module d'événements

La classe Control.Event <'T> aide à créer un objet ou un événement observable.

Il a les membres d'instance suivants pour travailler avec les événements -

Membre La description
Publier Publie une observation en tant que valeur de première classe.
Déclencheur Déclenche une observation en utilisant les paramètres donnés.

Le module Control.Event fournit des fonctions de gestion des flux d'événements -

Valeur La description
ajouter: ('T → unité) → Evénement <' Del, 'T> → unité Exécute la fonction donnée à chaque fois que l'événement donné est déclenché.
choisissez: (option 'T →' U) → IEvent <'Del,' T> → IEvent <'U> Renvoie un nouvel événement qui se déclenche sur une sélection de messages de l'événement d'origine. La fonction de sélection prend un message d'origine dans un nouveau message facultatif.
filtre: ('T → bool) → IEvent <' Del, 'T> → IEvent <' T> Renvoie un nouvel événement qui écoute l'événement d'origine et déclenche l'événement résultant uniquement lorsque l'argument de l'événement transmet la fonction donnée.
map: ('T →' U) → IEvent <'Del,' T> → IEvent <'U> Renvoie un nouvel événement qui transmet les valeurs transformées par la fonction donnée.
fusion: IEvent <'Del1,' T> → IEvent <'Del2,' T> → IEvent <'T> Déclenche l'événement de sortie lorsque l'un des événements d'entrée se déclenche.
par paire: IEvent <'Del,' T> → IEvent <'T *' T> Renvoie un nouvel événement qui se déclenche au deuxième déclenchement et au déclenchement suivant de l'événement d'entrée. leNth le déclenchement de l'événement d'entrée transmet les arguments du N-1th et Nthse déclenchant en paire. L'argument passé auN-1th le déclenchement est maintenu dans l'état interne caché jusqu'à ce que le Nth le déclenchement se produit.
partition: ('T → booléen) → IEvent <' Del, 'T> → IEvent <' T> * IEvent <'T> Renvoie un nouvel événement qui écoute l'événement d'origine et déclenche le premier événement résultant si l'application du prédicat aux arguments d'événement a renvoyé true, et le deuxième événement s'il a renvoyé false.
scan: ('U →' T → 'U) →' U → IEvent <'Del,' T> → IEvent <'U> Renvoie un nouvel événement constitué des résultats de l'application de la fonction d'accumulation donnée à des valeurs successives déclenchées sur l'événement d'entrée. Un élément d'état interne enregistre la valeur actuelle du paramètre d'état. L'état interne n'est pas verrouillé pendant l'exécution de la fonction d'accumulation, il faut donc veiller à ce que l'entrée IEvent ne soit pas déclenchée par plusieurs threads simultanément.
split: ('T → Choice <' U1, 'U2>) → IEvent <' Del, 'T> → IEvent <' U1> * IEvent <'U2> Renvoie un nouvel événement qui écoute l'événement d'origine et déclenche le premier événement résultant si l'application de la fonction aux arguments d'événement a renvoyé un Choice1Of2, et le deuxième événement s'il renvoie un Choice2Of2.

Créer des événements

Les événements sont créés et utilisés via le Eventclasse. Le constructeur Event est utilisé pour créer un événement.

Exemple

type Worker(name : string, shift : string) =
   let mutable _name = name;
   let mutable _shift = shift;
   let nameChanged = new Event<unit>() (* creates event *)
   let shiftChanged = new Event<unit>() (* creates event *)

   member this.Name
      with get() = _name
      and set(value) = _name <- value

   member this.Shift
      with get() = _shift
      and set(value) = _shift <- value

Après cela, vous devez exposer le champ nameChanged en tant que membre public, afin que les écouteurs puissent se connecter à l'événement pour lequel vous utilisez le Publish propriété de l'événement -

type Worker(name : string, shift : string) =
   let mutable _name = name;
   let mutable _shift = shift;

   let nameChanged = new Event<unit>() (* creates event *)
   let shiftChanged = new Event<unit>() (* creates event *)

   member this.NameChanged = nameChanged.Publish (* exposed event handler *)
   member this.ShiftChanged = shiftChanged.Publish (* exposed event handler *)

   member this.Name
      with get() = _name
      and set(value) = _name <- value
      nameChanged.Trigger() (* invokes event handler *)

   member this.Shift
      with get() = _shift
      and set(value) = _shift <- value
   shiftChanged.Trigger() (* invokes event handler *)

Ensuite, vous ajoutez des rappels aux gestionnaires d'événements. Chaque gestionnaire d'événements a le type IEvent <'T>, qui fournit plusieurs méthodes -

Méthode La description
val Ajouter: événement :( 'T → unité) → unité Connecte une fonction d'écoute à l'événement. L'écouteur sera appelé lorsque l'événement est déclenché.
val AddHandler: 'del → unité Connecte un objet délégué de gestionnaire à l'événement. Un gestionnaire peut être supprimé ultérieurement à l'aide de RemoveHandler. L'écouteur sera appelé lorsque l'événement est déclenché.
val RemoveHandler: 'del → unité Supprime un délégué d'écouteur d'un magasin d'écouteurs d'événements.

La section suivante fournit un exemple complet.

Exemple

L'exemple suivant montre le concept et les techniques discutés ci-dessus -

type Worker(name : string, shift : string) =
   let mutable _name = name;
   let mutable _shift = shift;

   let nameChanged = new Event<unit>() (* creates event *)
   let shiftChanged = new Event<unit>() (* creates event *)

   member this.NameChanged = nameChanged.Publish (* exposed event handler *)
   member this.ShiftChanged = shiftChanged.Publish (* exposed event handler *)

   member this.Name
      with get() = _name
      and set(value) = 
         _name <- value
         nameChanged.Trigger() (* invokes event handler *)

   member this.Shift
      with get() = _shift
      and set(value) = 
         _shift <- value
         shiftChanged.Trigger() (* invokes event handler *)

let wk = new Worker("Wilson", "Evening")
wk.NameChanged.Add(fun () -> printfn "Worker changed name! New name: %s" wk.Name)
wk.Name <- "William"
wk.NameChanged.Add(fun () -> printfn "-- Another handler attached to NameChanged!")
wk.Name <- "Bill"

wk.ShiftChanged.Add(fun () -> printfn "Worker changed shift! New shift: %s" wk.Shift)
wk.Shift <- "Morning"
wk.ShiftChanged.Add(fun () -> printfn "-- Another handler attached to ShiftChanged!")
wk.Shift <- "Night"

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Worker changed name! New name: William
Worker changed name! New name: Bill
-- Another handler attached to NameChanged!
Worker changed shift! New shift: Morning
Worker changed shift! New shift: Night
-- Another handler attached to ShiftChanged!

Selon la bibliothèque MSDN, un module F # est un regroupement de constructions de code F #, telles que les types, les valeurs, les valeurs de fonction et le code dans les liaisons do. Il est implémenté en tant que classe CLR (Common Language Runtime) qui n'a que des membres statiques.

Selon la situation, si le fichier entier est inclus dans le module, il existe deux types de déclarations de module -

  • Déclaration de module de niveau supérieur
  • Déclaration du module local

Dans une déclaration de module de niveau supérieur, le fichier entier est inclus dans le module. Dans ce cas, la première déclaration du fichier est la déclaration du module. Vous n'avez pas à mettre en retrait les déclarations dans un module de niveau supérieur.

Dans une déclaration de module local, seules les déclarations qui sont indentées sous cette déclaration de module font partie du module.

Syntaxe

La syntaxe de la déclaration du module est la suivante -

// Top-level module declaration.
module [accessibility-modifier] [qualified-namespace.]module-name
   declarations
// Local module declaration.
module [accessibility-modifier] module-name =
   declarations

Veuillez noter que le modificateur d'accessibilité peut être l'un des suivants - public, privé, interne. La valeur par défaut estpublic.

Les exemples suivants démontreront les concepts -

Exemple 1

Le fichier de module Arithmetic.fs -

module Arithmetic
let add x y =
   x + y

let sub x y =
   x - y
	
let mult x y =
   x * y
	
let div x y =
   x / y

Le fichier programme main.fs -

// Fully qualify the function name.
open Arithmetic
let addRes = Arithmetic.add 25 9
let subRes = Arithmetic.sub 25 9
let multRes = Arithmetic.mult 25 9
let divRes = Arithmetic.div 25 9

printfn "%d" addRes
printfn "%d" subRes
printfn "%d" multRes
printfn "%d" divRes

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

34
16
225
2
110
90
1000
10

Exemple 2

// Module1
module module1 =
   // Indent all program elements within modules that are declared with an equal sign.
   let value1 = 100
   let module1Function x =
      x + value1

// Module2
module module2 =
   let value2 = 200

   // Use a qualified name to access the function.
   // from module1.
   let module2Function x =
      x + (module1.module1Function value2)

let result = module1.module1Function 25
printfn "%d" result

let result2 = module2.module2Function 25
printfn "%d" result2

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

125
325

UNE namespaceest conçu pour fournir un moyen de séparer un ensemble de noms d'un autre. Les noms de classe déclarés dans un espace de noms n'entreront pas en conflit avec les mêmes noms de classe déclarés dans un autre.

Selon la bibliothèque MSDN, un namespace vous permet d'organiser le code dans des zones de fonctionnalités associées en vous permettant d'attacher un nom à un groupe d'éléments de programme.

Déclaration d'un espace de noms

Pour organiser votre code dans un espace de noms, vous devez déclarer l'espace de noms comme première déclaration dans le fichier. Le contenu du fichier entier devient alors partie de l'espace de noms.

namespace [parent-namespaces.]identifier

L'exemple suivant illustre le concept -

Exemple

namespace testing

module testmodule1 =
   let testFunction x y =
      printfn "Values from Module1: %A %A" x y
module testmodule2 =
   let testFunction x y =
      printfn "Values from Module2: %A %A" x y

module usermodule =
   do
      testmodule1.testFunction ( "one", "two", "three" ) 150
      testmodule2.testFunction (seq { for i in 1 .. 10 do yield i * i }) 200

Lorsque vous compilez et exécutez le programme, il produit la sortie suivante -

Values from Module1: ("one", "two", "three") 150
Values from Module2: seq [1; 4; 9; 16; ...] 200