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)
  • 相关阅读:
    SQL Server中的执行引擎入门
    SQL Server复制入门(一)复制简介
    Django 代码片断收集(持续更新)
    今天思路有点乱,随便记一点关于 xmlrpc 的
    PIL 学习笔记(1)
    Django newforms
    在 Django 的 View 中利用 function decorator 可实现一定程度的代码重用
    今天在 Linux 上顺利编译 PIL 1.1.6 成功
    用 PIL 写了个简单的缩略图生成程序
    [转贴] 中药内外合治急慢性鼻窦炎
  • 原文地址:https://www.cnblogs.com/olivetree123/p/5013853.html
Copyright © 2011-2022 走看看