EXAMEN. Durée : 2h, le seul document autorisé est une feuille A4 recto-verso manuscrite de notes personnelles

Documents pareils
Rappel. Analyse de Données Structurées - Cours 12. Un langage avec des déclaration locales. Exemple d'un programme

IN Cours 1. 1 Informatique, calculateurs. 2 Un premier programme en C

Classes et Objets en Ocaml.

Recherche dans un tableau

# let rec concat l1 l2 = match l1 with [] -> l2 x::l 1 -> x::(concat l 1 l2);; val concat : a list -> a list -> a list = <fun>

Algorithmique et Programmation, IMA

Travaux pratiques. Compression en codage de Huffman Organisation d un projet de programmation

Cours de Programmation 2

1. Structure d un programme C. 2. Commentaire: /*..texte */ On utilise aussi le commentaire du C++ qui est valable pour C: 3.

Suivant les langages de programmation, modules plus avancés : modules imbriqués modules paramétrés par des modules (foncteurs)

modules & compilation

Les arbres binaires de recherche

1/24. I passer d un problème exprimé en français à la réalisation d un. I expressions arithmétiques. I structures de contrôle (tests, boucles)

TP 1. Prise en main du langage Python

Projet d informatique M1BI : Compression et décompression de texte. 1 Généralités sur la compression/décompression de texte


Programmation C++ (débutant)/instructions for, while et do...while

Prénom : Matricule : Sigle et titre du cours Groupe Trimestre INF1101 Algorithmes et structures de données Tous H2004. Loc Jeudi 29/4/2004

Arbres binaires de recherche

Machines virtuelles fonctionnelles (suite) Compilation ML Java

STAGE IREM 0- Premiers pas en Python

Théorie de la Programmation

Vérification formelle de la plate-forme Java Card

Licence Sciences et Technologies Examen janvier 2010

Initiation à la Programmation en Logique avec SISCtus Prolog

Algorithmique et Programmation Fonctionnelle

LES TYPES DE DONNÉES DU LANGAGE PASCAL

1. Structure d'un programme FORTRAN 95

Centre CPGE TSI - Safi 2010/2011. Algorithmique et programmation :

INITIATION AU LANGAGE C SUR PIC DE MICROSHIP

UE Programmation Impérative Licence 2ème Année

length : A N add : Z Z Z (n 1, n 2 ) n 1 + n 2

KL5121. Pour activer des sorties en fonction de la position d'un codeur

SNT4U16 - Initiation à la programmation TD - Dynamique de POP III - Fichiers sources

Java Licence Professionnelle CISII,

Conventions d écriture et outils de mise au point

Bases de programmation. Cours 5. Structurer les données

Éléments d informatique Cours 3 La programmation structurée en langage C L instruction de contrôle if

Manuel d utilisation 26 juin Tâche à effectuer : écrire un algorithme 2

TP3 : Manipulation et implantation de systèmes de fichiers 1

Cours d Algorithmique-Programmation 2 e partie (IAP2): programmation 24 octobre 2007impérative 1 / 44 et. structures de données simples

TP : Gestion d une image au format PGM

Compilation (INF 564)

Probabilités. Rappel : trois exemples. Exemple 2 : On dispose d un dé truqué. On sait que : p(1) = p(2) =1/6 ; p(3) = 1/3 p(4) = p(5) =1/12

3. SPÉCIFICATIONS DU LOGICIEL. de l'expression des besoins à la conception. Spécifications fonctionnelles Analyse fonctionnelle et méthodes

Corrigé des TD 1 à 5

Cours de C++ François Laroussinie. 2 novembre Dept. d Informatique, ENS de Cachan

Cours 1 : La compilation

Expression des contraintes. OCL : Object C o n t r a i n t L a n g u a g e

Programme Compte bancaire (code)

Compléments de documentation Scilab : affichage de texte et formatage de nombres

