« PostScript » : différence entre les versions

Un article de Wikipédia, l'encyclopédie libre.
Contenu supprimé Contenu ajouté
SniperMaské (discuter | contributions)
m →‎Le langage : gram. + syntaxe + ponct.
Verbex (discuter | contributions)
m Annulation de la modification de 176.28.248.58 (d)
Balise : Annulation
 
(47 versions intermédiaires par 33 utilisateurs non affichées)
Ligne 1 : Ligne 1 :
{{Infobox Langage de programmation
{{Multi-infobox
|1={{Infobox Langage de programmation
| nom = PostScript
| nom = PostScript
| logo =
| logo =
| paradigme = [[Programmation multiparadigme|multiparadigme]] : [[orienté pile]], [[Programmation procédurale|procédurale]]
| paradigme = [[Programmation multiparadigme|multiparadigme]] : [[orienté pile]], [[Programmation procédurale|procédural]]
| année = [[1982]]
| date de première version = [[1982]]
| auteur = [[John Warnock]] & [[Chuck Geschke]]
| auteur1 = [[John Warnock]] |auteur2= [[Chuck Geschke]]
| développeur = [[Adobe Systems]]
| développeur = [[Adobe Systems]]
| dernière version stable= PostScript 3 (1997)
| dernière version = 3
| date de dernière version stable =
| date de dernière version = 1997
| typage = dynamique, fort
| typage = dynamique, fort
| implémentations = [[Adobe PostScript]], [[TrueImage]], [[Ghostscript]]
| implémentations = [[Adobe PostScript]], [[TrueImage]], [[Ghostscript]]
Ligne 18 : Ligne 17 :
| site web =
| site web =
}}
}}
|2={{Infobox Format de données
{{Infobox Format de données
| nom = PostScript
| nom = PostScript
| image =
| extension = .ps
| mime = application/postscript
| développeur = [[Adobe Systems]]
| genre = format de fichier d'impression
| genre = format de fichier d'impression
| conteneur de =
| contenu par =
| extension de =
| origine de = [[Encapsulated PostScript]]
| origine de = [[Encapsulated PostScript]]
| standard =
}}
}}
}}


