zoukankan      html  css  js  c++  java
  • 好代码系列(一):LazyObject

    site-packages/django/utils/functional.py
     1 def new_method_proxy(func):
     2     def inner(self, *args):
     3         if self_wrapped is empty:
     4             self._setup()
     5         return func(self._wrapped, *args)
     6     return inner
     7 
     8 class LazyObject(object):
     9     """
    10     A wrapper for another class that can be used to delay instantiation of the wrapped class.
    11 
    12     By subclassing, you have the opportunity to intercept and alter the
    13 instantiation. If you don't need to do that, use SimpleLazyObject.
    14     """
    15 
    16     # Avoid infinite recursion when tracing __init__ (#19456).
    17     _wrapped = None
    18 
    19     def __init__(self):
    20         self._wrapped = empty
    21 
    22     __getattr__ = new_method_proxy(getattr)
    23 
    24     def __setattr__(self, name, value):
    25         if name == "_wrapped":
    26             # Assign to __dict__ to avoid infinite __setattr__ loops.
    27             self.__dict__["_wrapped"] = value
    28         else:
    29             if self._wrapped is empty:
    30                 self._setup()
    31             setattr(self._wrapped, name, value)
    32 
    33     def __delattr__(self, name):
    34         if name == "_wrapped":
    35             raise TypeError("can't delete _wrapped.")
    36         if self._wrapped is empty:
    37             self._setup()
    38         delattr(self._wrapped, name)
    39 
    40     def _setup(self):
    41         """
    42         Must be implemented by subclasses to initialize the wrapped object.
    43         """
    44         raise NotImplementedError('subclasses of LazyObject must provide a _setup() method')
    45 
    46     # Because we have messed with __class__ below, we confuse pickle as to what
    47     # class we are pickling. We're going to have to initialize the wrapped
    48     # object to successfully pickle it, so we might as well just pickle the
    49     # wrapped object since they're supposed to act the same way.
    50     #
    51     # Unfortunately, if we try to simply act like the wrapped object, the ruse
    52     # will break down when pickle gets our id(). Thus we end up with pickle
    53     # thinking, in effect, that we are a distinct object from the wrapped
    54     # object, but with the same __dict__. This can cause problems (see #25389).
    55     #
    56     # So instead, we define our own __reduce__ method and custom unpickler. We
    57     # pickle the wrapped object as the unpickler's argument, so that pickle
    58     # will pickle it normally, and then the unpickler simply returns its
    59     # argument.
    60     def __reduce__(self):
    61         if self._wrapped is empty:
    62             self._setup()
    63         return (unpickle_lazyobject, (self._wrapped,))
    64 
    65     # We have to explicitly override __getstate__ so that older versions of
    66     # pickle don't try to pickle the __dict__ (which in the case of a
    67     # SimpleLazyObject may contain a lambda). The value will end up being
    68     # ignored by our __reduce__ and custom unpickler.
    69     def __getstate__(self):
    70         return {}
    71 
    72     def __deepcopy__(self, memo):
    73         if self._wrapped is empty:
    74             # We have to use type(self), not self.__class__, because the
    75             # latter is proxied.
    76             result = type(self)()
    77             memo[id(self)] = result
    78             return result
    79         return copy.deepcopy(self._wrapped, memo)
  • 相关阅读:
    网站搜索功能lucene
    RabbitMQ消息队列
    zookeeper
    RPC+SOA+dubbo
    石英定时任务-quartz
    通用mapper、图片上传、nginx
    通用mapper和分类实现
    后台商品管理功能实现
    构建框架
    海量数据的并发处理
  • 原文地址:https://www.cnblogs.com/olivetree123/p/5013853.html
Copyright © 2011-2022 走看看