Vérification de programmes et de preuves Première partie. décrire des algorithmes

Projet L1, S2, 2015: Simulation de fourmis, Soutenance la semaine du 4 mai.

Algorithmique & programmation

Image d un intervalle par une fonction continue

Reconstruction de bâtiments en 3D à partir de nuages de points LIDAR

Limites finies en un point

TP n 2 Concepts de la programmation Objets Master 1 mention IL, semestre 2 Le type Abstrait Pile

Rappel sur les bases de données

Cours d Algorithmique et de Langage C v 3.0

Chap III : Les tableaux

Présentation du langage et premières fonctions

Cours intensif Java. 1er cours: de C à Java. Enrica DUCHI LIAFA, Paris 7. Septembre Enrica.Duchi@liafa.jussieu.fr

EPREUVE OPTIONNELLE d INFORMATIQUE CORRIGE

ARDUINO DOSSIER RESSOURCE POUR LA CLASSE

ARBRES BINAIRES DE RECHERCHE

Les structures de données. Rajae El Ouazzani

Sub CalculAnnuite() Const TITRE As String = "Calcul d'annuité de remboursement d'un emprunt"

Informatique Générale

Découverte de Python

INTRODUCTION AUX SYSTEMES D EXPLOITATION. TD2 Exclusion mutuelle / Sémaphores

Algorithmique et programmation : les bases (VBA) Corrigé

ACTIVITÉ DE PROGRAMMATION

Les structures. Chapitre 3

Gestion mémoire et Représentation intermédiaire

Ensimag 1ère année Algorithmique 1 Examen 2ième session 24 juin Algorithmique 1

Cours Informatique Master STEP

Cours d introduction à l informatique. Partie 2 : Comment écrire un algorithme? Qu est-ce qu une variable? Expressions et instructions

Logiciel Libre Cours 2 Fondements: Programmation

EES : Engineering Equation Solver Fiche récapitulative - Marie-Sophie Cabot

Exceptions. 1 Entrées/sorties. Objectif. Manipuler les exceptions ;

Chapitre 10. Les interfaces Comparable et Comparator 1

Lambda! Rémi Forax Univ Paris-Est Marne-la-Vallée

Cours 1 : Introduction Ordinateurs - Langages de haut niveau - Application

Programmation en Caml pour Débutants

Seance 2: En respectant la méthode de programmation par contrat, implémentez les autres fonctions de jeu.

Chapitre 2. Eléments pour comprendre un énoncé

Le présent document a pour objet de définir les modalités d organisation de l élection des délégués du personnel (ou du C E) de

Notions fondamentales du langage C# Version 1.0

Application 1- VBA : Test de comportements d'investissements

Utilitaires méconnus de StrataFrame

Claude Delannoy. 3 e édition C++

Algorithmique, Structures de données et langage C

V- Manipulations de nombres en binaire

Le prototype de la fonction main()

Représentation d un entier en base b

MIS 102 Initiation à l Informatique

Document d aide au suivi scolaire

Java Licence Professionnelle Cours 7 : Classes et méthodes abstraites

Propagation sur réseau statique et dynamique

Transcription:

Nom : RICM3-2013/2014 Prénom : Algorithmique et Programmation Fonctionnelle EXAMEN Durée : 2h, le seul document autorisé est une feuille A4 recto-verso manuscrite de notes personnelles Cet énoncé comporte des parties indépendantes. Le barème est indicatif, le nombre de points correspond au nombre de minutes estimé nécessaire pour faire les exercices. Le total des points est de 120 points. Exercices Exercice 1 : Typage et sens d une expression (15 points) Donner le type et expliquer en une ou deux phrases ce que fait chacun des programmes suivants (qui sont tous des programmes corrects, utiles et bien typés). Vous pourrez vous appuyer sur des exemples. 1. let rec f1 f l = match l with [] -> () h::t -> f h ; f1 f t ( a -> unit) -> a list -> unit List.iter 2. let rec f2 m n = match m, n with x::y, z::t -> x::z::(f2 y t) [], _ -> n _, _ -> m a list -> a list -> a list Alterne les éléments de deux listes (pas forcément de mêmes tailles mais de même type). 3. let rec f3 l = match l with [] -> [] h::t -> match f3 t with [] -> [h] hh::tt -> (h+hh) :: hh :: tt 1

