Common Lisp Object SystemLe Common Lisp Object System, souvent abrégé en CLOS (prononcer « si-lauss »), est l'ensemble des primitives présentes dans le langage de programmation Common Lisp pour construire un programme orienté objet. Il existe également une version de CLOS pour le langage Scheme, nommée TinyClos. PrésentationCLOS est un système objet à classes (il existe des systèmes à prototypes). Il descend de Flavors et Common Loops, développés dans les années 1980. Common Lisp a été le premier langage à objet standardisé par l'ANSI, en décembre 1994, précédant de peu Ada 95 (), SmallTalk et C++ (tous deux en 1998). Les objets et LispD'un certain point de vue, Lisp est un langage orienté objet depuis le début : les structures de données manipulées ont une identité et sont réflexives. La seule chose qui lui manquait pour recevoir l'estampille orienté objet, est la capacité d'étendre le système de types de Lisp. Caractéristiques
CLOS est construit autour de deux axes : les classes et les fonctions génériques. Les classes de CLOS sont relativement similaires à celles des autres langages à objet. Ce sont des groupes d'attributs. La spécialisation par héritage multiple est permise. Il est possible de définir des accesseurs et des mutateurs particuliers pour chaque attribut. Contrairement à la plupart des systèmes à objets classiques, les méthodes ne sont pas définies dans l'espace de nom des classes. Elles appartiennent à des fonctions génériques, qui sont des objets distincts. Cela permet de réduire le couplage a priori entre types de données et opérations sur ces types. Cela autorise naturellement l'extension de la sélection dynamique de méthode à l'ensemble des arguments obligatoires, et non pas au premier paramètre privilégié (appelé généralement le « receveur » de l'appel de méthode). Expliquons ces deux points. Fonctions génériquesDes classes « conteneurs » peuvent être définies dans des bibliothèques ou des frameworks indépendants, et cependant, certains concepts génériques s'appliquent à toutes ces classes : nombre d'éléments, représentation sérialisée pour en prendre deux. Ces concepts génériques sont, dans des langages comme Java ou C++, obligatoirement fournis par héritage (interface en Java, mixin en C++), pour satisfaire le système de types statique. Si les concepteurs ont oublié de fournir une interface pour une telle fonctionnalité générique, il n'y a rien à faire (en pratique, on doit contourner le problème par des procédés alourdissant le code, par exemple écrire ses propres sous-classes du framework disposant de la fonctionnalité voulue, mais ce n'est pas toujours possible). Des langages dynamiques comme Python, permettent le « monkey-patching », c'est-à-dire l'ajout post hoc de méthodes à des classes existantes ; mais c'est une pratique déconseillée et posant des problèmes de maintenabilité. La solution de CLOS à ce problème est la notion de fonction générique indépendante du site de définition et de l'espace de nom d'une classe associée. Ainsi on peut définir une fonction Sélection multiplePuisque les méthodes sont groupées dans des fonctions génériques, il n'y a aucune raison de considérer que le premier argument est « privilégié » (comme le Exemple avec des types numériques. L'opération d'addition, class int: defmethod add(self, y): if isinstance(y, int): ... elif isinstance(y, float): ... Avec seulement deux types numériques En dissociant classes et méthodes, on se permet donc d'ajouter après coup autant de classes que nécessaires, de définir des méthodes non prévues initialement, et finalement de fournir de nouvelles implémentations de méthodes (fonctions génériques en fait) après coup, de façon non intrusive. class int: ... class float: ... defmethod add(x: int, y: int): ... defmethod add(x: int, y: float): ... defmethod add(x: float, y: int): ... defmethod add(x: float, y: float): ... Et après coup : defmethod product(x: int, y: int): ... Il faut naturellement produire l'ensemble des combinaisons pertinentes pour le domaine d'application : class rational: ... defmethod add(x: int, y: rational): ... |