什么时候装饰器(类)对象被破坏?

时间:2016-10-07 15:10:47

标签: python database-connection decorator python-decorators

我正在使用装饰器打开和关闭neo4j数据库会话(并允许我的装饰函数在该会话中运行查询)。起初我使用了装饰器功能:

 from neo4j.v1 import GraphDatabase, basic_auth
 def session(func,url="bolt://localhost:7474", user="neo4j",pwd="neo4j", *args, **kwargs):
     def operate(*args, **kwargs):
         driver = GraphDatabase.driver(url, auth=basic_auth(user,pwd))
         session=driver.session()
         kwargs["session"]=session 
         result=func(*args, **kwargs)
         session.close()
         return result
     return operate

例如我然后调用此函数:

@session
def RUN(command,session):
    result=session.run(command)
    return(result)

但是,这会为每个耗费资源的查询打开和关闭会话。因此,我尝试创建一个装饰器类,并存储会话:

class session(object):
    def __init__(self, func, url="bolt://localhost", user="neo4j", pwd="neo4j"):
        try:
            driver = GraphDatabase.driver(url, auth=basic_auth(user, pwd))
            print("session opened")
        except:
            print("Exception during authentification")
            self.__exit__()
        else:
            session=driver.session()
            self.func=func
            self.SESSION=session

    def __call__(self, *args, **kwargs):
        kwargs["session"]=self.SESSION 
        result=self.func(*args, **kwargs)
        return result
    def __del__(self):
        print("del")
        try:
            self.SESSION.close()
            print("session closed")
        except:
            print("could not close session") 

这似乎有效,因为"会话已开启"只出现一次。但是会话似乎没有结束("会话关闭"永远不会打印)。

所以我的第一个问题如下,如何在破坏装饰器时调用self.SESSION.close()?

我也想知道我是否理解我的代码正在做什么。当我调用RUN等装饰函数时,只创建了一个会话对象?如果我有另一个装饰函数MATCH

,该怎么办?
@session
def MATCH(*args,**kwargs):
    pass

会话对象是否相同?

1 个答案:

答案 0 :(得分:1)

如何

这是一个使用函数创建带参数的装饰器的模板:

def decorator_maker(param1, param2):
    print("The parameters of my decorator are: {0} and {1}".format(param1, param2))

    def my_decorator(function_to_decorate):
        def wrapper(arg1, arg2):
            print("before call")
            result = function_to_decorate(arg1, arg2)
            print("after call")
            return result

        return wrapper

    return my_decorator

用法如下:

@decorator_maker("hello", "How are you?")
def my_function(arg1, arg2):
    print("The parameters of my function are: {0} and {1}".format(arg1, arg2))
    return arg1 + "-" + arg2

通过课程,它更直接:

class decorator_maker(object):
    def __init__(self, param1, param2):
        print("The parameters of my decorator are: {0} and {1}".format(param1, param2))
        self.param1 = param1
        self.param2 = param2

    def __call__(self, function_to_decorate):
        def wrapper(arg1, arg2):
            print("before call")
            result = function_to_decorate(arg1, arg2)
            print("after call")
            return result

        return wrapper

答案

要在函数调用之前/之后打开和关闭会话,必须在包装函数中实现它,如下所示:

class with_session(object):
    def __init__(self, url="bolt://localhost", user="neo4j", pwd="neo4j"):
        self.url = url
        self.user = user
        self.pwd = pwd

    def __call__(self, f):
        def wrapper(*args, **kwargs):
            driver = GraphDatabase.driver(self.url, auth=basic_auth(self.user, self.pwd))
            kwargs["session"] = session = driver.session()
            try:
                return f(*args, **kwargs)
            finally:
                session.close()

        return wrapper

注意:

  • 你可以让异常加注......
  • 您应该重命名装饰器(就像我一样),以避免遮蔽名称“session”。
  • 此装饰器不是上下文管理器。为此,您需要使用__enter____exit__,这是另一个用例......

参考

请参阅:How to make a chain of function decorators?