4. let f4 f x y = f (x,y) int list -> int list Sommes cumulées décroissantes ( a * b -> c) -> a -> b -> c Curryfication Exercice 2 : λ-calcul (15 points) On rappelle que les booléens sont représentés en λ-calcul par les termes suivants : true = λxy.x et false = λxy.y. 1. Vérifier que λbuv.buv représente bien la fonction if thenelse, c est-à-dire que pour tous termes M et N : (λbuv.buv) true M N se β-réduit en M. (λbuv.buv) false M N se β-réduit en N. (λbuv.buv)true M N (λuv.true u v)m N (λv.true M v)n (λxy.x) M N (λy.m)n M (λbuv.buv)false M N (λuv.false u v)m N (λv.false M v)n (λxy.y) M N (λy.y)n N 2. Donner un terme not représentant la négation booléenne et vérifier que not true se β-réduit en false. not = λx.ifthenelsex false true = λx.x false true not true = (λx.x false true) true true false true (λy.false) true false 3. Donner un terme eq représentant l opérateur booléen d équivalence. eq = λxy.ifthenelsex y (not y) λxy.x y (not y) = λxy.x y (y false true) 2

Exercice 3 : Module (15 points) On donne la signature d un module Urne permettant de faire un vote à bulletin secret : Il y a deux candidats A et B. Chaque électeur est doté d un numéro entier, ce qui permet de s assurer qu un même électeur ne peut pas comptabiliser deux votes En cours de scrutin, un électeur peut changer d avis et re-voter (l ancien vote est alors détruit). On peut dépouiller l urne pour connaître les scores de chaque candidat et déterminer le vainqueur. Le type t est celui de l urne proprement dite. module type URNE = sig type electeur = int type candidat = A B type t ( type de l urne ) val vide : t val voter : electeur candidat t t val depouiller : t int int val vainqueur : t candidat end Dire pourquoi à l extérieur du module il n est possible à aucun moment de savoir qui a déjà voté, ni quel est le vote de quelqu un. Le type t est abstrait et les seules fonctions dans l interface qui fournissent une information lisible à partir de t sont dépouiller et vainqueur (le résultat de voter reste dans t) ; ces fonctions sont supposées n extraire qu une vue information globale, insuffisante pour analyser complètement un élément de t. Implémenter ce module. 3

module Urne : URNE = struct type electeur = int type candidat = A B type t = (electeur candidat) list let vide = [ ] let rec voter e c l = match l with [ ] [(e, c)] (eh, ch) :: t if eh = e then (eh, c) :: t else (eh, ch) :: (voter e c t) let rec depouiller l = match l with [ ] (0, 0) (eh, ch) :: t let (n, m) = depouiller t in match ch with A (n + 1, m) B (n, m + 1) let vainqueur l = let (n, m) = depouiller l in if n > m then A else B end Exercice 4 : Impératif (15 points) 1. Définir une fonction arraymap : ( a -> b) -> a array -> b array telle que arraymap f a construit un tableau dont les éléments sont ceux obtenus en appliquant la fonction f à chaque élément de a. let arraymap f a = let l = Array.length a in if l = 0 then [ ] else let b = Array.make l (f a.(0)) in for i = 0 to l 1 do b.(i) f (a.(i)) done ; b 2. Donner le type puis définir une fonction arrayexists telle que arrayexists p a détermine s il existe ou non un élément dans le tableau a qui vérifie le prédicat p. Une fonction qui ne parcourt pas tout le tableau si ce n est pas nécessaire rapportera un (petit) bonus de points. 4

