-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.py
executable file
·142 lines (119 loc) · 5.36 KB
/
main.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
#!/usr/bin/env python
# -*- coding: utf-8 -*-
__version__ = "Time-stamp: <2013-01-13 13:47:48 bpagani>"
__author__ = "Bruno Pagani <[email protected]>"
"""
Fichier principal du projet, qui utilise les autres pour afficher la simulation.
"""
# On importe le fichier de la classe Systeme
from systeme import *
# Parser du fichier de configuration du systeme
def parse (sysfile) :
"""
Lecture du fichier de configuration du système afin de générer celui-ci.
"""
sys = []
i = 0
j = -1
k = 1
for line in sysfile :
# On ignore les lignes commentées ou vides
if line.startswith(('#','\n',' ')) :
k += 1
continue
if (i % 8) == 0 :
sys.append([])
j += 1
try :
x = line.split()[2]
except IndexError :
raise ValueError("Fichier système incorrect, ligne %d." % (i+k))
if not (i % 8) in {0,1} :
try :
x = float(x)
except ValueError :
raise ValueError("Valeur non float, ligne '%d'." % (i+k))
sys[j].append(x)
i += 1
return sys
if __name__=='__main__':
from optparse import OptionParser
desc = u"Simulation numérique d'un problème N-corps."
# Définition des options
parser = OptionParser(usage="%prog [options]",
version=__version__, description=desc)
parser.add_option("-s", "--systeme", type=str, default="example.txt",
help=u"Fichier contenant les informations sur le système.")
parser.add_option("-t", "--temps_relatif", type=float, default=1,
help=u"Coefficient de dilatation du temps [%default].")
parser.add_option("-f", "--fps", type=int, default=20,
help=u"Nombre d'images par seconde pour l'affichage \
[%default fps].")
parser.add_option("-c", "--calc_per_frame", type=int, default=1,
help=u"Nombre de calculs effectués pour chaque image \
affichée [%default].")
parser.add_option("-m", "--method", type=int, default=0,
help=u"Méthode d'itération du système [0,1,2].")
parser.add_option("-w", "--wait", type=float, default=0.,
help=u"Temps d'attente entre chaque image, afin de \
synchroniser l'écoulement du temps.")
parser.add_option("-T", "--temps_sys", type=float, default=0.,
help=u"Pour faire tourner le système pendant temps_sys \
(temps système) et procéder à l'affichage ensuite.")
parser.add_option("-x", "--size", type=float, default=0.,
help=u"Fenêtre d'affichage, en m [auto].")
# Déchiffrage des options et arguments
opts,args = parser.parse_args()
# Quelques tests sur les options
if opts.fps <= 0 :
parser.error(u"Le nombre d'images par seconde doit être strictement \
positif.")
periode_affichage = 1./opts.fps # Inverse du nombre d'images par seconde
try:
sysfile = open(opts.systeme)
except (IOError):
parser.error(u"Le fichier %s n'existe pas." % opts.systeme)
temps_relatif = opts.temps_relatif
if temps_relatif <= 0 :
parser.error(u"Le coefficient de dilatation du temps doit être \
strictement positif.")
calc_per_frame = opts.calc_per_frame
if calc_per_frame <= 0 :
parser.error(u"Le nombre de calculs par image doit être strictement \
positif.")
method = opts.method
if not method in {0, 1, 2} :
parser.error(u"La méthode d'itération doit être 0, 1 ou 2.")
wait = opts.wait
if wait < 0 :
parser.error(u"Le temps d'attente entre chaque image doit être \
strictement positif.")
# Note : strictement, car si wait = 0, on utilise FuncAnimation avec
# l'intervalle classique, puisque c'est le seul cas ou wait ne sert à rien.
# Donc si l'utilisateur a rentré quelque chose, il cherche à avoir le temps
# synchronisé, et par conséquent wait différent de 0.
temps_sys = opts.temps_sys
if temps_sys <> 0 :
if temps_sys < temps_relatif * periode_affichage :
parser.error(u"Le temps d'évolution du système doit être supérieur \
à %.2f fois le coefficient de dilation du temps."
% periode_affichage)
size = opts.size
if size < 0 :
parser.error(u"La taille de la figure doit être strictement positive")
# Note : strictement, car si size = 0, on ajuste automatiquement.
# Donc si l'utilisateur a rentré quelque chose, il cherche à avoir une
# taille précise, et par conséquent size différent de 0.
systeme = Systeme(parse(sysfile))
if not size :
# On détermine la taille de la figure représentant le système
for corps in systeme.corps :
k = max(abs(corps.x),abs(corps.y))
if k > size :
size = k
size = size * 1.05 # Ajout de 5% pour ne pas avoir de points sur les bords
# On importe la fonction d'animation
from animation import animation
# Et c'est parti pour le show !
animation (systeme, periode_affichage, temps_relatif, wait,
calc_per_frame, method, temps_sys, size)