Programmer un robot (2)

Série d'exercices

Cet exercice fait partie d'une série :

Rappels sur le fonctionnement du Robot

On considÚre dans cet exercice un robot se déplaçant sur une grille de dimensions finies. Initialement, il se trouve sur la case en haut à gauche de la grille et est dirigé vers la droite.

Ce robot est représenté en Python par un objet de la classe Robot. L'interface de la classe Robot est la suivante :

  • robot = Robot(4, 5) : instancie un objet de type Robot Ă©voluant dans une grille de 4 cases de haut et 5 de large. Cet objet est affectĂ© Ă  la variable robot ;

  • robot.avance() : fait avancer le robot d'une case dans la direction actuelle. Un dĂ©placement qui ferait sortir le robot de la grille est ignorĂ© ;

  • robot.droite() : fait tourner le robot d'un quart de tour vers la gauche ;
  • robot.gauche() : fait tourner le robot d'un quart de tour vers la gauche ;
  • robot.dessine_parcours() : affiche la grille, les cases dĂ©jĂ  parcourues et la position actuelle du robot dans la console.

Un objet de type Robot contient aussi un attribut grille. Il s'agit d'une liste de listes gardant la trace des cases visitées (marquées par "*") ou non (laissées vides " ").

Exemple d'utilisation d'un Robot
>>> robot = Robot(4, 3)       # la grille fait 4 case de haut sur 3 de large
>>> robot.grille              # le robot est en haut Ă  gauche
[['*', ' ', ' '], [' ', ' ', ' '], [' ', ' ', ' '], [' ', ' ', ' ']]
>>> robot.dessine_parcours()  # le robot pointe vers la droite
┌───┐
│>  │
│   │
│   │
│   │
└───┘
>>> robot.avance()
>>> robot.avance()
>>> robot.droite()
>>> robot.avance()
>>> robot.avance()
>>> robot.avance()
>>> robot.dessine_parcours()
┌───┐
│***│
│  *│
│  *│
│  v│
└───┘
>>> robot.gauche()
>>> robot.avance()
>>> robot.dessine_parcours()
┌───┐
│***│
│  *│
│  *│
│  >│
└───┘
>>> robot.grille
[['*', '*', '*'], [' ', ' ', '*'], [' ', ' ', '*'], [' ', ' ', '*']]
La classe Robot
MOUVEMENTS = ((0, 1), (1, 0), (0, -1), (-1, 0))


class Robot:
    def __init__(self, hauteur, largeur):
        self.hauteur = hauteur
        self.largeur = largeur
        self.grille = [[" " for _ in range(largeur)] for _ in range(hauteur)]
        self.i = 0
        self.j = 0
        self.grille[self.i][self.j] = "*"
        self.direction = 0

    def avance(self):
        """Fait avancer le robot d'une case (seulement si possible)"""
        di, dj = MOUVEMENTS[self.direction]
        if 0 <= self.i + di < self.hauteur and 0 <= self.j + dj < self.largeur:
            self.i += di
            self.j += dj
            self.grille[self.i][self.j] = "*"

    def droite(self):
        """Fait tourner le robot d'un quart de tour vers la droite"""
        self.direction = (self.direction + 1) % 4

    def gauche(self):
        """Fait tourner le robot d'un quart de tour vers la gauche"""
        self.direction = (self.direction - 1) % 4

    def dessine_parcours(self):
        """Affiche les cases parcourues et la position actuelle du robot"""
        affichage = [
            ["" for _ in range(self.largeur + 2)] for _ in range(self.hauteur + 2)
        ]
        for j in range(1, self.largeur + 1):
            affichage[0][j] = "─"
            affichage[-1][j] = "─"
        for i in range(1, self.hauteur + 1):
            affichage[i][0] = "│"
            affichage[i][-1] = "│"
        affichage[0][0] = "┌"
        affichage[-1][0] = "└"
        affichage[0][-1] = "┐"
        affichage[-1][-1] = "┘"
        for i in range(self.hauteur):
            for j in range(self.largeur):
                affichage[1 + i][1 + j] = self.grille[i][j]
        affichage[self.i + 1][self.j + 1] = [">", "v", "<", "^"][self.direction]
        print("\n".join("".join(ligne) for ligne in affichage))