arrayexists : ( a -> bool) -> a array -> bool let arrayexists p a = let i = ref 0 and e = ref false in while!i < Array.length a!e do e := p a.(!i); i :=!i + 1 done ;!e Problème (60 points) Dans ce problème on cherche à représenter de façon efficace un ensemble d intervalles d entiers disjoints deux à deux. On pourra ainsi représenter par exemple l affectation de tâches à un processeur, chaque intervalle représentant les cycles durant lesquels le processeur sera occupé par une tâche donnée. Le cycle d horloge étant la plus petite unité de temps, deux tâches ne pourront avoir un cycle en commun, même aux bornes des intervalles qui les décrivent. De même que pour les entiers, on imagine bien qu une liste triée d intervalles n est pas la structure la plus efficace, surtout dans la mesure où on voudra rechercher, insérer ou supprimer des intervalles. On considère donc des arbres binaires de recherche d intervalles (en abrégé ABRI), dont les nœuds sont étiquetés non plus par un entier mais par un intervalle d entiers (en pratique, ses bornes). On définit pour cela les types OCaml suivants : type intervalle = int * int type abri = F N of abri * intervalle * abri Dans tout l énoncé, on pourra supposer que les bornes des intervalles donnés en argument d une fonction sont en ordre croissant et qu il n y a donc pas à le vérifier. Les propriétés d un tel arbre sont très similaires à ceux des arbres binaires de recherche usuels. Pour tout nœud N(g, (i, s), d) de l arbre, on a : g et d sont des ABRI ; i s ; tous les intervalles de g ont des bornes strictement inférieures à i ; tous les intervalles de d ont des bornes strictement supérieures à s. On note qu un intervalle peut avoir des bornes égales (cas d un processus qui ne prend qu un seul cycle pour s exécuter). Ainsi l arbre suivant est un ABRI. [10; 13] [5; 5] [17; 28] [2; 4] [8; 9] 5

Exercice 5 : Manipulations simples (5 points) 1. Dans l ABRI ci-dessus, peut-on insérer les intervalles suivants? Si oui placez-les sur la figure, sinon expliquez rapidement pourquoi. [7; 8] [14; 15] [7; 8] n a pas sa place car il n est pas disjoint avec [8; 9] [14; 15] doit être inséré à gauche de [17; 28]. 2. Construisez graphiquement l ABRI obtenu en insérant successivement dans un arbre vide les intervalles [16; 18], [25; 30], [32; 36], [11; 11], [13; 14], [19; 22]. [16; 18] [11; 11] [25; 30] [13; 14] [19; 22] [32; 36] Exercice 6 : Accesseurs (10 points) 1. Définir une fonction inserable déterminant si un intervalle donné est insérable dans un ABRI. let rec inserable (i, s) a = match a with F true N (g, (inf, sup), d) if s < inf then inserable (i, s) g else if i > sup then inserable (i, s) d else false 2. Définir une fonction localiser : int -> abri -> intervalle option déterminant si un entier appartient à un intervalle de l arbre, et si oui lequel. 6

