PCF reprend les types du lambda-calcul simplement typé en y adjoignant un type représentant comme types de base les booléens, , et un type représentant les entiers, [1],[3]. Les types sont donc , , ou avec et des types déjà formés. Ce type représente le type des fonctions de vers . La notation doit être lue comme .
De plus, la syntaxe de PCF comprend la syntaxe du lambda-calcul simplement typé, et les constructions de base y sont les mêmes : il y a des variables, notées ; si est un terme de type et un terme de type , est un terme de type qui représente l'évaluation de la fonction en ; et si est un terme de type dans un contexte où la variable est de type , est de type et représente la fonction . Si est un terme de type , on notera . Pour plus de détails, consulter les sections Syntaxe et Règles de typage de l'article sur le lambda-calcul simplement typé.
À cela se rajoutent les constantes propres permettant de manipuler les types de base. Pour chaque entier naturel , on ajoute une constante . De plus, il y a une constante pour chaque booléen, et . Pour chaque type de base , c'est-à-dire et , on a une constante représentant une instruction conditionnelle : l'idée étant que si vaut , l'expression vaut , si vaut , elle s'évalue en . De plus, il y a deux fonctions pour manipuler les entiers, la fonction successeur et la fonction prédécesseur , qui correspondent à ajouter ou soustraire à leur argument. Enfin, il y a une fonction qui renvoie si son argument vaut l'entier , sinon.
Enfin, le principal ingrédient de PCF est sa construction de point fixe : pour chaque type , on dispose d'une constante dont l'interprétation est qu'il renvoie le plus petit point fixe de la fonction qu'on lui passe en argument. Certains auteurs[4] rajoutent à chaque type une constante qui répresente un programme qui ne termine pas. Cette construction n'est pas incluse dans la présentation originelle de Gordon Plotkin[1], mais peut y être définie comme .
Sémantique opérationnelle
PCF peut être muni d'une sémantique opérationnelle[5]. Si et sont des termes, on a si se transforme en après une étape élémentaire de calcul. On note pour dire que se transforme en en zéro ou plusieurs étapes. La relation est définie comme suit :
: pour évaluer l'application de la fonction en , on remplace par dans ;
: cela exprime que est un point fixe de ;
;
;
;
;
;
;
Les règles contextuelles sont les suivantes : si alors , , , et . Elles donnnent à PCF une sémantique d'appel par nom.
Cette sémantique est déterministe, c'est-à-dire que si et , alors . De plus, elle préserve le typage : si et que est de type , alors aussi.
De plus, on pourrait autoriser la réduction dans tous les contextes, plutôt qu'uniquement dans ceux précisés dans la dernière règle (qui interdit, par exemple, la réduction avec et qui n'est pas une lambda-abstraction ou ou ou ). Dans ce cas, la réduction obtenue préserve encore le typage et est confluente. De plus, si avec cette réduction, et si est en forme normale, alors avec la version restreinte. En clair, la sémantique opérationnelle définie ici est standardisante[6].
Exemple
PCF permet par exemple de définir des programmes effectuant l'addition de deux entiers. En voici un[6] :
.
Si on note la fonction à l'intérieur de , la somme de deux et trois est calculée par la suite de réductions suivante :
.
Ainsi, se réduit bien en la valeur .
Sémantique dénotationnelle
La sémantique opérationnelle présentée plus haut permet de déterminer comment exécuter un terme de PCF comme un programme. Mais pour étudier PCF, il peut être interéssant de traduire un terme comme en une fonction au sens mathématique du terme. C'est le rôle de la sémantique dénotationnelle[7]. Mais contrairement au lambda-calcul simplement typé, on ne peut pas interpréter PCF directement dans les fonctions et les ensembles, puisqu'il existe des fonctions qui n'ont pas de point fixe. On va donc interpréter les termes de PCF comme des fonctions continues entre cpo. Un cpo est un ensemble ordonné dans lequel toutes les parties dirigées ont une borne supérieure, et qui possède un plus petit élément[8]. L'intérêt des cpo réside dans ce cas chaque fonction continue possède un plus petit point fixe. La catégorie des cpo est cartésienne fermée[9].
Modèle continu
Le modèle continu est l'interprétation usuelle de PCF[10],[11]. On va définir pour chaque type, chaque contexte et chaque terme une traduction telle que l'interprétation d'un contexte ou d'un type soit un cpo, et si dans PCF, alors est une fonction continue de l'interprétation du contexte vers l'interprétation du type de .
Les types de base sont interprétés comme des domaines plats : on a et , où est l'ensemble des entiers naturels, l'ensemble des booléens et pour tout ensemble , est le cpo défini sur l'ensemble auquel on adjoint un élément , avec l'ordre défini par pour tout , et les éléments de sont incomparables entre eux. Le type est interprété comme l'ensemble des fonctions continues de vers . Un contexte est interprété par . On remarque que chaque type est un cpo, donc a un plus petit élément .
Considérons maintenant un terme typé et son contexte , avec .
L'interprétation des constructions issues du lambda-calcul est la suivante :
.
.
.
L'opérateur de point fixe est définie comme suit :
est la fonction qui à chaque associe son plus petit point fixe. Il est défini comme , c'est-à-dire comme le supremum de la suite croissante .
Puisque intuitivement, , et que le plus petit point fixe de la fonction identité est , .
Les primitives sur les entiers et les booléens sont interprétées comme suit, l'idée étant que représente une valeur non déterminée, par exemple un programme qui ne termine pas :
si et .
si et , et .
, si et , et .
, et .
Modèles standards
Plus généralement, on peut interpréter PCF dans des catégories cartésiennes fermées enrichies dans les cpo. Un tel modèle est appelé modèle standard de PCF. Une catégorie cartésienne fermée enrichie dans les cpo est une catégorie cartésienne fermée telle que :
l'évaluation et la composition sont strictes : pour tous objets , et et tout morphisme , si l'on désigne par le plus petit élément d'un cpo , on a :
,
et si l'on désigne par l'objet des fonctions de vers dans la catégorie cartésienne fermée, par l'opération de pairage et par le morphisme d'évaluation, on a :
.
la catégorie possède deux objets et tels que soit isomorphe à et soit isomorphe à en tant que cpo, où l'objet terminal de la catégorie . Dans la catégorie des cpo, est l'ensemble à un élément muni de l'égalité comme relation d'ordre.
Dans ce cas, en interprétant dans les constructions issues du lambda-calcul simplement typé de la façon uselle, en définissant et et en interprétant les constructions sur les types primitifs de la même façon que dans le modèle continu en posant et , ce qui est rendu possible par les isomorphismes et , on obtient la sémantique désirée.
Expressivité
Turing-complétude
Si est un programme dans PCF qui prend entiers en entrée et renvoie un entier, on peut montrer assez facilement que détermine une fonction calculable par si et n'est pas définie sinon.
Réciproquement, étant donnée une fonction calculable , on peut se demander si elle est représentée par un terme de PCF tel que pour tous entiers et ,
.
Ce problème correspond à la notion de complétude au sens de Turing. La réponse est positive, donc PCF peut représenter toutes les fonctions calculables entre les entiers[12].
Démonstration
On sait que les fonctions calculables sont obtenues en ajoutant le schéma de minimisation non bornée aux fonctions primitives récursives, et on peut assez facilement montrer que toutes les fonctions primitives récursives sont encodables dans PCF.
Il reste donc à montrer que PCF supporte l'opération de minimisation non bornée : étant donné un terme , est le plus entier tel que et pour s'il existe, et ne normalise pas sinon.
Pour cela, on définit
par . On vérifie facilement que vérifie bien la propriété désirée.
↑(en) Dana S. Scott, « A type-theoretical alternative to ISWIM, CUCH, OWHY », Theoretical Computer Science, vol. 121, no 1, , p. 411–440 (ISSN0304-3975, DOI10.1016/0304-3975(93)90095-B, lire en ligne [PDF], consulté le ) — distribué à l'origine comme des notes non publiées d'un séminaire donné à Oxford en 1969 sous le nom A theory of computable functions of higher type.