La classe Robot est déjà chargée dans l'éditeur, vous pouvez l'utiliser sans l'importer.

On programme ce robot en lui faisant effectuer différentes actions :

  • le code "A" fait avancer le robot ;
  • le code "D" le fait tourner vers la droite ;
  • le code "G" le fait tourner vers la gauche.
Fonction decoupe

On fournit une fonction decoupe rudimentaire permettant de transformer une chaĂźne de caractĂšres formant une suite d'instructions en une liste.

def decoupe(instructions):
    return [int(c) if c.isnumeric() else c for c in instructions]

decoupe("A3GD5")  # renvoie ['A', 3, 'G', 'D', 5]

Cette fonction est déjà importée dans l'éditeur. Vous pouvez l'utiliser pour écrire des tests personnels.

Cette fonction ne gÚre que les entiers positifs strictement inférieurs à 10 et ne filtre pas les instructions incorrectes. Ainsi decoupe("Z12") renvoie ['Z', 1, 2].

Le premier exercice de la série a pour objectif d'écrire une version valide de cette fonction.

Le fonctionnement détaillé ci-dessus et utilisé dans le premier exercice nécessite d'écrire de longues séries d'instructions. Afin de les alléger, on autorise aussi des suites d'instructions du type [action, n] dans lesquelles :

  • action est une des trois actions dĂ©crites ci-dessus ;
  • n est un nombre entier positif ou nul.

Ainsi, les instructions ["A", 8, "D", 41, "A"] ont pour effet :

  • de faire avancer le robot de 8 cases,
  • de faire tourner le robot sur la droite 41 fois,
  • de faire enfin avancer le robot d'une seule case.

Lors d'une répétition, on exécute un maximum d'actions valides. Si par exemple, le robot ne peut avancer que de 5 cases, les instructions ["A", 8] le feront avancer de 5 cases, seules les trois derniÚres actions seront ignorées.

