IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

Python Discussion :

Import variable globale [Python 3.X]


Sujet :

Python

Vue hybride

Message précédent Message précédent   Message suivant Message suivant
  1. #1
    Membre éclairé
    Profil pro
    Inscrit en
    Février 2010
    Messages
    573
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Février 2010
    Messages : 573
    Par défaut Import variable globale
    Bonjour,

    Est ce que vous savez s'il est possible dans un fichier python de regrouper toutes les variables globales que l'on utilise et ensuite faire un import de ce fichier ?

    Juste histoire de ne pas faire des copier coller de ces variables globales dans tous les fichier *.py

    Merci pour votre aide

    Thais

  2. #2
    Membre Expert
    Homme Profil pro
    Développeur informatique
    Inscrit en
    Février 2003
    Messages
    1 603
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations professionnelles :
    Activité : Développeur informatique
    Secteur : Industrie

    Informations forums :
    Inscription : Février 2003
    Messages : 1 603
    Par défaut
    Bonjour.

    Avez-vous pensé aux variables d'environnement ?

    Exemple de fichier .env :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    var1=valeur
    var2=valeur
    va3=valeur
    Sous Windows, la librairie python-dotenv sera à installer mais si vous êtes sous Linux/Unix, normalement pas besoin.

    Et une p'tite classe Config par exemple, qui se charge d'instancier toutes ces variables :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    from os import getenv
    from dotenv import load_dotenv # si Windows
     
    load_dotenv() # si Windows
     
    class Config:
        def __init__(self) -> None:
            self.var1 = getenv('var1', default='valeur par defaut')
            self.var2 = getenv('var2', default='valeur par defaut')
            self.var3 = getenv('var3', default='valeur par defaut')
    Bien sûr, c'est un exemple parmi tant d'autres.

  3. #3
    Expert éminent
    Homme Profil pro
    Architecte technique retraité
    Inscrit en
    Juin 2008
    Messages
    21 737
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Manche (Basse Normandie)

    Informations professionnelles :
    Activité : Architecte technique retraité
    Secteur : Industrie

    Informations forums :
    Inscription : Juin 2008
    Messages : 21 737
    Par défaut
    Salut,

    Citation Envoyé par thais781 Voir le message
    Est ce que vous savez s'il est possible dans un fichier python de regrouper toutes les variables globales que l'on utilise et ensuite faire un import de ce fichier ?
    Techniquement, on pourrait le faire en ajoutant une instruction du genre INCLUDE: on ajoute du texte avant d'exécuter le module.

    Mais il serait plus simple de fabriquer de vraies variables globales en utilisant la technique décrite dans la FAQ Python.

    Ceci dit, les variables globales demandent des précautions pour être utilisées à défaut, c'est prendre le risque de passer (et perdre) beaucoup de temps pour la mise au point du programme.


    - W
    Architectures post-modernes.
    Python sur DVP c'est aussi des FAQs, des cours et tutoriels

  4. #4
    Membre prolifique
    Avatar de Sve@r
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Février 2006
    Messages
    12 830
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Oise (Picardie)

    Informations professionnelles :
    Activité : Ingénieur développement logiciels
    Secteur : Aéronautique - Marine - Espace - Armement

    Informations forums :
    Inscription : Février 2006
    Messages : 12 830
    Billets dans le blog
    1
    Par défaut
    Bonjour
    Citation Envoyé par thais781 Voir le message
    Juste histoire de ne pas faire des copier coller de ces variables globales dans tous les fichier *.py
    Si tu utilises les globales pour transmettre une info entre X et Y c'est mal. Si tu utilises des globales juste pour gérer une espèce de config alors ok. Mais dans ce cas, pas besoin de globales.
    Dans mes projets j'ai deux type d'objet
    • un objet "cEnv" qui contient l'environnement du projet (nom de l'application, nom de la bdd, etc etc etc)
    • un objet "cConfig" qui contient la configuration du projet (paramètres, distance, nb de jour avant fin de X, etc)

    L'objet "cEnv" est statique. L'objet "cConfig" charge un fichier ".ini" via la lib configparser.
    Et dans mon programme, me suffit d'importer les sources contenant ces deux objets pour avoir accès à leurs attributs. Ce qui évite de devoir se coltiner des globales
    Mon Tutoriel sur la programmation «Python»
    Mon Tutoriel sur la programmation «Shell»
    Sinon il y en a pleins d'autres. N'oubliez pas non plus les différentes faq disponibles sur ce site
    Et on poste ses codes entre balises [code] et [/code]

  5. #5
    Expert confirmé Avatar de papajoker
    Homme Profil pro
    Développeur Web
    Inscrit en
    Septembre 2013
    Messages
    2 323
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Nièvre (Bourgogne)

    Informations professionnelles :
    Activité : Développeur Web
    Secteur : High Tech - Multimédia et Internet

    Informations forums :
    Inscription : Septembre 2013
    Messages : 2 323
    Par défaut
    bonjour
    Citation Envoyé par thais781 Voir le message
    Juste histoire de ne pas faire des copier coller de ces variables globales dans tous les fichier *.py
    Du travail de refactoring ! très très bien

    Mais comme dit plus haut, il faut surtout qu'il soit bien fait... sinon autant ne rien faire

    Tu sembles ici, simplement vouloir gagner du temps. Donc écrire ton propre script va-t-il t'en faire gagner ? je ne pense pas ou bien ton projet est très complexe ?

    Sinon, tu peux parcourir tous tes .py, avec une regex du type ligne qui débute par <MOT> suivit de '=' avec éventuellement un ou des espaces.
    Tu sauvegardes ces "globales" dans une liste, puis tu crées un fichier init (ou fichier config.py) où tu les écris.

    note:
    Et si tu n'as pas respecté PEP 8 alors un linter type pylint peut retrouvé tes "variables globales"
    Citation Envoyé par pylint;
    (fichier)(ligne) C0103: Constant name "xxx" doesn't conform to UPPER_CASE naming style

  6. #6
    Membre prolifique
    Avatar de Sve@r
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Février 2006
    Messages
    12 830
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Oise (Picardie)

    Informations professionnelles :
    Activité : Ingénieur développement logiciels
    Secteur : Aéronautique - Marine - Espace - Armement

    Informations forums :
    Inscription : Février 2006
    Messages : 12 830
    Billets dans le blog
    1
    Par défaut
    Citation Envoyé par papajoker Voir le message
    Sinon, tu peux parcourir tous tes .py, avec une regex du type ligne qui débute par <MOT> suivit de '=' avec éventuellement un ou des espaces.
    Tu sauvegardes ces "globales" dans une liste, puis tu crées un fichier init (ou fichier config.py) où tu les écris.
    Ou tu passes par configparser qui fait tout ça tout seul (et qui gère aussi la notion de "chapitre" dans le fichier)
    Mon Tutoriel sur la programmation «Python»
    Mon Tutoriel sur la programmation «Shell»
    Sinon il y en a pleins d'autres. N'oubliez pas non plus les différentes faq disponibles sur ce site
    Et on poste ses codes entre balises [code] et [/code]

  7. #7
    Expert confirmé Avatar de papajoker
    Homme Profil pro
    Développeur Web
    Inscrit en
    Septembre 2013
    Messages
    2 323
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Nièvre (Bourgogne)

    Informations professionnelles :
    Activité : Développeur Web
    Secteur : High Tech - Multimédia et Internet

    Informations forums :
    Inscription : Septembre 2013
    Messages : 2 323
    Par défaut
    Nous sommes tout à fait d'accord
    Citation Envoyé par papajoker Voir le message
    il faut surtout qu'il soit bien fait... sinon autant ne rien faire
    Je n'ai répondu qu'à la question "j'ai la flemme de faire des copier/coller". Une fois les variables/constantes regroupées, commence le véritable travail de refactoring.

  8. #8
    Expert confirmé
    Avatar de tyrtamos
    Homme Profil pro
    Retraité
    Inscrit en
    Décembre 2007
    Messages
    4 486
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Var (Provence Alpes Côte d'Azur)

    Informations professionnelles :
    Activité : Retraité

    Informations forums :
    Inscription : Décembre 2007
    Messages : 4 486
    Billets dans le blog
    6
    Par défaut
    Bonjour

    Vouloir conserver des données modifiées d'une session à l'autre est un problème général. Pour ça, j'utilise les fichiers "ini" (module Python "configparser"):

    A l'installation du programme, il doit y avoir un fichier .ini avec le programme installé qui donne les données initialisées (une sorte de "configuration usine").

    Si les données à conserver d'une session à l'autre sont spécifiques à l'un des utilisateurs, le fichier ini devra se trouver dans son "home". Sinon, dans le home "public". Trouver le home de l'utilisateur sans connaître son nom est facile, tant sous Windows que sous Linux:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    rephome = os.path.expanduser('~')
    Au lancement du programme, celui-ci commence par chercher le programme ini dans le "home" (le répertoire) de l'utilisateur. Si le fichier ini s'y trouve, on le charge.

    Si le fichier ini ne se trouve pas dans le répertoire utilisateur, c'est que c'est la 1ère exécution. On charge donc les données d'initialisation.

    Le programme se déroule, et certaines données sont modifiées.

    A la fermeture du programme, les données modifiées sont placées dans le dictionnaire ordonné, et celui-ci est enregistré dans le fichier ini du home de l'utilisateur.

    Pour le code, voilà ce que j'ai fait. On met ce code dans un fichier (que j'appelle "bibini.py") qu'il suffira d'importer dans tous les programmes qui l'utilisera:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    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
    # -*- coding: utf-8 -*-
     
    """
    Charge et sauve le contenu d'un fichier de type .ini sous la forme d'un
    dictionnaire de dictionnaires (OrderedDict) en mémoire
    """
     
    from configparser import ConfigParser
    from collections import OrderedDict
     
    #############################################################################
    def chargini(ficini, encodage="utf-8"):
        """Charge le contenu du fichier ficini (type .ini) avec l'encodage donné
           Retourne le dictionnaire de dictionnaires (OrderedDict) des données
        """
        cfg = ConfigParser(interpolation=None)
        with open(ficini, 'r', encoding=encodage) as fsrce:
            cfg.read_file(fsrce)
        dicini = OrderedDict()
        for section in cfg.sections():
            dicini[section] = OrderedDict()
            for option, valeur in cfg.items(section):
                dicini[section][option] = valeur
        return dicini
     
    #############################################################################
    def sauvini(dicini, ficini, encodage="utf-8"):
        """Sauve le dictionnaire de dictionnaires (OrderedDict) dicini dans le
           fichier ficini avec l'encodage donné
        """
        cfg = ConfigParser(interpolation=None)
        cfg.read_dict(dicini, source='<OrderedDict>')
        with open(ficini, 'w', encoding=encodage) as fdest:
            cfg.write(fdest)
    J'utilise un dictionnaire ordonné, afin que les données du fichier ini soit toujours dans le même ordre.

    la fonction chargini charge le contenu du fichier ini donné "ficini", et retourne un dictionnaire ordonné "dicini".

    La fonction sauvini enregistre le dictionnaire ordonné "dicini" dans le fichier "ficini".

    L'avantage supplémentaire de cette solution est que l'on peut consulter le fichier ini avec un simple éditeur de texte (avec le bon encodage, "utf-8" ici) et, si nécessaire, le modifier en dehors d'une session du programme.

  9. #9
    Membre éclairé
    Profil pro
    Inscrit en
    Février 2010
    Messages
    573
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Février 2010
    Messages : 573
    Par défaut
    Bonjour à tous,

    Merci beaucoup pour votre aide.
    Le but est juste de factoriser des infos sur les PATH que j'utilise et de ne devoir les modifier qu'a un seul endroit si besoin

    La solution de Sve@r me semble la plus adapté à ma problématique en mode complètement statique
    Tu peux me donner un exemple de ton fichier 'cEnv' et comment tu l'appel/charge dans tes autres fichiers ??

    Encore merci pour ces réponses de très haut niveau ;-)

    Thais

  10. #10
    Expert confirmé Avatar de papajoker
    Homme Profil pro
    Développeur Web
    Inscrit en
    Septembre 2013
    Messages
    2 323
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Nièvre (Bourgogne)

    Informations professionnelles :
    Activité : Développeur Web
    Secteur : High Tech - Multimédia et Internet

    Informations forums :
    Inscription : Septembre 2013
    Messages : 2 323
    Par défaut
    Citation Envoyé par thais781 Voir le message
    Le but est juste de factoriser des infos sur les PATH
    Le shéma (mot faible) que tu montres n'a rien d'exceptionnel, par exemple avec django il est plus que classique d'avoir une arborescence de ton type avec 3 niveaux.

    MAIS

    que le fichier de dernier niveau ait besoin de constantes "path" n'est pas du tout normal ! Normalement les fonctions(classes) de haut niveau sont justement au début et, au 3éme niveau on ne trouve que des fonctions(classes) de très bas niveau. Seules les fonctions de haut niveau ont besoin d'un environnement de travail.

    Haut niveau exemple : Application.sauvegarder()
    Bas niveau exemple : multiplier_facteur_x(x,y) , assainir_titre(), filtrer_dataframe(filtre_fonction, colonne), download_url() Et justement le travail sur panda est regroupé, le travail sur le web est regroupé, le travail sur ... Puisque tu les regroupes, tu peux parfois créer des classes "haut niveau" pour ces tâches (cela dépend beaucoup de ton app, donc pas forcément vrai)

    -------
    Et vouloir regrouper tes 4 librairies ... me fait penser que c'est (encore) une mauvaise chose puisque ce ne sont pas des librairies qui font les mêmes choses et devraient justement ne pas êtres toutes utiles dans les fichiers de bas niveau.

  11. #11
    Expert confirmé Avatar de papajoker
    Homme Profil pro
    Développeur Web
    Inscrit en
    Septembre 2013
    Messages
    2 323
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Nièvre (Bourgogne)

    Informations professionnelles :
    Activité : Développeur Web
    Secteur : High Tech - Multimédia et Internet

    Informations forums :
    Inscription : Septembre 2013
    Messages : 2 323
    Par défaut
    pour ta fonction chargini(),
    je vais préférer passer des valeurs "usine" par défaut (injecter). Et ne pas utiliser open() mais read() si le fichier n'existe pas encore

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    def chargini(ficini, defaults=None, encodage="utf-8"):
        cfg = configparser.ConfigParser(interpolation=None)
        if defaults:
            cfg.read_dict(defaults)
        cfg.read(ficini, encoding=encodage)
        ret = {}
        for section in cfg.sections():
            for key, value in cfg[section].items():
                ret.setdefault(section, {})[key] = value
        return ret
     
    ini = chargini("truc.ini", {'dev': {'debug': 0}})
    Citation Envoyé par thais781 Voir le message
    Bonjour à tous,
    ma problématique en mode complètement statique
    Tu peux me donner un exemple de ton fichier 'cEnv' et comment tu l'appel/charge dans tes autres fichiers ??
    ici, on parles donc de constantes !
    ?? c'est ce que tu fais déjà ? si tu as déjà quelques fichiers.

    Un fichier est un objet (en python), il te suffit d'y mettre tes constantes (ce que je te disais)

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    #fichier config.py
      VAR_ONE = 1
     
    # autre fichier
    import config
       print(config.VAR_ONE)
    Parfois, on peut créer une classe "statique" dans ce fichier, mais ce n'est pas une solution pythonnesque, en particulier pour des constantes et cela n'apporte rien de plus (sauf créer X classes si tu as X types de constantes).

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    # constantes.py
    class Config:
        VAR_ONE = 1
        VAR_TWO = 2
     
    # autre fichier
    from constantes import Config
       print(Config.VAR_ONE)

  12. #12
    Expert confirmé
    Avatar de tyrtamos
    Homme Profil pro
    Retraité
    Inscrit en
    Décembre 2007
    Messages
    4 486
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Var (Provence Alpes Côte d'Azur)

    Informations professionnelles :
    Activité : Retraité

    Informations forums :
    Inscription : Décembre 2007
    Messages : 4 486
    Billets dans le blog
    6
    Par défaut
    Citation Envoyé par papajoker Voir le message
    pour ta fonction chargini(),
    je vais préférer passer des valeurs "usine" par défaut (injecter) et ne pas utiliser open() si le fichier n'existe pas encore
    Dans ce que je fais, le test d'existence du fichier est fait avant avec os.access(...).

+ Répondre à la discussion
Cette discussion est résolue.

Discussions similaires

  1. [Python 3.X] Variable globale et import de fichiers
    Par ypcman dans le forum Général Python
    Réponses: 2
    Dernier message: 04/11/2021, 20h14
  2. Cacher Variable Url avec Variable Globale
    Par danuz dans le forum XMLRAD
    Réponses: 6
    Dernier message: 26/09/2003, 17h20
  3. Variable "globale" (dans plusieurs templa
    Par mattmat dans le forum XSL/XSLT/XPATH
    Réponses: 3
    Dernier message: 17/06/2003, 19h22
  4. question sur les variables globales et les thread posix
    Par souris_sonic dans le forum POSIX
    Réponses: 5
    Dernier message: 13/06/2003, 13h59
  5. les variables globales static
    Par gRRosminet dans le forum C
    Réponses: 8
    Dernier message: 27/04/2002, 08h34

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo