Skip to content

Introduction à la programmation avec Python

Vous avez dit "programme" ?

Un programme est la description d’un algorithme dans un langage compréhensible par un humain et par une machine, qui l’exécute afin de traiter des données.

Il existe de nombreux langages de programmation, dont certains sont plus proches du langage naturel (on parle de langages de haut niveau), tandis que d’autres sont plus proches de celui de la machine (on parle de langages de bas niveau), on rappelle que la machine ne comprend que le binaire, c'est-à-dire une suite de bits 0 ou 1 (nous reviendrons sur cela dans le cadre d'un autre chapitre).

On peut citer, parmi les langages :

  • de haut niveau : Python, JavaScript, Java, C#,
  • de bas niveau : C, Assembleur, langage machine (le plus bas niveau possible).

Le langage de programmation Python, que l'on utilisera dans le cadre de cet enseignement, est déjà utilisé au lycée en mathématiques, et est également présent dans d’autres disciplines et dans le nouvel enseignement de sciences numériques et Technologie (SNT) en seconde.
Au-delà du cadre de l'enseignement, c'est un langage extrêmement populaire, car l’un des plus versatiles et généralistes. Il est utilisé aussi bien par des développeurs débutants que par des développeurs d’applications web et mobile, des ingénieurs logiciels, des Data Scientists...

Guido Van Rossum

Guido Van Rossum, créateur de Python
Source : Wikipédia

Le langage Python a été créé par un ingénieur informaticien néerlandais du nom de Guido Van Rossum.

La première version publique date de 1991. Van Rossum a ensuite poursuivi son travail sur le projet et a travaillé entre autres pour Google puis Dropbox. La version de Python que nous utiliserons est la version 3, disponible depuis 2008 avec des mises à jour régulières. La dernière version en date lors de l'écriture de cet article est la version 3.11.5.

Le langage Python est multiplateforme, vous pouvez donc l'installer sur les systèmes d’exploitation Linux, MacOs, Windows, et même sur des smartphones dotés du système Android ou d'iOS. Il est gratuit et placé sous licence libre, la Python Software Foundation License (PSFL) .

Les constructions élémentaires propes au langage Python sont communes à de nombreux autres langages de programmation.

Un programme est ainsi composé :

  • de séquences, (des instructions exécutées l’une après l’autre dans l’ordre où elles sont écrites),
  • de définitions de variables et de fonctions,
  • d’affectations,
  • d’instructions conditionnelles,
  • de boucles (bornées et non-bornées),
  • d'expressions et d'appels de fonctions.

Un IDE pour débuter

Il existe de nombreux environnements de développement (EDI, ou IDE en anglais) permettant de développer en Python. Celui que je vous conseille pour débuter est Thonny, cliquez-ici pour télécharger et découvrir l'IDE Thonny.

Expressions arithmétiques, variables et instructions

Le langage de programmation Python permet d'interagir avec la machine à l'aide d'un programme appelé interpréteur Python. On peut l'utiliser de deux manières différentes :

  • en mode interactif, qui consiste à dialoguer directement avec l'interpréteur,
  • en mode programme, qui consiste à écrire un programme dans un fichier et à le faire exécuter par l'interpréteur.

Mode interactif

Le mode interactif s'apparente à une calculatrice.
Les trois chevrons >>> indiquent que l'interpréteur attend vos instructions.

Par exemple, si vous saisissez 1 + 2 après les chevrons et que vous appuyez sur la touche Entrée, l'interpréteur effectuera le calcul et affichera le résultat :

>>> 1 + 2
3

À vous de jouer !

Saisissez dans l'interpréteur Python ci-dessous plusieurs expressions en utilisant les opérateurs + (addition), - (soustraction), * (multiplication) et / (division).

Arithmétique

En Python, on peut saisir des combinaisons d'opérations arithmétiques.

Par exemple :

>>> 2 + 5 * (10 - 1 / 2)
49.5

À noter que les espaces sont purement décoratifs, on aurait pu écrire :

>>> 2+6*(10-1/2)
49.5

Erreur de syntaxe

L'interpréteur n'accepte que les expressions arithmétiques bien formées. Autrement, une SyntaxError indiquant une erreur de syntaxe sera levée, par exemple :

>>> 1 + * 2
    File "<stdin>", line 1
    1 + * 2
        ^
SyntaxError: invalid syntax

<stdin> signifie standard input (entrée standard), on reviendra sur les notions d'entrée standard et de sortie standard plus loin dans ce cours lorsque l'on présentera les fonctions input et print.

Erreur division par zéro

Un autre type d'erreur qui peut être levée par l'interpréteur Python est une zeroDivisionError, indiquant que l'expression contient une division par zéro. Par exemple :

>>> 2 / (3 - 3)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ZeroDivisionError: division by zero

Les nombres de Python peuvent être des nombres entiers relatifs, de type int, et des nombres décimaux, que l'on appelle flottants, de type float.

Les nombres entiers peuvent être de taille arbitraire, et ne sont limités que par la mémoire disponible de la machine pour les stocker.
Les nombres flottants quant à eux ont une capacité limitée, et les nombres décimaux très grands et très petits ne sont pas représentables. (On reviendra plus tard sur la représentation des nombres réels en machine).

Important : En Python, la virgule séparant la partie entière de la partie décimale doit être représentée par un point, auquel cas vous pourrez observer des comportements étranges de l'interpréteur.

Différentes façons de diviser

Lorsque l'on utilise l'opérateur de division classique de Python /, on obtient un nombre flottant (type float) :

>>> 7 / 2
3.5
>>> type(7 / 2)
<class 'float'>

Les types de base

Vous pouvez voir ci-dessus apparaître le mot-clé class. Les classes sont une notion qui n'est abordée qu'en terminale, mais en gros, il faut comprendre que tout ce qui est manipulé en Python est objet. Il y a des objets de type int (les nombres entiers), des objets de type float (les nombres flottants), des objets de type str (les chaînes de caractères, etc.)

