recherche

Maison  >  Questions et réponses  >  le corps du texte

python - 如何交换两个shelve对象?

假定db1, db2 是shelve对象

if switch_d1_and_db2:
    func(db1, db2)
else:
    func(db2, db1)

怎么才能改写成:

if switch_d1_and_db2:
    db1, db2 = db2, db1 # 错误写法
func(db1, db2)

db1, db2 = db2, db1 肯定是不行的,怎么改写呢

PHPzPHPz2893 Il y a quelques jours666

répondre à tous(1)je répondrai

  • 高洛峰

    高洛峰2017-04-18 09:18:15

    Bonjour, après avoir étudié cette question pendant un moment, je suis arrivé à la conclusion :

    1. C'est trop difficile à faire et la syntaxe que vous souhaitez utiliser ne correspond pas à ce que vous essayez de faire

    2. Je pense qu'il n'y a rien de mal à utiliser la méthode originale

    3. Si vous souhaitez réaliser l'échange que vous avez défini, alors j'ai une alternative qui n'est pas trop belle, vous pouvez vous référer à

    Ce qui suit est une explication des trois points ci-dessus :

    Pour le premier point, vous souhaitez :

    db1, db2 = db2, db1

    Peu importe le type d'objet db1, db2, le sens de cet échange est

    Laissez la variable db1 faire référence à l'objet initialement référencé par db2, et laissez la variable db2 faire référence à l'objet initialement référencé par db1.

    Mais ce que vous voulez faire, c'est :

    Que le contenu du db1 fichier et du db2 fichier soit échangé

    Réfléchissez bien, ces deux choses ne sont pas identiques. Pour le dire autrement, db1, db2 = db2, db1 fera uniquement échanger les éléments référencés par les variables (le nom de la variable n'est pas égal au nom du fichier de la base de données) , mais le nom de chaque fichier Le contenu n'est toujours pas échangé.

    Donc utiliser cette syntaxe pour échanger n'est pas cohérent avec l'effet que vous souhaitez obtenir.

    Je ne m’étendrai pas sur le deuxième point car c’est raisonnable, mais cela ne vous plaira peut-être pas.

    Le troisième point est que j'ai donné une alternative moins belle, qui consiste simplement à définir une classe proxy shelf de ShelfProxy. Cette classe tente de simuler le comportement de la classe Shelf (seule l'interface est similaire. ), et l'opérateur surchargé ^ est défini comme échange :

    import shelve
    
    class ShelfProxy:
    
        def __init__(self, *args, **kwargs):
            self.args = args
            self.kwargs = kwargs
            self.file = args[0]
            self.loaddb()
    
        @classmethod
        def open(cls, *args, **kwargs):
            return cls(*args, **kwargs)
    
        def __getattr__(self, name):
            return getattr(self.dic, name)
    
        def __setitem__(self, name, value):
            self.dic[name] = value
    
        def __getitem__(self, name):
            return self.dic[name]
    
        def __xor__(self, other):
            self.dic, other.dic = other.dic, self.dic
            return True
    
        def __str__(self):
            return str(self.dic)
    
        def loaddb(self):
            db = shelve.open(*self.args, **self.kwargs)
            self.dic = dict(db.items())
            db.close()
    
        def close(self):
            newdb = shelve.open(self.file, *self.args[1:], **self.kwargs)
            for key in newdb.keys():
                del newdb[key]
            for key, value in self.dic.items():
                newdb[key] = value
            newdb.close()

    Je définis ^ comme l'échange de contenu dans . La raison pour laquelle j'ai choisi ^ est simplement parce que je ne peux pas penser à un symbole plus approprié. De manière générale, la surcharge ne sera pas. fait de cette façon, et ce n'est pas très facile. Les instances d'autres classes seront renvoyées, mais j'ai fait cela pour plus de commodité et parce que vous voulez une interface simple.

    Ensuite nous définissons quelques fonctions de test :

    def define():
        db1 = ShelfProxy.open('db1', writeback=True)
        db2 = ShelfProxy.open('db2', writeback=True)
        db1['name'] = 'db1'
        db2['name'] = 'db2'
        db1.close()
        db2.close()
    
    def check():
        db1 = ShelfProxy.open('db1', writeback=True)
        db2 = ShelfProxy.open('db2', writeback=True)
        print('db1:', db1)
        print('db2:', db2)
        db1.close()
        db2.close()
    
    def switch():
        print('switch')
        db1 = ShelfProxy.open('db1', writeback=True)
        db2 = ShelfProxy.open('db2', writeback=True)
        db1 ^ db2
        db1.close()
        db2.close()

    Code de test :

    if __name__ == '__main__':
        define()
        check()
        switch()
        check()

    Résultat :

    db1: {'name': 'db1'}
    db2: {'name': 'db2'}
    switch
    db1: {'name': 'db2'}
    db2: {'name': 'db1'}

    Conclusion

    La plupart du temps, vous pouvez utiliser la même interface que Shelf pour faire fonctionner ShelfProxy, et l'effet global est similaire, mais ne pensez-vous pas qu'il est plus facile d'utiliser la méthode depuis le début après avoir autant écrit ?XD


    Questions auxquelles j'ai répondu : Python-QA

    répondre
    0
  • Annulerrépondre