Le '''PostScript''' est un [[langage de description de page]] mis au point par [[Adobe Systems|Adobe]]. Il repose sur des formulations [[vecteur|vectorielles]] de la plupart de ses éléments. Il sait aussi traiter les [[image matricielle|images matricielles]] (en mode point).
'''PostScript''' est un [[langage de description de page]] mis au point par [[Adobe Systems|Adobe]]<ref>{{Lien web|langue=en|titre=How Adobe Became a Successful $95 Billion SaaS Company|url=https://producthabits.com/adobe-95-billion-saas-company/|site=Product Habits|date=2018-01-15|consulté le=2019-03-14}}</ref>{{,}}<ref name=LM2021>{{article | langue=fr | titre=Charles Geschke, pionnier de l’informatique et cofondateur d’Adobe, est mort | auteur1=Nicolas Six | périodique=[[Le Monde]] | jour=19 | mois=avril | année=2021 | url texte=https://www.lemonde.fr/pixels/article/2021/04/19/charles-geschke-pionnier-de-l-informatique-et-cofondateur-d-adobe-est-mort_6077319_4408996.html}}</ref>. Il repose sur des formulations [[vecteur|vectorielles]] de la plupart de ses éléments. Il sait aussi traiter les [[image matricielle|images matricielles]] (point par point).

Ce langage inter-plateformes permet d'obtenir un [[Fichier (informatique)|fichier]] unique comportant tous les éléments décrivant la page (textes, images, polices, couleurs, etc.).


PostScript est devenu pratiquement un standard, la plupart des [[imprimante]]s laser contiennent un interpréteur et peuvent traiter directement un document décrit selon ce format. Sur les autres, il faut utiliser un filtre logiciel en entrée pour convertir le langage PostScript au [[Raster Image Processor|format raster]] compréhensible par ces imprimantes.
Ce langage entre plates-formes permet d'obtenir un [[Fichier (informatique)|fichier]] unique comportant tous les éléments décrivant la page (textes, images, polices, couleurs, etc.).


Le développement du PostScript est arrêté par Adobe depuis 2007, afin que le [[PDF]] puisse prendre la relève.
PostScript est devenu pratiquement un standard, la plupart des imprimantes laser haut de gamme peuvent traiter directement le format PostScript. Sur les autres ou les plus anciennes, il fallait utiliser un filtre logiciel en entrée pour [[Raster Image Processor|convertir le langage PostScript au format raster]] compréhensible par les anciennes imprimantes.
Le développement du PostScript est arrêté par Adobe depuis 2007, afin que le PDF puisse prendre la relève.


D'autres descendants de PostScript sont :
D'autres descendants de PostScript sont :
* [[Display PostScript]] et [[NeWS]], tous deux concernant les systèmes de fenêtrage ;
* [[Display PostScript]] et [[NeWS]], tous deux concernant les systèmes de fenêtrage ;
* [[Encapsulated PostScript|EPS (pour Encapsulated PostScript)]], format d'image, basé sur PostScript.
* [[Encapsulated PostScript]], format d'image basé sur PostScript.


== Le langage ==
== Le langage ==


Postscript est un langage complet, qui permet le codage de tout [[Algorithmique|algorithme]]. Bien qu'il soit tout à fait possible d'écrire directement de tels programmes, ils sont en général fabriqués par d'autres programmes, des pilotes d'impression par exemple.
Postscript est un langage complet, qui permet le codage de tout [[Algorithmique|algorithme]]. Bien qu'il soit tout à fait possible d'écrire directement de tels programmes<ref name="BlueBook">{{ouvrage|auteur1=Charles Geschke|auteur2=John Warnock|lien auteur2=John Warnock|titre=PostScript Language - Tutorial and Cookbook|éditeur=Addison & Wesley Publ. |année=1985 |isbn=0-201-10179-3 |langue=en}}</ref>, ils sont en général fabriqués par d'autres programmes, des pilotes d'impression par exemple.


Le Postscript est indissociable de l'environnement dans lequel il sera exécuté. Étant donné le caractère totalement dynamique de ce langage, il sera en fait [[Langage interprété informatique|interprété]].
Postscript est indissociable de l'environnement dans lequel il sera exécuté. Étant donné le caractère totalement dynamique de ce langage, il sera en fait [[Langage interprété informatique|interprété]].


L'interpréteur est composé d'une pile, et d'un ou plusieurs dictionnaires.
L'interpréteur fonctionne en manipulant une pile et plusieurs dictionnaires.
La pile sert de stockage temporaire pour les paramètres des fonctions, puis pour leurs résultats.
La pile sert de stockage temporaire pour les paramètres des fonctions, puis pour leurs résultats.
Les dictionnaires permettent le stockage des variables, ainsi que du code des fonctions.
Les dictionnaires permettent le stockage des variables, ainsi que le code des fonctions.


Un programme PostScript est composé d'une séquence de mots séparés par des espaces, tabulation (TAB), [[retour chariot]] (CR), avance de ligne (LF), ou commentaires. L'interpréteur analyse chaque mot du programme PostScript séquentiellement en fonctionnant comme un calculateur en [[notation polonaise inverse]], c’est-à-dire que chaque mot est évalué, puis le (ou les) résultat de cette évaluation est placé au sommet de la pile, et ainsi de suite.
Un programme PostScript est composé d'une séquence de mots séparés par des espaces, tabulation (TAB), [[retour chariot]] (CR), avance de ligne (LF), ou commentaires. L'interpréteur analyse chaque mot du programme PostScript séquentiellement en fonctionnant comme un calculateur en [[notation polonaise inverse]], c’est-à-dire que chaque mot est évalué, puis le (ou les) résultat(s) de cette évaluation est placé au sommet de la pile, et ainsi de suite.


Exemple, pour effectuer le calcul numérique simple ''b''{{2}} - 4''ac'', cela pourra se coder :
Exemple, pour effectuer le calcul numérique simple ''b''{{2}} - 4''ac'', cela pourra se coder :
: <code>b b mul 4 a mul c mul sub</code>
{{retrait|1= <code>b b mul 4 a mul c mul sub</code> }}


PostScript supporte cinq types de mots :
PostScript utilise les cinq types de mots suivants  :
* constante numérique entière ou réelle : <code>123</code> ou <code>3.14159</code>, l’évaluation ajoute la valeur numérique au sommet de la pile ;
; Constante numérique : entière (<code>123</code>) ou réelle (<code>3.14159</code>), l’évaluation ajoute la valeur numérique au sommet de la pile ;
* constante chaîne : ce sont des tableaux de caractères encadrés par des parenthèses, ex <code>(Abc)</code> représente la chaîne « Abc » ;
; Constante chaîne : ce sont des tableaux de caractères encadrés par des parenthèses, ex <code>(Abc)</code> représente la chaîne « Abc » ;
* référence à un nom : <code>/a</code> représente le nom « a », qui pourra permettre de nommer une variable ou une fonction ; la référence est ajoutée au sommet de la pile ;
; Référence à un nom : <code>/a</code> représente le nom « a », qui pourra permettre de nommer une variable ou une fonction ; la référence est ajoutée au sommet de la pile ;
* nom : ce peut être le nom d'une procédure prédéfinie ou créée, d'une variable, d'un dictionnaire ; le nom est cherché dans les dictionnaires actifs, puis si c’est une valeur, elle est placée sur la pile, et si c’est une fonction, elle est appelée (exécutée) ; l'évaluation d’une fonction pourrait utiliser les valeurs au sommet de la pile et les remplacer par les éventuels résultats ;
; Nom : ce peut être le nom d'une procédure prédéfinie ou créée, d'une variable, d'un dictionnaire ; le nom est cherché dans les dictionnaires actifs, puis si c’est une valeur, elle est placée sur la pile, et si c’est une fonction, elle est appelée (exécutée) ; l'évaluation d’une fonction pourrait utiliser les valeurs au sommet de la pile et les remplacer par les éventuels résultats ;
* constructeur : quatre constructeurs permettent de créer des structures de données de taille variable, ils sont composés de caractères appariés suivants, qui délimitent le début et la fin de la structure :
; Constructeur : quatre constructeurs permettent de créer des structures de données de taille variable. Ils sont composés de caractères appariés qui délimitent le début et la fin de la structure :
*Caractères de délimitation :
** <code>[</code> et <code>]</code> : des tableaux quelconques ;
** <code>&lt;</code> et <code>&gt;</code> : des tableaux d’octets codés en hexadécimal ;
** <code>[</code> et <code>]</code> : pour un tableau quelconque ;
** <code>&lt;&lt;</code> et <code>&gt;&gt;</code> : des dictionnaires ;
** <code>&lt;</code> et <code>&gt;</code> : pour un tableau d’octets codés en hexadécimal ;
** <code>{</code> et <code>}</code> : du code exécutable.
** <code>&lt;&lt;</code> et <code>&gt;&gt;</code> : pour un dictionnaire ;
** <code>{</code> et <code>}</code> : pour du code exécutable.
: La première partie du constructeur place une marque sur la pile, puis les éléments de cette structure sont placés successivement sur la pile, enfin la partie droite du constructeur récupère tous les éléments depuis la dernière marque dans la pile et « fabrique » la structure correspondante et la met sur la pile.
: L'ouverture du constructeur place une marque sur la pile, puis, successivement, tous les éléments de cette structure. La fermeture du constructeur récupère tous les éléments depuis la marque d'ouverture dans la pile, « fabrique » la structure correspondante et la place sur la pile.


Les dictionnaires sont des tableaux avec seulement deux colonnes :
Les dictionnaires sont des tableaux avec seulement deux colonnes :
* la première colonne peut contenir une valeur utilisée comme index, ou une référence à un nom ou à tout autre objet, cependant le nom ou la valeur est unique dans cette colonne pour toutes les lignes du tableau ;
* La première colonne contient une valeur utilisée comme index, qui peut être une référence à un nom ou à tout autre objet. Cet index devant être univoque, on ne peut le rencontrer sur deux lignes distinctes.
* et la seconde, une valeur quelconque associée à ce nom ou cette valeur.
* La seconde ligne contient une valeur quelconque, l'associant ainsi à l'index.
Avec cette propriété, un dictionnaire fonctionne donc comme un tableau à une colonne, mais dont l'index peut être de valeur ou de type quelconque, et n'est pas restreint à un simple intervalle d'entiers.


Avec cette propriété, un dictionnaire fonctionne donc comme un tableau à une colonne, mais dont l'index peut être de type quelconque, sans être restreint à un simple intervalle d'entiers.
Le dictionnaire est géré comme une [[table de hachage]] (grâce à une fonction de hachage prédéfinie par le langage lui-même) et on peut en prédéfinir lors de sa création la taille initiale en fonction du nombre de couples nom-valeur qu'on veut y stocker, pour limiter les collisions (cependant PostScript sait redimensionner dynamiquement un dictionnaire en fonction du nombre de collisions sur les noms, ou de son taux de remplissage). Les dictionnaires de PostScript servent principalement (mais pas seulement) à définir (dynamiquement) la [[Portée (informatique)|portée]] des variables nommées et référencées ailleurs dans le langage.


Le dictionnaire est géré comme une [[table de hachage]] (grâce à une fonction de hachage prédéfinie par le langage lui-même), dont on peut définir à la création la taille initiale, en fonction du nombre de couples nom-valeur qu'on veut y stocker, afin de limiter les occurrences de collisions (cependant PostScript sait redimensionner dynamiquement un dictionnaire en fonction du nombre de collisions sur les noms, ou de son taux de remplissage). Les dictionnaires de PostScript servent principalement (mais pas seulement) à définir (dynamiquement) la [[Portée (informatique)|portée]] des variables nommées et référencées ailleurs dans le langage.
Parmi les [[opérateur (informatique)|opérateurs]] prédéfinis, le plus important est <code>def</code>, il permet d'ajouter une association nom-valeur dans le dictionnaire courant, ce qui permet de définir de nouvelles variables avec leur valeur, ou de modifier leur valeur, et de définir ou redéfinir des fonctions. On constate donc, qu’en Postscript, le code exécutable est une donnée, presque comme les autres, et peut être créé à la volée, modifié.

Parmi les [[Opérateur (informatique)|opérateurs]] prédéfinis, le plus important est <code>def</code>, il permet d'ajouter une association nom-valeur dans le dictionnaire courant, ce qui permet de définir de nouvelles variables avec leur valeur, ou de modifier leur valeur, et de définir ou redéfinir des fonctions. On constate donc qu’en Postscript, le code exécutable est une donnée presque comme les autres, et peut être créé à la volée, modifié.


Exemples :
Exemples :
Ligne 86 : Ligne 79 :
* <code>/compteur compteur incremente def</code> : recherche la valeur associée à la variable nommée « compteur » dans le dictionnaire actuel et remplace ce nom par sa valeur, puis cette valeur est incrémentée grâce à la fonction définie au-dessus, et le résultat est stocké (par la fonction « def ») dans la variable nommée par « /compteur » dans le dictionnaire actuel. Autrement dit, la variable nommée « compteur » dans la portée d'un dictionnaire (en commençant la recherche par le dictionnaire actuel) sera incrémentée.
* <code>/compteur compteur incremente def</code> : recherche la valeur associée à la variable nommée « compteur » dans le dictionnaire actuel et remplace ce nom par sa valeur, puis cette valeur est incrémentée grâce à la fonction définie au-dessus, et le résultat est stocké (par la fonction « def ») dans la variable nommée par « /compteur » dans le dictionnaire actuel. Autrement dit, la variable nommée « compteur » dans la portée d'un dictionnaire (en commençant la recherche par le dictionnaire actuel) sera incrémentée.


Dans le dernier exemple ci-dessus, rien n'indique que la variable nommée « compteur » sera la même que celle dont on a extrait la valeur. En effet, « def » sert à stocker une association nom-valeur uniquement dans le dictionnaire actuel, et aucun des autres dictionnaires dans la pile de portée. Or, la lecture de la variable compteur (la seconde référence dans le code ci-dessus) peut retourner la valeur d'une variable trouvée dans autre dictionnaire que le dictionnaire actuel (ou produire une exception à l'exécution si aucun des dictionnaires dans la pile de portée ne contient une variable de ce nom) : dans ce cas, une nouvelle variable sera ajoutée par « def » dans le dictionnaire actuel, sans modifier la variable d'origine où elle a été trouvée et qui conservera donc sa valeur ; cependant tant que le dictionnaire courant sera actif, la nouvelle variable masquera l'ancienne. Ce dispositif permet donc de gérer des variables locales autrement que par une position relative dans la pile.
Dans le dernier exemple ci-dessus, rien n'indique que la variable nommée « compteur » sera la même que celle dont on a extrait la valeur. En effet, « def » sert à stocker une association nom-valeur uniquement dans le dictionnaire actuel, et dans aucun autre dictionnaire de la pile de portées. Or, la lecture de la variable compteur (la seconde référence dans le code ci-dessus) peut retourner la valeur d'une variable trouvée dans un autre dictionnaire que le dictionnaire actuel (ou produire une exception à l'exécution si aucun des dictionnaires dans la pile de portées ne contient une variable de ce nom) : dans ce cas, une nouvelle variable sera ajoutée par « def » dans le dictionnaire actuel, sans modifier la variable d'origine où elle a été trouvée, qui conservera donc sa valeur ; cependant tant que le dictionnaire courant sera actif, la nouvelle variable masquera l'ancienne. Ce dispositif permet donc de gérer des variables locales autrement que par une position relative dans la pile.


PostScript peut différencier les références à une variable (par son nom indiqué après un « <tt>/</tt> ») et celles à sa valeur. Cependant la référence n'est résolue dans aucune dictionnaire de portée tant qu'on ne l'a pas associée à un dictionnaire pour la rechercher. Quand un nom est utilisé sans « <tt>/</tt> » initial, il est immédiatement recherché lors de la compilation de la fonction dans les dictionnaires de portée actifs (en commençant par le dictionnaire courant lors de la compilation et non celui qui sera actif à l'exécution de la fonction), alors le nom de la variable est remplacé à l'exécution dans la pile par la valeur associée à ce nom dans le dictionnaire où le nom a été trouvé.
PostScript peut différencier les références à une variable (par son nom indiqué après un <code>/</code>) et celles à sa valeur. Cependant la référence n'est résolue dans aucun dictionnaire de portée tant qu'on ne l'a pas associée à un dictionnaire pour la rechercher. Quand un nom est utilisé sans <code>/</code> initial, il est immédiatement recherché lors de la compilation de la fonction dans les dictionnaires de portée actifs (en commençant par le dictionnaire courant lors de la compilation et non celui qui sera actif à l'exécution de la fonction), alors le nom de la variable est remplacé à l'exécution dans la pile par la valeur associée à ce nom dans le dictionnaire où le nom a été trouvé.


Aussi, pour créer une référence complète à une variable bien définie et non sa valeur ou une autre variable de même nom dans un autre dictionnaire, on doit indiquer non seulement le nom de cette variable, mais aussi une référence au dictionnaire qui la contient et où elle doit être recherchée. Si on ne référence pas le dictionnaire (par exemple en ne préfixant pas par un <tt>/</tt>), la résolution du nom est dynamique et peut donc référencer des variables différentes en fonction du contexte d'exécution.
Aussi, pour créer une référence complète à une variable bien définie et non sa valeur ou une autre variable de même nom dans un autre dictionnaire, on doit indiquer non seulement le nom de cette variable, mais aussi une référence au dictionnaire qui la contient et où elle doit être recherchée. Si on ne référence pas le dictionnaire (par exemple en ne préfixant pas par un <code>/</code>), la résolution du nom est dynamique et peut donc référencer des variables différentes en fonction du contexte d'exécution.


Postcript définit donc deux contextes distincts d'utilisation d'une référence à une variable :
Postcript définit donc deux contextes distincts d'utilisation d'une référence à une variable :
Ligne 99 : Ligne 92 :
=== Opérateurs ===
=== Opérateurs ===
{| border="0"
{| border="0"
| || || || ||''tout''||'''pop'''||—|| || || || || || ||élimine l'élément de la pile
| ||''quelconque''||'''pop'''||—|| || || || || || ||élimine l'élément de la pile
|-
|-
| || || ||''tout''<sub>1</sub>||''tout''<sub>2</sub>||'''exch'''||''tout''<sub>2</sub>||''tout''<sub>1</sub>|| || || || || ||échange les deux éléments
| quelconque<sub>1</sub>||quelconque<sub>2</sub>||'''exch'''||quelconque<sub>2</sub>||quelconque<sub>1</sub>|| || || || || ||échange les deux éléments
|-
|-
| || || || || ||'''…'''|| || || || || || || ||…
| || ||'''…'''|| || || || || || || ||…
|-
|-
| || || ||''nombre''<sub>1</sub>||''nombre''<sub>2</sub>||'''add'''||''somme''|| || || || || || ||renvoie ''nombre''<sub>1</sub> plus ''nombre''<sub>2</sub>
|''nombre''<sub>1</sub>||''nombre''<sub>2</sub>||'''add'''||''somme''|| || || || || || ||renvoie ''nombre''<sub>1</sub> plus ''nombre''<sub>2</sub>
|-
|-
| || || ||''nombre''<sub>1</sub>||''nombre''<sub>2</sub>||'''div'''||''quotient''|| || || || || || ||renvoie ''nombre''<sub>1</sub> divisé ''nombre''<sub>2</sub>
|''nombre''<sub>1</sub>||''nombre''<sub>2</sub>||'''div'''||''quotient''|| || || || || || ||renvoie ''nombre''<sub>1</sub> divisé ''nombre''<sub>2</sub>
|-
|-
| || || ||''entier''<sub>1</sub>||''entier''<sub>2</sub>||'''idiv'''||''quotient''|| || || || || || ||renvoie ''entier''<sub>1</sub> divisé ''entier''<sub>2</sub>
|''entier''<sub>1</sub>||''entier''<sub>2</sub>||'''idiv'''||''quotient''|| || || || || || ||renvoie ''entier''<sub>1</sub> divisé ''entier''<sub>2</sub>
|-
|-
| || || || || ||'''…'''|| || || || || || || ||…
| || ||'''…'''|| || || || || || || ||…
|}
|}


== Implémentation ==
== Implémentation ==
PostScript est sous licence de la société Adobe. Néanmoins il existe un interpréteur PostScript libre, [[Ghostscript]].
PostScript est fourni sous licence de la société Adobe. Néanmoins il existe un interpréteur PostScript libre, [[Ghostscript]].

== Notes et références ==
{{Références}}


== Voir aussi ==
== Voir aussi ==
Ligne 122 : Ligne 118 :
* [[Ghostscript]] - Interpréteur PostScript et PDF, libre, toutes plates-formes et ses interfaces graphiques : GSView, etc.
* [[Ghostscript]] - Interpréteur PostScript et PDF, libre, toutes plates-formes et ses interfaces graphiques : GSView, etc.
* [[Portable Document Format]]
* [[Portable Document Format]]
* [[Printer Postscript Description]]
* [[PostScript Printer Description]]
* [[Fontes PostScript]]
* [[Fonte PostScript]]
* [[ReGIS]]
* [[SVG]]
* [[SVG]]
* [[TeX]]
* [[TeX]], [[LaTeX]]
* [[Printer Command Language]]
* [[LaTeX]]
* [[PCL]]
* [[XML Paper Specification]] - Format de description de pages développé par Microsoft
* [[XML Paper Specification]] - Format de description de pages développé par Microsoft
* [[Forth (langage)|Forth]] - Langage de programmation basé sur les mêmes principes (pile, dictionnaire, notation polonaise inverse...)
* [[Forth (langage)|Forth]] - Langage de programmation basé sur les mêmes principes (pile, dictionnaire, notation polonaise inverse...)
Ligne 133 : Ligne 129 :
=== Liens externes ===
=== Liens externes ===
* {{en}} [http://www.adobe.com/products/postscript/resources.htm Adobe PostScript 3] sur le site d'Adobe
* {{en}} [http://www.adobe.com/products/postscript/resources.htm Adobe PostScript 3] sur le site d'Adobe
* {{en}} [http://www.tailrecursive.org/postscript/ ''{{Lang|en|A First Guide to PostScript}}'']


{{Portail|informatique}}
{{Portail|informatique}}
Ligne 139 : Ligne 134 :
[[Catégorie:Imprimerie]]
[[Catégorie:Imprimerie]]
[[Catégorie:Format de données numériques]]
[[Catégorie:Format de données numériques]]
[[Catégorie:Format de fichier graphique]]
[[Catégorie:Format de fichier graphique vectoriel]]
[[Catégorie:Typographie]]
[[Catégorie:Typographie]]
[[Catégorie:Langage de description de page]]
[[Catégorie:Langage de description de page]]

Dernière version du 16 avril 2024 à 08:53

PostScript
Logo.

Date de première version 1982
Paradigme multiparadigme : orienté pile, procédural
Développeur Adobe Systems
Dernière version 3 (1997)
Typage dynamique, fort
Influencé par Forth
A influencé PDF
Implémentations Adobe PostScript, TrueImage, Ghostscript
Extension de fichier psVoir et modifier les données sur Wikidata
PostScript
Caractéristiques
Extension
.psVoir et modifier les données sur Wikidata
Type MIME
application/postscriptVoir et modifier les données sur Wikidata
Développé par
Version initiale
Type de format
format de fichier d'impression
Origine de

PostScript est un langage de description de page mis au point par Adobe[1],[2]. Il repose sur des formulations vectorielles de la plupart de ses éléments. Il sait aussi traiter les images matricielles (point par point).

Ce langage inter-plateformes permet d'obtenir un fichier unique comportant tous les éléments décrivant la page (textes, images, polices, couleurs, etc.).

PostScript est devenu pratiquement un standard, la plupart des imprimantes laser contiennent un interpréteur et peuvent traiter directement un document décrit selon ce format. Sur les autres, il faut utiliser un filtre logiciel en entrée pour convertir le langage PostScript au format raster compréhensible par ces imprimantes.

Le développement du PostScript est arrêté par Adobe depuis 2007, afin que le PDF puisse prendre la relève.

D'autres descendants de PostScript sont :

Le langage[modifier | modifier le code]

Postscript est un langage complet, qui permet le codage de tout algorithme. Bien qu'il soit tout à fait possible d'écrire directement de tels programmes[3], ils sont en général fabriqués par d'autres programmes, des pilotes d'impression par exemple.

Postscript est indissociable de l'environnement dans lequel il sera exécuté. Étant donné le caractère totalement dynamique de ce langage, il sera en fait interprété.

L'interpréteur fonctionne en manipulant une pile et plusieurs dictionnaires. La pile sert de stockage temporaire pour les paramètres des fonctions, puis pour leurs résultats. Les dictionnaires permettent le stockage des variables, ainsi que le code des fonctions.

Un programme PostScript est composé d'une séquence de mots séparés par des espaces, tabulation (TAB), retour chariot (CR), avance de ligne (LF), ou commentaires. L'interpréteur analyse chaque mot du programme PostScript séquentiellement en fonctionnant comme un calculateur en notation polonaise inverse, c’est-à-dire que chaque mot est évalué, puis le (ou les) résultat(s) de cette évaluation est placé au sommet de la pile, et ainsi de suite.

Exemple, pour effectuer le calcul numérique simple b2 - 4ac, cela pourra se coder : b b mul 4 a mul c mul sub

PostScript utilise les cinq types de mots suivants  :

Constante numérique
entière (123) ou réelle (3.14159), l’évaluation ajoute la valeur numérique au sommet de la pile ;
Constante chaîne
ce sont des tableaux de caractères encadrés par des parenthèses, ex (Abc) représente la chaîne « Abc » ;
Référence à un nom
/a représente le nom « a », qui pourra permettre de nommer une variable ou une fonction ; la référence est ajoutée au sommet de la pile ;
Nom
ce peut être le nom d'une procédure prédéfinie ou créée, d'une variable, d'un dictionnaire ; le nom est cherché dans les dictionnaires actifs, puis si c’est une valeur, elle est placée sur la pile, et si c’est une fonction, elle est appelée (exécutée) ; l'évaluation d’une fonction pourrait utiliser les valeurs au sommet de la pile et les remplacer par les éventuels résultats ;
Constructeur
quatre constructeurs permettent de créer des structures de données de taille variable. Ils sont composés de caractères appariés qui délimitent le début et la fin de la structure :
  • Caractères de délimitation :
    • [ et ] : pour un tableau quelconque ;
    • < et > : pour un tableau d’octets codés en hexadécimal ;
    • << et >> : pour un dictionnaire ;
    • { et } : pour du code exécutable.
L'ouverture du constructeur place une marque sur la pile, puis, successivement, tous les éléments de cette structure. La fermeture du constructeur récupère tous les éléments depuis la marque d'ouverture dans la pile, « fabrique » la structure correspondante et la place sur la pile.

Les dictionnaires sont des tableaux avec seulement deux colonnes :

  • La première colonne contient une valeur utilisée comme index, qui peut être une référence à un nom ou à tout autre objet. Cet index devant être univoque, on ne peut le rencontrer sur deux lignes distinctes.
  • La seconde ligne contient une valeur quelconque, l'associant ainsi à l'index.

Avec cette propriété, un dictionnaire fonctionne donc comme un tableau à une colonne, mais dont l'index peut être de type quelconque, sans être restreint à un simple intervalle d'entiers.

Le dictionnaire est géré comme une table de hachage (grâce à une fonction de hachage prédéfinie par le langage lui-même), dont on peut définir à la création la taille initiale, en fonction du nombre de couples nom-valeur qu'on veut y stocker, afin de limiter les occurrences de collisions (cependant PostScript sait redimensionner dynamiquement un dictionnaire en fonction du nombre de collisions sur les noms, ou de son taux de remplissage). Les dictionnaires de PostScript servent principalement (mais pas seulement) à définir (dynamiquement) la portée des variables nommées et référencées ailleurs dans le langage.

Parmi les opérateurs prédéfinis, le plus important est def, il permet d'ajouter une association nom-valeur dans le dictionnaire courant, ce qui permet de définir de nouvelles variables avec leur valeur, ou de modifier leur valeur, et de définir ou redéfinir des fonctions. On constate donc qu’en Postscript, le code exécutable est une donnée presque comme les autres, et peut être créé à la volée, modifié.

Exemples :

  • /pi 3.14159 def : définit la variable de nom pi, avec la valeur numérique 3,14159 ;
  • /compteur compteur 1 add def : ajoute 1 à la variable nommée compteur ;
  • /incremente {1 add} def : définit la fonction incremente, qui ajoute 1 au paramètre ;
  • /compteur compteur incremente def : recherche la valeur associée à la variable nommée « compteur » dans le dictionnaire actuel et remplace ce nom par sa valeur, puis cette valeur est incrémentée grâce à la fonction définie au-dessus, et le résultat est stocké (par la fonction « def ») dans la variable nommée par « /compteur » dans le dictionnaire actuel. Autrement dit, la variable nommée « compteur » dans la portée d'un dictionnaire (en commençant la recherche par le dictionnaire actuel) sera incrémentée.

Dans le dernier exemple ci-dessus, rien n'indique que la variable nommée « compteur » sera la même que celle dont on a extrait la valeur. En effet, « def » sert à stocker une association nom-valeur uniquement dans le dictionnaire actuel, et dans aucun autre dictionnaire de la pile de portées. Or, la lecture de la variable compteur (la seconde référence dans le code ci-dessus) peut retourner la valeur d'une variable trouvée dans un autre dictionnaire que le dictionnaire actuel (ou produire une exception à l'exécution si aucun des dictionnaires dans la pile de portées ne contient une variable de ce nom) : dans ce cas, une nouvelle variable sera ajoutée par « def » dans le dictionnaire actuel, sans modifier la variable d'origine là où elle a été trouvée, qui conservera donc sa valeur ; cependant tant que le dictionnaire courant sera actif, la nouvelle variable masquera l'ancienne. Ce dispositif permet donc de gérer des variables locales autrement que par une position relative dans la pile.

PostScript peut différencier les références à une variable (par son nom indiqué après un /) et celles à sa valeur. Cependant la référence n'est résolue dans aucun dictionnaire de portée tant qu'on ne l'a pas associée à un dictionnaire pour la rechercher. Quand un nom est utilisé sans / initial, il est immédiatement recherché lors de la compilation de la fonction dans les dictionnaires de portée actifs (en commençant par le dictionnaire courant lors de la compilation et non celui qui sera actif à l'exécution de la fonction), alors le nom de la variable est remplacé à l'exécution dans la pile par la valeur associée à ce nom dans le dictionnaire où le nom a été trouvé.

Aussi, pour créer une référence complète à une variable bien définie et non sa valeur ou une autre variable de même nom dans un autre dictionnaire, on doit indiquer non seulement le nom de cette variable, mais aussi une référence au dictionnaire qui la contient et où elle doit être recherchée. Si on ne référence pas le dictionnaire (par exemple en ne préfixant pas par un /), la résolution du nom est dynamique et peut donc référencer des variables différentes en fonction du contexte d'exécution.

Postcript définit donc deux contextes distincts d'utilisation d'une référence à une variable :

  • dans le contexte de compilation d'une définition de fonction, la référence trouvée dans le dictionnaire courant (au moment de la compilation) est résolue immédiatement, mais la valeur n'est pas immédiatement déréférencée ; au lieu de cela, la référence est empilée et sera stockée ensuite dans le code au moment où la fonction sera effectivement définie ;
  • puis lorsque le code de la fonction est exécutée, toute référence de variable empilée s'exécute en la remplaçant dans par la valeur de la variable indépendamment du dictionnaire courant.

Ce comportement pour le déréferencement d'une variable (accès en lecture) est donc très différent de celui du stockage d'une variable avec def qui se fait toujours selon le dictionnaire courant au seul moment de l'exécution.

Opérateurs[modifier | modifier le code]

quelconque pop élimine l'élément de la pile
quelconque1 quelconque2 exch quelconque2 quelconque1 échange les deux éléments
nombre1 nombre2 add somme renvoie nombre1 plus nombre2
nombre1 nombre2 div quotient renvoie nombre1 divisé nombre2
entier1 entier2 idiv quotient renvoie entier1 divisé entier2

Implémentation[modifier | modifier le code]

PostScript est fourni sous licence de la société Adobe. Néanmoins il existe un interpréteur PostScript libre, Ghostscript.

Notes et références[modifier | modifier le code]

  1. (en) « How Adobe Became a Successful $95 Billion SaaS Company », sur Product Habits, (consulté le )
  2. Nicolas Six, « Charles Geschke, pionnier de l’informatique et cofondateur d’Adobe, est mort », Le Monde,‎ (lire en ligne)
  3. (en) Charles Geschke et John Warnock, PostScript Language - Tutorial and Cookbook, Addison & Wesley Publ., (ISBN 0-201-10179-3)

Voir aussi[modifier | modifier le code]

Articles connexes[modifier | modifier le code]

Liens externes[modifier | modifier le code]