Ici, on a utilisé une fonction appelée type afin de voir de quel type est l'objet qui résulte de l'opération 7 / 2. (On a ici la confirmation que l'on obtient bien un flottant.)

Il existe deux autres opérateurs bien utiles :

  • // permettant d'obtenir le quotient de la division euclidienne de deux opérandes,
  • % permettant d'obtenir le reste de la division euclidienne de deux opérandes.

Par exemple :

>>> 7 // 2
3
>>> 7 % 2
1
>>> type(7 // 2)
<class 'int'>
>>> type(7 % 2)
<class 'int'>

Ces deux opérateurs renvoient donc des entiers. On rappelle que pour deux nombres entiers \(a\) et \(b\), \(a = q \times b + r\), avec \(q~=~a~//~b\) le quotient et \(r~=~a~\%~b\) le reste.

Variables

Saisir des expressions arithmétiques, c'est bien. Mais une calculatrice sait déjà le faire !

Nous allons maintenant introduire la notion de variable. Une variable permet de stocker une donnée utilisée par un programme.

Cela se fait par une affectation qui associe une donnée, représentée par une valeur ou une expression, avec un nom.
Une expression stockée dans une variable peut elle-même contenir d'autres variables.

Une variable peut s'apparenter à une "boîte" sur laquelle est écrit un nom et dans laquelle on place des informations diverses (même si dans la réalité, ce n'est pas tout à fait ça). Un nom peut être n’importe quelle chaîne alphanumérique, à l'exception de certains mots réserés, et ne doit pas commencer par un chiffre.

L’opérateur d’affectation est noté =.
Par exemple, l’instruction x = 4 associe la valeur 4 au nom x :

>>> x = 4
>>>

Si l'on saisit une instruction d'affectation dans l'interpréteur Python, aucun résultat n'est affiché. Si l'on souhaite accéder à la valeur mémorisée dans a, il suffit de saisir :

>>> a
4

L’instruction y = 3 + 5 associe la valeur de l’expression située à droite du signe =, ici 8, au nom y.
L’instruction z = x + y associe la valeur de l’expression située à droite du signe =, ici 12 (la somme des valeurs contenues dans les variables x et y), au nom z.

Python permet par ailleurs d'effectuer des affectations multiples, par exemple x, y, z = 1, 3, 5, qui associe les valeurs 1, 3 et 5 respectivement aux noms x, y et z. Ceci est équivalent à écrire x=1; y=3; z=5 sur une ligne ou à effectuer les 3 affectations sur 3 lignes successives.

À retenir

  • Une variable est composée d’un nom (ou identificateur), d’une adresse en mémoire où est enregistrée une valeur (ou un ensemble de valeurs), et d’un type qui définit ses propriétés.
  • Une expression a une valeur qui est le résultat d’une combinaison de variables ou d’objets, de constantes et d’opérateurs.
  • Une instruction est une commande qui doit être exécutée par la machine.
  • Une affectation est une instruction qui commande à la machine de créer une variable en lui précisant son nom et la valeur qui lui est associée.

Il est important de bien distinguer une expression, qui se calcule et a une valeur, d'une instruction, qui est exécutée par la machine.

À vous de jouer !

Voici une suite d'instructions :

###
a = 4bksl-nlb = 6bksl-nlc = a + bbksl-nlbksl-nl

Quelle sera la valeur contenue dans la variable c après avoir saisi ces instructions ?
Vérifiez en exécutant le programme.

Même question avec la suite d'instructions suivante :

###
a = 4bksl-nla = a py-str abksl-nlb = a + 4bksl-nlbksl-nl

À vous de jouer 2 !

Voici une suite d'instructions :

x = 3
y = 6
z = 4 + x * y
x = z / 2

Quelle sera la valeur contenue dans la variable x après avoir saisi ces instructions ?
Vérifier en utilisant la console de Thonny.
Que constatez-vous de particulier sur la valeur contenue dans x ?

Réponse - À vous de jouer 2 !

Normalement, la console devrait afficher la valeur 11.0. On constate que l'on obtient ici un nombre décimal, et non pas un nombre entier. En fait, il existe plusieurs types de données que peuvent contenir les variables : les entiers, les flottants (nombres décimaux), mais également d'autres types de données que l'on verra plus tard, comme les chaînes de caractères, les booléens...

Si l'on avait utilisé l'opérateur //, c'est-à-dire si l'on avait saisi z // 2, on aurait obtenu le quotient de la division euclidienne entre les opérandes z et 2, c'est-à-dire l'entier 11. L'opérateur permettant d'obtenir le reste d'une division euclidienne est %.

Mode programme

Le mode programme de Python consiste à écrire une suite d’instructions dans un fichier et à les faire exécuter par l’interpréteur Python. Cette suite d’instructions s’appelle un programme (ou code source). Cela permet d'éviter d'avoir à ressaisir à chaque fois les instructions dans le mode interactif. Cela permet de faire la distinction entre le rôle de programmeur et celui d’utilisateur d’un programme.

Affichage sur la sortie standard

Contrairement au mode interactif, en mode programme, les résultats des expressions calculées ne sont plus affichés à l’écran. Il faut utiliser pour ceci une instruction explicite d’affichage. En Python, elle s’appelle print. Par exemple :

###

print(3)bksl-nl

On peut également fournir à print une expression, qui sera calculée puis affichée :

###

4 + 2 py-str 3bksl-nl

print est également capable d'afficher du texte, qui doit être mis entre guillemets " ou apostrophes ', par exemple :

###

print("Bienvenue à tous !")bksl-nl

On peut également afficher la valeur d'une variable, exemple :

###

a = 34bksl-nlb = 21 + abksl-nlprint(a) # afficher la valeur de abksl-nlprint(b) # afficher la valeur de bbksl-nl

Si l'on souhaite inclure la valeur d'une variable dans un texte affiché par print, on peut procéder de différentes façons :

  • avec la concaténation : on peut inclure notre variable dans une chaîne de caractères en utilisant la concaténation de chaînes de caractères. Pour cela, on transforme notre variable de type int en type str (c'est-à-dire en chaîne de caractères) avec la fonction str(), et on effectue la concaténation en utilisant un +.
  • avec des f-strings (pas au programme) : si on ajoute un f devant notre chaîne de caractères, on peut inclure nos variables entre accolades, et elles seront remplacées par leur valeur. Il n'y a, dans ce cas, pas besoin de convertir le type de nos variables en str.
  • en donnant plusieurs valeurs (plusieurs arguments) à notre fonction print : dans ce cas, print affichera chaque valeur les unes à la suite des autres, que cela soit des chaînes de caractères ou non.
###

nombre = 42bksl-nlprint("Votre nombre est " + str(nombre) + " !") # avec une concaténation de chaînes de caractèresbksl-nlbksl-nlprint(f"Votre nombre est {nombre} !") # avec un f-stringbksl-nlbksl-nlprint("Votre nombre est ", nombre, " !") # en donnant plusieurs valeurs à printbksl-nl

La fonction print effectue par défaut un retour à la ligne après avoir affiché les valeurs que vous lui avez donné. Pour changer ce comportement, on peut ajouter le paramètre end :

###

''' Utilisation du paramètre end de print. '''bksl-nlbksl-nlprint("abc", end="")bksl-nlprint("def", end="")bksl-nlprint("ghi", end=".")bksl-nl

Un autre paramètre que l'on peut utiliser avec la fonction print est le paramètre sep.
Ce paramètre permet d'indiquer un autre caractère pour séparer plusieurs valeurs données à la fonction print, par exemple :

###

''' Utilisation du paramètre end de print. '''bksl-nlbksl-nl# Initialement, la séparation est un espace :bksl-nlprint("Bonjour", "Monsieur", "Demerville")bksl-nl# Remplaçons le séparateur par un tiret :bksl-nlprint("Bonjour", "Monsieur", "Demerville", sep="-")bksl-nl

Interaction avec l'utilisateur, lire sur l'entrée standard

Pour demander à l'utilisateur de saisir une valeur, de manière à pouvoir la stocker dans une variable et en faire quelque chose, on utilise la fonction input.

###

''' Programme qui calcule le nombre suivant celuibksl-nldonné par l'utilisateur. '''bksl-nlbksl-nls = input()bksl-nla = int(s)bksl-nlprint("le nombre suivant est ", a + 1)bksl-nl

Attention au type

La valeur renvoyée par la fonction input est de type str (chaîne de caractères). Si vous voulez utiliser cette valeur dans une opération arithmétique par exemple, il faut donc la convertir en int (nombre entier), d'où l'utilisation de la fonction int ci-dessus.

Un programme contenant un appel à input ne s'arrête que lorsque l'utilisateur a saisi une valeur et appuyé sur la touche Entrée. En attendant, le programme reste en stand-by.

Notebook bloqué

Lorsque vous travaillez sur un notebook (sur Capytale par exemple), faites attention lorsque vous exécutez une cellule de code contenant un input : tant que vous ne saisissez pas de valeur attendue par la fonction input, l'exécution de la cellule ne s'arrêtera pas, et vous ne pourrez exécuter aucune autre cellule de code pendant ce temps (si vous le faites, vous obtiendrez un message d'erreur).

On peut également indiquer dans un input un message à afficher, voici une exemple :

###

''' Programme qui calcule le nombre suivant celuibksl-nldonné par l'utilisateur. '''bksl-nlbksl-nlage = input("Indiquez votre âge : ")bksl-nlprint("Votre âge est de :", age, "ans.")bksl-nl

Cela permet d'indiquer à l'utilisateur la nature de la valeur attendue.

Les boucles bornées for

Répéter plusieurs fois les mêmes instructions est assez rébarbatif. C'est pour cela qu'il existe une instruction appelée boucle bornée, utilisant le mot-clé for, qui permet de répéter plusieurs fois un bloc d'instructions. Par exemple :

###

for i in range(10):bksl-nl print("Je ne dois pas bavarder en cours.")bksl-nl

Dans la fonction range, on indique le nombre de fois que l'on souhaite afficher l'instruction print. Ici, on l'affiche 10 fois.

En réalité, ce qu'il se passe, c'est que la boucle va itérer de la valeur i = 0 à la valeur i = 9 (la valeur indiquée dans le range moins 1), la variable i fournie à notre boucle étant ce l'on appelle l'indice de boucle. On dit que i incrémente (augmente de 1) à chaque itération de la boucle.

En pseudo-langage, on pourrait traduire ce programme de la manière suivante :

POUR i ALLANT DE 0 à 9 :
    AFFICHER "Je ne dois pas bavarder en cours"

Si l'on affiche ce que contient notre variable i à chaque fois :

###

for i in range(10):bksl-nl print("i = ", end="")bksl-nl print(i)bksl-nl

On peut utiliser le range de plusieurs manières différentes :

  • range(valeur) : avec une seule valeur entière, la boucle va itérer de 0 à valeur - 1,
  • range(min, max) : avec deux valeurs entières, la boucle va itérer de min à max - 1,
  • range(min, max, pas) : avec trois valeurs entières, la boucle va itérer de min à max - 1 avec un pas de pas. Si l'on n'indique pas ce pas, il est de 1 par défaut.

Par exemple, si l'on souhaite afficher tous les nombres pairs de 2 à 98 :

###

''' Affichage des nombres pairs de 2 à 98 '''bksl-nlbksl-nlfor nb in range(2, 99, 2):bksl-nl print(nb)bksl-nl

Ici, on a appelé l'indice de boucle nb. On peut l'appeller comme on veut, mais on utilise souvent des noms à une lettre comme i, j et k.

On peut également passer la valeur de retour d'un input à l'intérieur d'un range, par exemple :

###

''' Un compte à rebours '''bksl-nlbksl-nlnb = input("Valeur initiale du compte à rebours : ")bksl-nlfor i in range(int(nb), -1, -1):bksl-nl print(i)bksl-nl

Attention au type

Le range ne prend que des entiers. Si vous souhaitez lui passer la valeur de retour d'un input, il faut donc convertir cette valeur en valeur entière avec la fonction int.

Comparaisons, booléens, tests

Une part importante de la conception d’un programme consiste à imaginer les différents cas de figure possibles, notamment selon les entrées fournies par l’utilisateur ou les valeurs des différentes variables, de manière à adapter le programme à chacun des cas. Pour traduire cela dans l’écriture du programme, on peut utiliser des instructions de branchement (if, elif, else) qui rassemblent plusieurs blocs de code alternatifs, chacun associé à une condition logique, et qui à chaque exécution sélectionne au plus l’un de ces blocs.

Les instructions de branchement

Les instructions if (si), elif (sinon si) et else (sinon) permettent d'exécuter des blocs de code uniquement lorsqu'une condition est remplie. Par exemple, voici un programme qui demande à l'utilisateur de saisir un nombre et affiche un message différent selon le nombre saisi :

###

nb = int(input("Saisissez votre âge : "))bksl-nlif nb < 5:bksl-nl print("Vous êtes un bébé.")bksl-nlelif nb < 12:bksl-nl print("Vous êtes un enfant.")bksl-nlelif nb < 18:bksl-nl print("Vous êtes un ado")bksl-nlelif nb < 40:bksl-nl print("Vous êtes un adulte")bksl-nlelif nb < 60:bksl-nl print("Vous êtes un vieil adulte")bksl-nlelse:bksl-nl print("Vous êtes un vieillard.")bksl-nl

Le programme est executé séquentiellement, autrement dit, on vérifie d'abord la première condition, si elle est vraie, alors on affiche le message indiqué, sinon on passe à la deuxième condition, et ainsi de suite...

Le comportement de ce programme est donc le suivant :

  • si nb est inférieur à 5 (donc compris entre 0 et 4 inclus), on affiche "Vous êtes un bébé.",
  • sinon si nb est inférieur à 12 (donc compris entre 5 et 11 inclus), on affiche "Vous êtes un enfant.",
  • sinon si nb est inférieur à 18 (donc compris entre 12 et 17 inclus), on affiche "Vous êtes un ado.",
  • sinon si nb est inférieur à 40 (donc compris entre 18 et 39 inclus), on affiche "Vous êtes un adulte.",
  • sinon si nb est inférieur à 60 (donc compris entre 40 et 59 inclus), on affiche "Vous êtes un vieil adulte.",
  • sinon, (donc si l'âge est au moins de 60), on affiche "Vous êtes un vieillard.".

Une seule branche est donc choisie en fonction de la valeur de nb, et un seul message sera donc affiché.

Voici les symboles permettant d'effectuer des comparaisons :

> plus grand que >= supérieur ou égal à == égal à
< plus petit que <= inférieur ou égal à != différent de

Homogénéité des valeurs comparées

Lorsque vous effectuez une comparaison (utilisant l'un des symboles ci-dessus), il faut vous assurer que vous compariez bien ce qui est comparable. Vous pouvez comparer des entiers entre eux, des flottants entre eux, des chaînes de caractères entre eux, ou encore comparer un entier avec un flottant... Mais vous ne pouvez pas, par exemple, comparer un entier avec une chaîne de caractères, ou vous obtiendrez une erreur.

>>> 1 < "123"
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: '<' not supported between instances of 'int' and 'str'

Le type bool

Lorsque vous effectuez une expression conditionnelle (qu'on appelera parfois test), cette expression est évaluée par Python soit à la valeur True, soit à la valeur False. Voici un exemple (exécuter le programme pour voir le résultat) :

###
x = 10bksl-nltest1 = x > 5bksl-nltest2 = x > 15bksl-nlbksl-nlprint("test1 =", test1)bksl-nlprint("test2 =", test2)bksl-nl

Les valeurs True et False sont des valeurs booléennes (de type bool), ce sont d'ailleurs les deux seules valeurs que peut prendre une variable de ce type.

Lorsque vous utilisez une instruction de branchement, comme if <condition>, la condition est une expression qui est évaluée à True ou False. Les instructions du bloc if ne sont exécutées que si la condition est évaluée à True.

Par exemple :

age = 30
majeur = age >= 18
if majeur:
    print("Vous êtes majeur")

Ici, l'expression age >= 18 est évaluée à True, et la valeur True est stockée dans la variable majeur. Ainsi, on rentre bien dans le bloc if.

Cela revient au même résultat que d'écrire :

age = 30
if age >= 18:
    print("Vous êtes majeur")

Les opérateurs logiques

En Python, des opérateurs logiques permettent de combiner plusieurs conditions et ainsi réduire le nombre d'instructions de branchement nécessaires.

L'opérateur and

Si l'on souhaite par exemple vérifier deux conditions à la fois et afficher le message "OK" lorsque les deux conditions sont vérifiées, on peut procéder ainsi :

if <condition 1>:
    if <condition 2>:
        print("OK !")

Plutôt que d'utiliser deux instructions if, on peut obtenir le même comportement en utilisant l'opérateur logique and de la manière suivante :

if <condition 1> and <condition 2>:
    print("OK")

En effet, ici, on entre dans le bloc if seulement si le test <condition 1> and <condition 2> est évalué à True, c'est-à-dire si la condition 1 ET la condition 2 sont toutes les deux vraies (évaluées au booléen True).

On peut bien sûr utiliser autant de fois l'opérateur and que l'on souhaite à l'intérieur d'un test, voici un exemple :

###

taille = 175bksl-nlpoids = 70bksl-nlif taille > 170 and taille < 180 and poids > 60 and poids < 80:bksl-nl print("Vous pouvez entrer")bksl-nl

Ici, le message "Vous pouvez entrer" est affiché uniquement si les 4 conditions taille > 170, taille < 180, poids > 60, poids < 80 sont vérifiées.

L'opérateur or

Autre situation : on souhaite maintenant tester deux conditions, et vérifier si seulement l'une ou l'autre est vraie, ou si les deux sont vraies à la fois.

En utilisant des instructions de branchement, on pourrait écrire :

if <condition 1>:
    print("OK !")
elif <condition 2>:
    print("OK !")
else:
    print("Pas ok.")

Ici encore, on peut réduire le nombre d'instructions de branchement en introduisant l'opérateur logique or :

if <condition 1> or <condition 2>:
    print("OK !")
else:
    print("Pas ok.")

Un opérateur paresseux

L'opérateur or possède une caractéristique intéressante. En effet, si vous utilisez un or entre deux conditions, Python n'évaluera pas la deuxième condition si la première est évaluée à True. Par exemple :

x = 35
y = 15
if x >= 30 or y <= 20:
    print("OK")

Ici, l'expression y <= 20 ne sera pas évaluée car la première expression x >= 30 est évaluée à True. Étant donné que seule l'une ou l'autre des conditions doit être vraie pour que le test x >= 30 or y <= 20 soit vérifié, on a pas besoin de tester la deuxième condition.

Voici un autre exemple en utilisant deux opérateurs or :

x = 35
y = 15
z = 'hello'
if x >= 50 or y <= 20 or z == 'hello':
    print("OK")

Ici, on évaluera le première et la deuxième condition, mais pas la troisième, car la deuxième condition y <= 20 est évaluée à True.

Tout comme pour le and, on peut accumuler autant de fois l'opérateur or qu'on le souhaite dans un test, on peut également combiner des and et des or, par exemple :

###

age = 15bksl-nlaccompagne = Truebksl-nlif age >= 18 and age < 70 or accompagne:bksl-nl print("Vous pouvez entrer")bksl-nl

Priorité des opérations :

En Python, la priorité des opérateurs logiques est la suivante :

  • notplus haute priorité
  • and – priorité intermédiaire
  • orplus basse priorité

Cela signifie que dans une expression contenant plusieurs opérateurs, les termes de l'expression utilisant le not sont évalués en premier, puis viennent ensuite les termes de l'expression utilisant le and, et enfin les termes de l'expression utilisant le or.
Après avoir appliqué les priorités, l'évaluation s'effectue de gauche à droite.

On peut toutefois obtenir des différences dans le résultat booléen obtenu en ajoutant des parenthèses pour forcer un ordre d’évaluation différent, par exemple :

>>> x = 5
>>> y = 10
>>> z = 15
>>> x > 0 or y > 10 and z > 20  # True or False and False => True or False => True
True
>>> (x > 0 or y > 10) and z > 20  # (True or False) and False => True and False => False
False
>>> x > 0 or y < 0 and not z == 15  # True or False and not True => True or False and False => True or False => True
True
>>> (x > 0 or y < 0) and not z == 15  # (True or False) and not True => True and False => False
False

L'opérateur not

L'opérateur logique not utilisée sur une expression booléenne renvoie True si l'expression est évaluée à False, et False si l'expression est évaluée True.

Voici un exemple :

###

x = 10bksl-nltest1 = x > 5bksl-nltest2 = not (x > 5)bksl-nlbksl-nlprint("test1 =", test1)bksl-nlprint("test2 =", test2)bksl-nl

Tables de vérité

Les opérateurs logiques and, or et not permettent de combiner des expressions booléennes pour créer des conditions plus complexes. Voici les tables de vérité qui montrent comment ces opérateurs fonctionnent.

A B A and B
True True True
True False False
False True False
False False False
A B A or B
True True True
True False True
False True True
False False False
A not A
True False
False True

Ces tables montrent que l'opérateur and ne renvoie True que si les deux opérandes sont vraies, tandis que l'opérateur or renvoie True si au moins une des opérandes est vraie. L'opérateur not, quant à lui, inverse la valeur booléenne.

Les fonctions

Vous avez jusqu'ici utilisé plusieurs fonctions natives de Python, comme print, input, int (la fonction, pas le type), range...

Les fonctions permettent de créer des fragments de code réutilisables. Cela va nous permettre d'aller beaucoup plus loin dans la conception de nos programmes, en créant des fonctions pour effectuer certaines tâches autant de fois qu'on le souhaite, et pour différentes valeurs d'entrée.

Pour définir une fonction, on utilise le mot-clé def, suivi du nom de la fonction, suivi de :, puis vient ensuite (à la ligne, avec une identation) le bloc d'instructions à exécuter dans la fonction.

Une fonction = une tâche

Il est fortement conseillé, pour la clarté du code, de faire en sorte qu'une fonction ne représente qu'une seule tâche.

Par exemple, voici une fonction qui prend deux nombres entiers et renvoie la somme de ces deux nombres :

###

def somme(a, b):bksl-nl ''' Fonction qui renvoie la somme des entiers a et b. '''bksl-nl return a + bbksl-nl

Pour tester cette fonction, vous pouvez l'appeler dans l'interpréteur (en saisissant l'instruction d'appel après les trois chevrons >>> et en appuyant sur la touche Entrée) après avoir exécuté le programme. Par exemple, entrez l'instruction somme(3,5) dans l'interpréteur ci-dessus après exécution du programme pour observer le résultat.

Vous pouvez également appeller cette fonction directement dans le programme, mais pour afficher la valeur renvoyée, il faudra ajouter un print :

###

def somme(a, b):bksl-nl ''' Fonction qui renvoie la somme des entiers a et b. '''bksl-nl return a + bbksl-nlbksl-nlprint("La somme de 4 et de 8 vaut : ", somme(4, 8))bksl-nl

La valeur renvoyée par la fonction est indiquée avec le mot-clé return. Lorsque le return est rencontré, on sort de la fonction. Ainsi, si vous rajoutez des instructions après ce return, elles ne seront pas exécutées, exemple :

###

def somme(a, b):bksl-nl ''' Fonction qui renvoie la somme des entiers a et b. '''bksl-nl return a + bbksl-nl print("Ce message n'apparaitra pas.")bksl-nl

Fonctions sans paramètres

Une fonction peut ne pas avoir de paramètres. Par exemple, voici une fonction qui renvoie une valeur aléatoire entre 1 et 100, en appelant la fonction randint du module random :

###

from random import randintbksl-nlbksl-nldef valeurpy-undaleatoire():bksl-nl return randint(1,100)bksl-nlbksl-nlprint("Valeur aléatoire : ", valeurpy-undaleatoire())bksl-nl

Ici, on appelle la fonction sans lui fournir de valeurs, mais la fonction appelée renvoie bien une valeur (le mot-clé return est bien présent).

Fonctions sans valeur de retour : procédures

Il est également possible de créer une fonction qui ne contient pas de valeur de retour (pas de return). Par exemple :

###

def hurler(message):bksl-nl messagepy-undmaj = message.upper() # mettre le message en majusculesbksl-nl print(messagepy-undmaj) # afficher le nouveau messagebksl-nlbksl-nlhurler("houraaaaa !!!")bksl-nl

On crée ici une fonction qui contient un paramètre message, et qui affiche avec print ce message, mais ne renvoie rien. La fonction est ensuite appelée, avec le message "HOURAAAAAAAAAAA !!!" qui est donc affiché.

Les fonctions qui ne renvoient rien peuvent être appelées procédures. Toutefois, en Python, il n'y a techniquement pas de distinction entre les fonctions et les procédures, contrairement à d'autres langages comme le Java par exemple.

Variables locales et globales

Lorsque vous définissez une variable à l'intérieur d'une fonction, cette variable n'existe qu'à l'intérieur de la fonction. C'est également le cas pour les paramètres des fonctions.

Par exemple, dans la fonction ci-dessous, on a créé une variable resultat. Cette variable, ainsi que les paramètres a et b (qui sont également des variables), ne sont définis qu'à l'intérieur de la fonction. :

###

def produit(a, b):bksl-nl resultat = 1bksl-nl for py-und in range(b):bksl-nl resultat = resultat py-str abksl-nl return resultatbksl-nl

On peut donc très bien, par exemple, avoir une autre variable avec le nom resultat à l'extérieur de la fonction, qui contiendrait une autre valeur :

###

def produit(a, b):bksl-nl resultat = 1bksl-nl for py-und in range(b):bksl-nl resultat = resultat py-str abksl-nl return resultatbksl-nlbksl-nlresultat = produit(5, 6) # appeler produit sur 5 et 6 et stocker la valeur de retour dans resultatbksl-nlprint(resultat) # afficher le résultatbksl-nl

Les boucles non bornées while

Nous avons vu qu'il était possible de créer des boucles bornées à l'aide d'instructions utilisant le mot-clé for. Nous allons voir ici que l'on peut également créer des boucles non bornées à l'aide du mot-clé while (qui signifie TANT QUE en français).

Une boucle non bornée permet d'exécuter un bloc d'instructions plusieurs fois, et de continuer TANT QUE une condition donnée est vérifiée.

La structure d'une boucle while en Python est la suivante :

while condition:
    # Bloc d'instructions à répéter tant que la condition est vraie

La boucle while commence par évaluer la condition :

  • Si la condition est vraie, le bloc d'instructions à l'intérieur de la boucle est exécuté.
  • Après chaque exécution du bloc, la condition est à nouveau évaluée :
    • Tant que la condition reste vraie, la boucle continue de s'exécuter.
    • Dès que la condition devient fausse, la boucle s'arrête.

Voici un premier exemple d'utilisation d'une boucle while (cliquez sur le premier bouton, Lancer, pour exécuter le programme):

###

i = 1bksl-nlwhile i <= 5:bksl-nl print(i)bksl-nl i += 1bksl-nl

Dans cet exemple, la boucle while est utilisée pour afficher les nombres de 1 à 5.
La variable i est initialisée à 1, et la boucle continue TANT QUE i est inférieur ou égal à 5. À chaque itération de la boucle, la valeur de i est ici affichée (à l'aide du print), puis est incrémentée de 1 à l'aide de l'instruction i += 1. La boucle s'arrête lorsque i atteint 6, car la condition devient fausse.

Voici un autre exemple un peu plus concret de l'utilisation d'une boucle while :

###

motpy-unddepy-undpasse = "secret"bksl-nlessaispy-undrestants = 3bksl-nlbksl-nlwhile essaispy-undrestants > 0:bksl-nl tentative = input("Entrez le mot de passe : ")bksl-nl bksl-nl if tentative == motpy-unddepy-undpasse:bksl-nl print("Accès autorisé !")bksl-nl break # Sortir de la bouclebksl-nl else:bksl-nl essaispy-undrestants -= 1bksl-nl print(f"Mot de passe incorrect. Il vous reste {essaispy-undrestants} essais.")bksl-nlbksl-nlif essaispy-undrestants == 0:bksl-nl print("Nombre maximal d'essais atteint. Accès refusé.")bksl-nl

Dans cet exemple, une boucle while est utilisée pour demander à l'utilisateur (à l'aide de la fonction input) de saisir un mot de passe jusqu'à trois essais.
La boucle continue tant qu'il reste des essais (tant que essais_restants > 0) :

  • Si l'utilisateur saisit le mot de passe correct (c'est-à-dire lorsque tentative == mot_de_passe), la boucle s'arrête à l'aide de l'instruction break. L'instruction break permet de sortir directement de la boucle, et évite donc de ré-évaluer sa condition.
  • Sinon, le nombre d'essais restants est décrémenté (cela signifie que l'on soustrait 1 à essais_restants), et un message est affiché (à l'aide de la fonction print) pour informer l'utilisateur du nombre d'essais restants.

Une fois que l'on est sorti de la boucle (ce qui se produit soit si on a rencontré l'instruction break, soit si la condition de la boucle est fausse), on affiche un message d'accès refusé si le nombre maximal d'essais a été atteint.

Attention aux boucles infinies !

Dans une boucle non bornée while, contrairement à une boucle bornée for, il y a un risque de créer boucle infinie, c'est-à-dire une boucle dont la condition n'est jamais évaluée à False. Par exemple :

i = 1
while i < 5:
    print(i)

Ici, on n'a pas écrit d'instruction permettant d'incrémenter la variable i, donc la condition i < 5 sera toujours vraie. Le programme ne s'arrêtera donc jamais.

À vous de jouer !

Écrivez un programme en Python qui demande à l'utilisateur d'entrer un nombre entier positif n. Ensuite, utilisez une boucle while pour calculer la somme de tous les entiers de 1 à n. Affichez ensuite le résultat.

Voici un exemple d'exécution du programme :

Entrez un nombre entier positif : 5
La somme des entiers de 1 à 5 est : 15
Cliquez ici pour afficher l'aide

Voici comment procéder pour réaliser ce programme :

  1. Utilisez la fonction input pour demander à l'utilisateur de saisir un nombre entier positif, vous stockerez la valeur saisie dans une variable que vous pourrez appeler n. N'oubliez de convertir cette valeur en entier à l'aide de la fonction int.
  2. Créez une variable pour accumuler la somme totale, que vous initialiserez à 0 avant d'entrer dans la boucle.
  3. Créez une variable que vous appelerez i par exemple, qui prendra chaque valeur de 1 à n.
  4. Utilisez une boucle while qui continue tant que i est inférieur ou égal à n, pour ajouter les entiers de 1 à n à la somme totale.
  5. N'oubliez pas d'incrémenter la valeur de i à chaque itération de la boucle.
  6. Enfin, après la boucle, affichez le résultat à l'aide de la fonction print.

Vous pouvez réaliser ce programme sur Thonny ou dans l'IDE ci-dessous (vous pourrez télécharger votre programme en cliquant sur le deuxième bouton pour le conserver.)

###

Les listes

En programmation en langage Python, les listes sont l'une des structures de données les plus couramment utilisées. Une liste (objet de type list) est une collection ordonnée d'éléments pouvant être de différents types (nombres entiers, flottants, chaînes de caractères, objets, etc.). Les listes sont extrêmement flexibles et permettent de stocker et de manipuler des données de manière efficace.

Création d'une liste

Pour créer une liste en Python, on utilise les crochets [].

Créer une liste vide

Une liste vide est créée simplement avec des crochets vides [].

Si l'on veut stocker une nouvelle liste vide dans une variable, par exemple dans une variable appelée ma_liste_vide, il suffit de faire une affectation :

>>> ma_liste_vide = []
>>> ma_liste_vide
[]

Créer une liste non vide

On peut également créer une liste et directement y mettre des éléments, en les séparant par des virgules. Voici un exemple :

>>> ma_liste = [1, 2, 3, 4, 5]
>>> ma_liste
[1, 2, 3, 4, 5]

On aurait également pu créer cette liste en utilisant la méthode de création de liste par compréhension, en utilisant une boucle :

>>> ma_liste = [v for v in range(1, 6)]
>>> ma_liste
[1, 2, 3, 4, 5]

Si l'on voulait créer une liste contenant tous les nombres pairs de 2 à 98 en utilisant la méthode par compréhension, on aurait écrit :

>>> ma_liste = [v for v in range(2, 99, 2)]

ou bien, en ajoutant une condition if :

>>> ma_liste = [v for v in range(2, 99) if v % 2 == 0]

Une liste peut contenir n'importe quel type d'élément, y compris d'autres listes. Par exemple :

>>> liste_mixte = [1, "texte", 3.14, [10, 20, 30]]
>>> liste_mixte
[1, "texte", 3.14, [10, 20, 30]]

Accès aux éléments d'une liste

Vous pouvez accéder aux éléments individuels d'une liste en utilisant leur indice (position) dans la liste. L'indice commence à 0 pour le premier élément, puis aumgente de 1 en 1.

Principe des indices
Une liste d'éléments avec les indices (en vert) associés.

Voici comment accéder aux éléments d'une liste :

>>> ma_liste = [10, 20, 30, 40, 50]  # création d'une liste
>>> ma_liste[0] # accès au premier élément (indice 0)
10
>>> ma_liste[2] # accès au troisième élément (indice 2)
30

On peut bien sûr stocker ces éléments dans de nouvelles variables :

>>> ma_liste = [10, 20, 30, 40, 50]  # création d'une liste
>>> el = ma_liste[3]  # accès au quatrième élément (indice 3)
>>> el
40

La fonction len

La fonction len est une fonction permettant d'obtenir le nombre d'éléments d'une liste, d'un tuple, d'un dictionnaire ou encore d'une chaîne de caractères.

Il suffit d'appeler len en lui donnant comme entrée une liste dont on cherche à déterminer le nombre d'éléments, et la fonction renverra le nombre d'éléments de la liste.

Exemple :

>>> lst = [2, 4, 6, 8, 10, 12]
>>> len(lst)
6

Si l'on dipose d'une liste dont on ne sait pas combien elle contient d'éléments, et qu'on souhaite trouver le dernier élément de cette liste, on peut alors utiliser la fonction len pour trouver l'indice de cet élément, et ainsi accéder à l'élément.

###

from random import randintbksl-nlbksl-nl# Créer une liste de taille et d'éléments aléatoiresbksl-nllst = [randint(0, 100) for py-und in range(1, randint(1, 51))]bksl-nlbksl-nl# On cherche le dernier élément de cette listebksl-nlprint("taille : ", len(lst)) # afficher la taille de lstbksl-nldernierpy-undel = lst[len(lst) - 1] # récupérer le dernier élémentbksl-nlprint("dernier élément : ", dernierpy-undel) # afficher l'élémentbksl-nl

Explication

Pour obtenir l'indice du dernier élément d'une liste, il suffit donc de retirer 1 à la taille de cette liste.

Modification des éléments d'une liste

Les éléments d'une liste peuvent être modifiés en à l'aide d'une simple affectation.
Il suffit de connaître l'indice de l'élément à modifier.

Par exemple :

>>> ma_liste = [10, 20, 30, 40, 50]
>>> ma_liste[1] = 25  # modification du deuxième élément (indice 1)
>>> ma_liste
[10, 25, 30, 40, 50]

Ajout et suppression d'éléments

Une méthode, kézako ?

Une méthode est une fonction associée à un type d'éléments spécifique.
Pour appeler une méthode, on écrit une instruction de la forme : <objet>.<methode>.

Par exemple, le type list de Python dispose d'une méhode appelée append qui permet d'ajouter un élément à une liste.

Pour ajouter un élément à la fin d'une liste, vous pouvez utiliser la méthode append() :

>>> ma_liste = [1, 2, 3]
>>> ma_liste.append(4)
>>> ma_liste
[1, 2, 3, 4]

On peut utiliser append dans une boucle pour créer une liste contenant beaucoup d'éléments. Il s'agit d'une solution alternative à la création de liste "par compréhension" vue précédemment.

Par exemple, si l'on souhaite créer une liste dans une variable lst qui contient tous les entiers de 1 à 100, on peut procéder comme suit (cliquez sur Lancer pour voir le résultat) :

###

lst = [] # créer une liste videbksl-nlfor i in range(1, 101): # itérer de 1 jusqu'à 100bksl-nl lst.append(i) # ajouter l'entier ibksl-nlbksl-nlprint(lst) # afficher la liste lstbksl-nl

Pour supprimer un élément en fonction de sa valeur, vous pouvez utiliser la méthode remove() :

>>> ma_liste = [1, 2, 3, 4, 5]
>>> ma_liste.remove(3)
>>> ma_liste
[1, 2, 4, 5]

Pour supprimer un élément en fonction de son indice, vous pouvez utiliser la méthode pop, qui supprime et renvoie l'élément de la liste dont l'indice est donné en entrée :

ma_liste = [1, 2, 3, 4, 5]
del ma_liste[2]  # La liste devient maintenant [1, 2, 4, 5]

Hors-programme

À noter que la suppression d'éléments dans une liste n'est pas au programme de première, ni même de terminale. Vous ne serez donc pas interrogé dessus, mais cela peut s'avérer bien utile tout de même.

Parcours d'une liste avec des boucles

Les listes sont souvent parcourues à l'aide de boucles bornées for. Voici comment parcourir une liste et afficher ses éléments :

###

mapy-undliste = [10, 20, 30, 40, 50]bksl-nlbksl-nlfor element in mapy-undliste:bksl-nl print(element)bksl-nl

Ce code affichera chaque élément de la liste sur une ligne différente.

Une autre méthode pour parcourir une liste et afficher ses éléments consiste à utiliser l'indice de ses éléments : On souhaite maintenant parcourir, à l'aide d'une boucle for, chaque élément à partir de l'indice 0 (indice du premier élément), et jusqu'à l'indice du dernier élément.

On peut faire cela en utilisant une boucle for avec la fonction range() pour générer des indices, puis en accédant aux éléments de la liste à l'aide de ces indices. Voici comment cela fonctionne :

###

mapy-undliste = [10, 20, 30, 40, 50]bksl-nlbksl-nllongueur = len(mapy-undliste) # obtenir la longueur de la listebksl-nlbksl-nl# Utiliser une boucle for avec la fonction range() pour parcourir les indicesbksl-nlfor i in range(longueur):bksl-nl element = mapy-undliste[i]bksl-nl print(f"Élément à l'indice {i} : {element}")bksl-nl

Dans cet exemple, nous avons utilisé la fonction len sur ma_liste pour obtenir la longueur de la liste, c'est-à-dire son nombre d'éléments.
Ensuite, nous avons utilisé une boucle for pour parcourir les indices de 0 à longueur - 1.
À chaque itération de la boucle, nous avons accédé à l'élément d'indice i à l'aide de ma_liste[i], puis nous l'avons affiché.

On aurait pu écrire ce programme de manière plus simple, comme ceci :

###

mapy-undliste = [10, 20, 30, 40, 50]bksl-nlbksl-nl# Utilisez une boucle for avec la fonction range() pour générer des indicesbksl-nlfor i in range(len(mapy-undliste)):bksl-nl element = mapy-undliste[i]bksl-nl print(f"Élément à l'indice {i} : {element}")bksl-nl

Ici, on a mis directement len(ma_liste) à l'intérieur du range, ce qui évite de créer une variable supplémentaire.

À vous de jouer !

Exercice : Manipulation de listes

Créez un programme en réalisant les tâches suivantes :

  1. Créez une liste appelée nombres contenant les entiers de 1 à 5.
  2. Utilisez une boucle for pour parcourir la liste nombres et afficher chaque élément.
  3. Modifiez la liste nombres pour qu'elle contienne les carrés des nombres de 1 à 5. Cela signifie que la liste doit maintenant contenir [1, 4, 9, 16, 25].
  4. Utilisez à nouveau une boucle for pour parcourir la liste mise à jour et afficher chaque élément.
Aide - Code à trous

Pour vous aider, vous pouvez réutiliser ce programme à trous :

# Créer une liste contenant les entiers de 1 à 5
nombres = .......

# Parcourir la liste et afficher chaque élément
print("Liste d'origine :")
for nombre in .......:
    print(.......)

# Modifier la liste pour qu'elle contienne les carrés des nombres de 1 à 5
for ... in range(.......):
    nombres[i] = ....... ** 2

# Parcourir la liste mise à jour et afficher chaque élément
print("\nListe mise à jour :")
for nombre in .......:
    print(.......)

Vous pouvez réaliser ce programme sur Thonny ou dans l'IDE ci-dessous (vous pourrez télécharger votre programme en cliquant sur le deuxième bouton pour le conserver.)

###

Les chaînes de caractères

Les chaînes de caractères offrent une certaine ressemblance avec les tuples, dans le sens où l'on peut lire les caractères de la même manière que l'on lit les éléments d'un tuple, mais on ne peut pas modifier une chaîne de caractères.

En particulier, on peut obtenir la taille d’une chaîne de caractères, c’est-à-dire son nombre de caractères, avec l’opération len et accéder au i-ième caractère d’une chaîne avec les crochets :

###

s = "abcdefgh"bksl-nlbksl-nlprint(len(s)) # afficher la taille de la chaîne sbksl-nlbksl-nlprint(s[0]) # afficher le premier caractère de sbksl-nlbksl-nlprint(s[len(s) - 1]) # afficher le dernier caractère de sbksl-nl

Voici deux manière dont on peut parcourir une chaîne de caractères pour afficher chaque caractère un à un, le principe est le même que pour les listes et les tuples :

En récupérant directement chaque élément à chaque itération d'une boucle :

###

s = "abcdefgh"bksl-nlbksl-nlfor car in s:bksl-nl print(car)bksl-nl

En récupérant chaque élément par son indice (la variable i de la boucle prendra les valeurs 0, puis 1, puis 2, juste qu'à l'indice du dernier élément):

###

s = "abcdefgh"bksl-nlbksl-nlfor i in range(len(s)):bksl-nl print(s[i])bksl-nl

Une autre opération bien utile sur les chaînes de caractères est l'opération de concaténation. Cela permet de créer une nouvelle chaîne en combinant deux chaînes de caractères à l'aide du symbole +, par exemple (exécutez le script suivant pour observer le résultat de la concaténation de a et b):

###

a = "abc"bksl-nlb = "def"bksl-nlbksl-nlprint(a + b)bksl-nl

On peut également chercher un caractère ou même un mot dans une chaîne de caractères avec le mot-clé in :

###

s = "Bonjour tout le monde"bksl-nlbksl-nlprint("tout" in s) # affichera Truebksl-nlprint("la" in s) # affichera Falsebksl-nl