let rec localiser t a = match a with F None N (g, (inf, sup), d) if t < inf then localiser t g else if t > sup then localiser t d else Some (inf, sup) Exercice 7 : Insertion (15 points) 1. Définir une fonction insere permettant d insérer un intervalle dans un ABRI, si cela est possible. Dans le cas où l intervalle ne peut pas être inséré, on renverra l arbre passé en argument sans modification. let rec insere (i, s) a = match a with F N (F, (i, s), F) N (g, (inf, sup), d) if s < inf then N (insere (i, s) g, (inf, sup), d) else if i > sup then N (g, (inf, sup), insere (i, s) d) else N (g, (inf, sup), d) 2. On note nb noeuds(a) le nombre de nœuds d un arbre a. Démontrer que pour tout intervalle (i, s) et pour tout arbre a nb noeuds(a) nb noeuds(insere (i,s) a) nb noeuds(a) + 1 Par récurrence structurelle sur a : Si a = F alors insere (i,s) a possède 1 nœud et a en possède 0 : 0 1 0+1. Si a = N(g, (inf, sup), d) alors trois cas sont possibles : s < inf alors par HR l appel récursif insere (i,s) g produit un arbre dont le nombre de nœuds n est compris entre nb noeuds(g) et nb noeuds(g) + 1. L arbre que l on renvoie possède n + nb noeuds(d) + 1 nœuds, ce qui est donc bien compris entre nb noeuds(a) et nb noeuds(a) + 1. sup < i est similaire. inf s i sup d où il résulte que (i, s) n est pas insérable dans a. On renvoie alors a, la propriété est triviale dans ce cas. 7

Exercice 8 : Optimisation (20 points) 1. Écrire deux fonctions min et max déterminant la plus petite (respectivement la plus grande) borne de tous les intervalles présents dans un ABRI. let rec min a = match a with F failwith "Arbre vide" N (F, (inf, ), ) inf N (g, (, ), ) min g let rec max a = match a with F failwith "Arbre vide" N (, (, sup), F) sup N (, (, ), d) max d 2. Définir une fonction déterminant si un arbre d intervalles donné vérifie les propriétés des ABRI. Commenter l efficacité de votre solution. let rec verif a = match a with F true N (g, (inf, sup), d) verif g verif d (g = F max g < inf ) inf sup (d = F sup < min d) Les appels à min et max sont redondants, on calcule plusieurs fois la même chose : il faudrait les calculer à la volée. 3. Dans le cas où les tâches à effectuer sont indépendantes d autres tâches, on peut vouloir les effectuer le plus tôt possible. Définir une fonction calculant l ABRI obtenu en tassant les intervalles à gauche : l intervalle ayant les bornes les plus petites ne doit pas être modifié ; l intervalle suivant doit être décalé vers la gauche pour venir se juxtaposer au premier ; et ainsi de suite. Ainsi, avec l arbre donné en exemple en début de problème, les intervalles [2; 4] et [5; 5] sont inchangés, puis les suivants deviennent [6; 7], [8; 11] et [12; 23]. 8

let tasser a = let rec tasser min a m = match a with F F N (g, (inf, sup), d) let gg = tasser min g m in let mm = if gg = F then m 1 else max gg in N (gg, (mm + 1, sup inf + mm + 1), tasser min d (sup inf + mm + 2)) in tasser min a (min a) Exercice 9 : Analyse Syntaxique (10 points) On définit la grammaire suivante pour représenter des ABRI sous forme de flots de caractères (sans espaces) : S ::= () (S[int; int]s) La chaîne de caractères () représente l arbre vide ; dans le second cas, le non-terminal S situé entre la parenthèse et le crochet ouvrants représente le sous-arbre gauche et celui situé entre la parenthèse et le crochet fermants représente le sous-arbre droit. En pratique, on utilisera plutôt la grammaire suivante, équivalente mais non ambiguë : S ::= (S S ::= ) S[int; int]s) Écrire une fonction qui lit un flot de caractères respectant cette grammaire et l interprète comme un ABRI. Votre fonction ne réalisera que la construction de l arbre et ne s occupera pas de vérifier qu il respecte les propriétés d ABRI. On rappelle deux fonctions vues en TD/TP que vous êtes autorisés à utiliser : let int of digit c = int of char c int of char 0 let rec horner i = parser [ 0.. 9 as c ; n = horner (10 i + int of digit c) ] n [ ] i let rec abri parser = parser [ ( ; s ] apaux s and apaux = parser [ ) ] F [ g = abri parser ; [ ; inf = horner 0 ; ; ; sup = horner 0 ; ] ; d = abri parser ; ) ] N (g, (inf, sup), d) 9