Notez-bien que les instructions simples (non suivies d'un nombre entier) restent valides.

Écrire la fonction execute qui prend en paramùtres un objet de type Robot et une liste d'instructions et fait effectuer chacune de ces instructions par le robot.

On garantit que toutes les instructions sont valides ("A", "D" ou "G" ou un entier positif ou nul) et que la suite d'instructions est bien formée (chaque entier suit une action valide).

Exemple
>>> robot = Robot(4, 5)
>>> instructions = ["A", 8, "D", 41, "A"]
>>> execute(robot, instructions)
>>> robot.dessine_parcours()
┌─────┐
│*****│
│    v│
│     │
│     │
└─────┘
Aide

On pourra construire le code autour d'une boucle qui lit les instructions tant que l'on n'a pas atteint la fin de la liste.

La premiĂšre instruction est toujours une action ("A", "D" ou "G").

Si l'instruction suivante est un entier, il faut répéter cette action autant de fois que nécessaire. Dans le cas contraire, on effectue l'action une seule fois.

On rappelle que si la variable n a pour valeur un nombre entier, l'instruction isinstance(n, int) est évaluée à True.

###(Dés-)Active le code aprÚs la ligne # Tests (insensible à la casse)
(Ctrl+I)
Entrer ou sortir du mode "deux colonnes"
(Alt+: ; Ctrl pour inverser les colonnes)
Entrer ou sortir du mode "plein écran"
(Esc)
Tronquer ou non le feedback dans les terminaux (sortie standard & stacktrace / relancer le code pour appliquer)
Si activĂ©, le texte copiĂ© dans le terminal est joint sur une seule ligne avant d'ĂȘtre copiĂ© dans le presse-papier
Évaluations restantes : 10/10

.1280135[4})2R,a- i8mG16l7.e:9A;S/dktf{+Ir3sogu0x]Ppnh=céyvD(wq_b050C0v0E0j0m0s0L0l0X0s0j0L0L0W010E0m0T010406050L0O0o0o0j0J0Z040A0M0s0O0~0M0U0l020j0o0T0z0l0h0v180J0(0O0v0L050B1517191b130T041z1G051J0B1J1L1G130C0m0!0?0^0`0|0V0m0N0V0s1Z0V0E11050.0*0s0v1U0_0{011Y1!1$1!0E1,1.1*0E0*0M0C1b1+0J1H0E0V0?1e0L0T0j0U0|0g011:1W010F0:0v0U1m0v1*2b2d2i1=2l1.2o0o2q040a0l0S0J0M0T0M0L0m1h1j0,290J0J0v0X2L1z2s0U1H0B272X0E2524260C2u0|1$0U2n2I1*1R1T0@1;2+0m2-0U211S1*0T2Q1H2V2X32142c1j2?2j2{0J180s110l0q2U3612352t381=3a3c3e0g3h2d3j2V2*013o0j3d040l0K3s2W133v3m0|3y3A0l0d3E3u363w3K3e0b3O3G3Q3I3x0M3b3z3e0r3V3k371V3n3!3p3B0t3)3H3,3J3.3$3B0n3=3X3@3Z3#3L0x3}3l3 3S040q0P443+2@403/0q3g1A3i3W454d470q3r4i3t4k4c393_3A0q3D4q3F3*3R4v110q3N4z3P4l4u414E3U4H4s4C4L483(4O4B3Y4n3;4U3?4m4D483|4Z3~4#4R0q434)4J3-4R0g4a4H1I301z2;2!0C2(3w0X212A0+1S1H2 0v313i3O05510,594:0|0D110,0F5b4!2j0%3e5m4*390F110v0Q2R1g0v5r5g0110040$5B4t3n112F200E5H3w5E0i3O0l4V46112_0L2#0O2S0m1i1y4^5n1=5E0f0w3V0l5:5T5*3J5W5S5U4d0M110W5_5?010o0m114@32065;5=5s1=5i040%1Y1.5 6a5@040m6h5C5|04020s0E0z6m5I3J0*112x5O3Y5E5G5)6i3x5W0U5Y0J5!2L5(34605,5.4O68685`2j0X0q11030l1.2S0O2Q0l2N0s005X5Z5#1i5/6T5:6V5J6k6I5Z6u3w6o5~4H695C0U6H6J6L5$6I6A3 5E0c7a4m5^6E5C5E0R6=6?6^5h5W5l717n6G6k6}3Y6o6q6s7v466x046z7h6v5D116D6O6F746`766:797F5P110f7A5{110k7V2j624E7Z1=0M5p042d0C7%6w5W2J6{2d0X5A7R6B7I7e39756/6M7|5+117d7_5V7u7r606o0H7.616348810|7j5R887L7C2_5N854d6Q7l6?5;7s7M0U8c6 8c7M6.6K7P6N5a6P838g7t6l8k6n118b8M7G7#8f8p2j7j8s8t727G8B8y8O70328Z3w8S4p668Y7s6c0v0;7^7K7i116R8.8Y8u608w8$048(3i8*3Y8S4h8|8}95868L8)7s8a933t9b4d973V678t7s6X6Z0l5w0Y0X1g786*1/6,8C776;6S9o606c0F3!8A110)8y7*2_8A7C0J2d0N8@8G6F6C8J908U82045-8X9F6F6c0m7q9e8 7~6K910W9h2W9j2j0L2g04010#018J5E8{4j9a7m9:045L0M8o8^7G6o0u9Z5j2F2K9V3t7s6C7U9E9a8:5v1$9.948v9;0O9?9^3B7s9|11010pa19#8h8`9)ara8aaac9W8N04agaJ7t0N0j5!0Val2Wan7Iap998}as048=0La#5f7Ga3aMa+aO0M5M8Jafah7+0!7?a:a%5Fa)4j1z5d584_b90B4|1z0E4~be2$2Y20222!0j1-bb4|1Fa;3w2Q0o0)0F0j0D0v0)0V0K111r1t1v1x0la4am1M3j1G0I0s9s5Y0laY0`0m0l2HbVbo6*000O0U9-0m0v0J6#bI9I3#0E0=0,0=9B8E9s1j2 0M0X0Y0C2d0EbT5XbW6R1P542=3 1@1#1%1)bs3Y2w2n2p112C0A0X0J0 c20S0Z279D3457a;335ab8cf3 6c5k8J7*5TaV0U5ua-5x9v0Eb28H5Fa~aPa2118j9/7Lay8EcU9%bK3F6Ua82c0m2Q8F9i9f5}8c7c7kaq9`6_9daw89c;8Q8+8e65a58~9+116e2m9K87cXaS7y6tc 4W7C7Ead7ScRcIcZ80aV8rc^c)cY7N6|df3 8zdw7fdu8Ddodj7`0484dE9cc#c@a*ds5C9,avc/a8c{dR6F7x6rdedb8!dh2nc#7JaR8!dn78c.a$cQb5dUaS7Ydz7!8e98c|dV7*7,9Q7:6.b1d%a~b^dDd)dkdHe74W7gdZ6~8O8c9ldp110RcWd{738m0UaQamd/c%12a6c_6jc+c-a|11aUdIdA0j0T0T2n7-ei5Fd(erdt9Bc#eld;d*dB9C7QeD8V8Idmdaemaeefd@1=eheX9$0R0fd:c(a6axe#eSee040HaBew8d118-d3a7d5a-8?c#et9naNdtey1xeAaTa~eFeH0UeJe,aKeLe46{9=eKeR9_7se+ea7b7Te:eue=dS91e{egd_9mdNeT9I0U7P9?c=110G8JaE9~0yaIfk7H04ete}7McTaVa}e!0jb0fKcP9XcV8cfR9 fUfv8qaLe)6jf#fVf%fV7M0CajcOeQf/9}aGf=eN8_fXd9f{f?2jf}gd6_aXaZf,g90e8sa,b/d90jfLfqd98xf_017)6Hd9fbd-cBf@gadrd4dO11gpgw7M9Mgwgy6`d a99Tgka=7{e!gvfVdqdM9*73gJ1igsgZeZf~11gr2TeK0R0$fy130BcAba2Xbq4{0-0/0;04.

###(Dés-)Active le code aprÚs la ligne # Tests (insensible à la casse)
(Ctrl+I)
Entrer ou sortir du mode "deux colonnes"
(Alt+: ; Ctrl pour inverser les colonnes)
Entrer ou sortir du mode "plein écran"
(Esc)
Tronquer ou non le feedback dans les terminaux (sortie standard & stacktrace / relancer le code pour appliquer)
Si activĂ©, le texte copiĂ© dans le terminal est joint sur une seule ligne avant d'ĂȘtre copiĂ© dans le presse-papier
Évaluations restantes : 10/10

.1280135[4})2R,a- i8mG16l7.e:9A;S/dktf{+Ir3sogu0x]Ppnh=céyvD(wq_b050C0v0E0j0m0s0L0l0X0s0j0L0L0W010E0m0T010406050L0O0o0o0j0J0Z040A0M0s0O0~0M0U0l020j0o0T0z0l0h0v180J0(0O0v0L050B1517191b130T041z1G051J0B1J1L1G130C0m0!0?0^0`0|0V0m0N0V0s1Z0V0E11050.0*0s0v1U0_0{011Y1!1$1!0E1,1.1*0E0*0M0C1b1+0J1H0E0V0?1e0L0T0j0U0|0g011:1W010F0:0v0U1m0v1*2b2d2i1=2l1.2o0o2q040a0l0S0J0M0T0M0L0m1h1j0,290J0J0v0X2L1z2s0U1H0B272X0E2524260C2u0|1$0U2n2I1*1R1T0@1;2+0m2-0U211S1*0T2Q1H2V2X32142c1j2?2j2{0J180s110l0q2U3612352t381=3a3c3e0g3h2d3j2V2*013o0j3d040l0K3s2W133v3m0|3y3A0l0d3E3u363w3K3e0b3O3G3Q3I3x0M3b3z3e0r3V3k371V3n3!3p3B0t3)3H3,3J3.3$3B0n3=3X3@3Z3#3L0x3}3l3 3S040q0P443+2@403/0q3g1A3i3W454d470q3r4i3t4k4c393_3A0q3D4q3F3*3R4v110q3N4z3P4l4u414E3U4H4s4C4L483(4O4B3Y4n3;4U3?4m4D483|4Z3~4#4R0q434)4J3-4R0g4a4H1I301z2;2!0C2(3w0X212A0+1S1H2 0v313i3O05510,594:0|0D110,0F5b4!2j0%3e5m4*390F110v0Q2R1g0v5r5g0110040$5B4t3n112F200E5H3w5E0i3O0l4V46112_0L2#0O2S0m1i1y4^5n1=5E0f0w3V0l5:5T5*3J5W5S5U4d0M110W5_5?010o0m114@32065;5=5s1=5i040%1Y1.5 6a5@040m6h5C5|04020s0E0z6m5I3J0*112x5O3Y5E5G5)6i3x5W0U5Y0J5!2L5(34605,5.4O68685`2j0X0q11030l1.2S0O2Q0l2N0s005X5Z5#1i5/6T5:6V5J6k6I5Z6u3w6o5~4H695C0U6H6J6L5$6I6A3 5E0c7a4m5^6E5C5E0R6=6?6^5h5W5l717n6G6k6}3Y6o6q6s7v466x046z7h6v5D116D6O6F746`766:797F5P110f7A5{110k7V2j624E7Z1=0M5p042d0C7%6w5W2J6{2d0X5A7R6B7I7e39756/6M7|5+117d7_5V7u7r606o0H7.616348810|7j5R887L7C2_5N854d6Q7l6?5;7s7M0U8c6 8c7M6.6K7P6N5a6P838g7t6l8k6n118b8M7G7#8f8p2j7j8s8t727G8B8y8O70328Z3w8S4p668Y7s6c0v0;7^7K7i116R8.8Y8u608w8$048(3i8*3Y8S4h8|8}95868L8)7s8a933t9b4d973V678t7s6X6Z0l5w0Y0X1g786*1/6,8C776;6S9o606c0F3!8A110)8y7*2_8A7C0J2d0N8@8G6F6C8J908U82045-8X9F6F6c0m7q9e8 7~6K910W9h2W9j2j0L2g04010#018J5E8{4j9a7m9:045L0M8o8^7G6o0u9Z5j2F2K9V3t7s6C7U9E9a8:5v1$9.948v9;0O9?9^3B7s9|11010pa19#8h8`9)ara8aaac9W8N04agaJ7t0N0j5!0Val2Wan7Iap998}as048=0La#5f7Ga3aMa+aO0M5M8Jafah7+0!7?a:a%5Fa)4j1z5d584_b90B4|1z0E4~be2$2Y20222!0j1-bb4|1Fa;3w2Q0o0)0F0j0D0v0)0V0K111r1t1v1x0la4am1M3j1G0I0s9s5Y0laY0`0m0l2HbVbo6*000O0U9-0m0v0J6#bI9I3#0E0=0,0=9B8E9s1j2 0M0X0Y0C2d0EbT5XbW6R1P542=3 1@1#1%1)bs3Y2w2n2p112C0A0X0J0 c20S0Z279D3457a;335ab8cf3 6c5k8J7*5TaV0U5ua-5x9v0Eb28H5Fa~aPa2118j9/7Lay8EcU9%bK3F6Ua82c0m2Q8F9i9f5}8c7c7kaq9`6_9daw89c;8Q8+8e65a58~9+116e2m9K87cXaS7y6tc 4W7C7Ead7ScRcIcZ80aV8rc^c)cY7N6|df3 8zdw7fdu8Ddodj7`0484dE9cc#c@a*ds5C9,avc/a8c{dR6F7x6rdedb8!dh2nc#7JaR8!dn78c.a$cQb5dUaS7Ydz7!8e98c|dV7*7,9Q7:6.b1d%a~b^dDd)dkdHe74W7gdZ6~8O8c9ldp110RcWd{738m0UaQamd/c%12a6c_6jc+c-a|11aUdIdA0j0T0T2n7-ei5Fd(erdt9Bc#eld;d*dB9C7QeD8V8Idmdaemaeefd@1=eheX9$0R0fd:c(a6axe#eSee040HaBew8d118-d3a7d5a-8?c#et9naNdtey1xeAaTa~eFeH0UeJe,aKeLe46{9=eKeR9_7se+ea7b7Te:eue=dS91e{egd_9mdNeT9I0U7P9?c=110G8JaE9~0yaIfk7H04ete}7McTaVa}e!0jb0fKcP9XcV8cfR9 fUfv8qaLe)6jf#fVf%fV7M0CajcOeQf/9}aGf=eN8_fXd9f{f?2jf}gd6_aXaZf,g90e8sa,b/d90jfLfqd98xf_017)6Hd9fbd-cBf@gadrd4dO11gpgw7M9Mgwgy6`d a99Tgka=7{e!gvfVdqdM9*73gJ1igsgZeZf~11gr2TeK0R0$fy130BcAba2Xbq4{0-0/0;04.