翼度科技»论坛 编程开发 python 查看内容

Python面向对象-高级用法

7

主题

7

帖子

21

积分

新手上路

Rank: 1

积分
21
1.魔法函数

python中常见的内置类型

什么是魔法函数?
python的魔法函数总被双下划线包围,它们可以给你的类增加特殊的方法。如果你的对象实现了这些方法中的一个,那么这个方法就会在特殊情况下被调用,你可以定义想要
的行为,而这一切都是自动发生的。
魔法函数一览

魔法函数举例
1.1.__getitem__

把对象变成可迭代的对象
例子:
  1. class Company(object):
  2.     def __init__(self,employee_list):
  3.         self.employee = employee_list
  4.     #魔法函数,给类加可迭代类型
  5.     def __getitem__(self, item):
  6.         return self.employee[item]
  7. company = Company(['11','22','33'])
  8. #加了魔法函数“__getitem__”,类就成了可迭代的了
  9. for em in company:
  10.     print(em)    #11,22,33
复制代码
如果不用魔法函数循环出每个员工的方法
  1. class Company(object):
  2.     def __init__(self,employee_list):
  3.         self.employee = employee_list
  4. company = Company(['11','22','33'])
  5. for em in company.employee:
  6.     print(em)
复制代码
还可以切片和获取长度
  1. class Company(object):
  2.     def __init__(self,employee_list):
  3.         self.employee = employee_list
  4.     #魔法函数
  5.     def __getitem__(self, item):
  6.         return self.employee[item]
  7. company = Company(['11','22','33'])
  8. #可以切片
  9. company1 = company[:2]
  10. #可以判断len长度
  11. print(len(company1))   #2
  12. for em in company1:
  13.     print(em)    #11,22
复制代码
1.2.__len__
  1. class Company(object):
  2.     def __init__(self, employee_list):
  3.         self.employee = employee_list
  4.     #
  5.     # def __getitem__(self, item):
  6.     #     return self.employee[item]
  7.     def __len__(self):
  8.         return len(self.employee)
  9. company = Company(["11", "22", "33"])
  10. #如果不加魔法函数,len(company)会报错的
  11. print(len(company))     #3
复制代码
1.3. __repr__和__str__

1.安装交互环境(jupyter)
  1. pip install -i https://pypi.douban.com/simple ipython
  2. pip install -i https://pypi.douban.com/simple notebook
  3. #启动
  4. ipython notebook
复制代码
2.__repr__和__str__的用法


2.深入类和对象

1.1.鸭子类型和多态

“当看到一只鸟走起来像鸭子、游泳起来像鸭子、叫起来也像鸭子,那么这只鸟就可以被称为鸭子。”
我们并不关心对象是什么类型,到底是不是鸭子,只关心行为。
实例一:
  1. # 鸭子类型和多态简单实例
  2. class Dog(object):
  3.     def say(self):
  4.         print('a dog')
  5. class Cat(object):
  6.     def say(self):
  7.         print('a cat')
  8. class Duck(object):
  9.     def say(self):
  10.         print('a duck')
  11. animal_list = [Dog,Cat,Duck]
  12. for animal in animal_list:
  13.     animal().say()
  14. #运行结果
  15. a dog
  16. a cat
  17. a duck
复制代码
实例二:
类只要实现了__getitem__方法,它就是可迭代的,并不关心对象的本身,只关心行为,然后就可以当做extend的参数。
  1. class Company(object):
  2.     def __init__(self, employee_list):
  3.         self.employee = employee_list
  4.     def __getitem__(self, item):
  5.         return self.employee[item]
  6. company = Company(["11", "22", "33"])
  7. a = ['derek1','derek2']
  8. name_set = set()
  9. name_set.add('tom1')
  10. name_set.add(('tom2'))
  11. #extend里面的参数介绍
  12. #def extend(self, iterable):  # real signature unknown; restored from __doc__
  13.     #""" L.extend(iterable) -> None -- extend list by appending elements from the iterable """
  14. #extend里面可以添加任何可迭代的参数,给类添加一个魔法函数__getitem__,类就变成可迭代的,所以可以extend进去
  15. a.extend(company)
  16. print(a)       #['derek1', 'derek2', '11', '22', '33']
  17. a.extend(name_set)
  18. print(a)       #['derek1', 'derek2', '11', '22', '33', 'tom2', 'tom1']
复制代码
1.2.抽象基类(abc模块)

抽象基类的作用类似于JAVA中的接口。在接口中定义各种方法,然后继承接口的各种类进行具体方法的实现。抽象基类就是定义各种方法而不做具体实现的类,任何继承自抽象基类的类必须实现这些方法,否则无法实例化
(1)判断类时候有某种属性
  1. #判断类是否有某种属性
  2. class Company(object):
  3.     def __init__(self, employee_list):
  4.         self.employee = employee_list
  5.     def __len__(self):
  6.         return len(self.employee)
  7. com = Company(["11", "22", "33"])
  8. #hasattr判断类有没有某种属性,方法也是类的属性
  9. print(hasattr(com,"__len__"))   #True
  10. #虽然用hasattr可以判断,但正确的方式是定义一个抽象基类
  11. #我们在某些情况下希望判定某个对象的类型,可以用抽象基类
  12. from collections.abc import Sized
  13. print(isinstance(com,Sized))    #True
  14. # print(len(com))
复制代码
  1. class Sized(metaclass=ABCMeta):
  2.     __slots__ = ()
  3.     @abstractmethod
  4.     def __len__(self):
  5.         return 0
  6.     @classmethod
  7.     def __subclasshook__(cls, C):
  8.         if cls is Sized:
  9.             return _check_methods(C, "__len__")
  10.         return NotImplemented
复制代码
(2)abc模块

简单抽象基类实例
  1. #模拟一个抽象基类
  2. #写一个抽象基类,它的子类必须要重写抽象基类里面的方法
  3. import abc
  4. #定义一个抽象基类
  5. class CacheBase(metaclass=abc.ABCMeta):
  6.     @abc.abstractclassmethod
  7.     def get(self,key):
  8.         pass
  9.     @abc.abstractclassmethod
  10.     def set(self,key,value):
  11.         pass
  12. #子类,必须有抽象基类里面的方法,get和set
  13. #假入不写set方法会报错
  14. class RedisCache(CacheBase):
  15.     def get(self,key):
  16.         pass
  17.    
  18.     # def set(self,key,value):
  19.     #     pass
  20. redis_cache = RedisCache()
复制代码
  1. #模拟一个抽象基类
  2. #写一个抽象基类,它的子类必须要重写抽象基类里面的方法
  3. import abc
  4. #定义一个抽象基类
  5. class CacheBase(metaclass=abc.ABCMeta):
  6.     @abc.abstractclassmethod
  7.     def get(self,key):
  8.         pass
  9.     @abc.abstractclassmethod
  10.     def set(self,key,value):
  11.         pass
  12. #子类,必须有抽象基类里面的方法,get和set
  13. #假入不写set方法会报错
  14. class RedisCache(CacheBase):
  15.     def get(self,key):
  16.         pass
  17.     def set(self,key,value):
  18.         pass
  19. redis_cache = RedisCache()
复制代码

(3)abc里面所有的抽象基类
  1. from collections.abc import *
复制代码
所有的抽象基类
  1. # Copyright 2007 Google, Inc. All Rights Reserved.
  2. # Licensed to PSF under a Contributor Agreement.
  3. """Abstract Base Classes (ABCs) for collections, according to PEP 3119.
  4. Unit tests are in test_collections.
  5. """
  6. from abc import ABCMeta, abstractmethod
  7. import sys
  8. __all__ = ["Awaitable", "Coroutine",
  9.            "AsyncIterable", "AsyncIterator", "AsyncGenerator",
  10.            "Hashable", "Iterable", "Iterator", "Generator", "Reversible",
  11.            "Sized", "Container", "Callable", "Collection",
  12.            "Set", "MutableSet",
  13.            "Mapping", "MutableMapping",
  14.            "MappingView", "KeysView", "ItemsView", "ValuesView",
  15.            "Sequence", "MutableSequence",
  16.            "ByteString",
  17.            ]
  18. # This module has been renamed from collections.abc to _collections_abc to
  19. # speed up interpreter startup. Some of the types such as MutableMapping are
  20. # required early but collections module imports a lot of other modules.
  21. # See issue #19218
  22. __name__ = "collections.abc"
  23. # Private list of types that we want to register with the various ABCs
  24. # so that they will pass tests like:
  25. #       it = iter(somebytearray)
  26. #       assert isinstance(it, Iterable)
  27. # Note:  in other implementations, these types might not be distinct
  28. # and they may have their own implementation specific types that
  29. # are not included on this list.
  30. bytes_iterator = type(iter(b''))
  31. bytearray_iterator = type(iter(bytearray()))
  32. #callable_iterator = ???
  33. dict_keyiterator = type(iter({}.keys()))
  34. dict_valueiterator = type(iter({}.values()))
  35. dict_itemiterator = type(iter({}.items()))
  36. list_iterator = type(iter([]))
  37. list_reverseiterator = type(iter(reversed([])))
  38. range_iterator = type(iter(range(0)))
  39. longrange_iterator = type(iter(range(1 << 1000)))
  40. set_iterator = type(iter(set()))
  41. str_iterator = type(iter(""))
  42. tuple_iterator = type(iter(()))
  43. zip_iterator = type(iter(zip()))
  44. ## views ##
  45. dict_keys = type({}.keys())
  46. dict_values = type({}.values())
  47. dict_items = type({}.items())
  48. ## misc ##
  49. mappingproxy = type(type.__dict__)
  50. generator = type((lambda: (yield))())
  51. ## coroutine ##
  52. async def _coro(): pass
  53. _coro = _coro()
  54. coroutine = type(_coro)
  55. _coro.close()  # Prevent ResourceWarning
  56. del _coro
  57. ## asynchronous generator ##
  58. async def _ag(): yield
  59. _ag = _ag()
  60. async_generator = type(_ag)
  61. del _ag
  62. ### ONE-TRICK PONIES ###
  63. def _check_methods(C, *methods):
  64.     mro = C.__mro__
  65.     for method in methods:
  66.         for B in mro:
  67.             if method in B.__dict__:
  68.                 if B.__dict__[method] is None:
  69.                     return NotImplemented
  70.                 break
  71.         else:
  72.             return NotImplemented
  73.     return True
  74. class Hashable(metaclass=ABCMeta):
  75.     __slots__ = ()
  76.     @abstractmethod
  77.     def __hash__(self):
  78.         return 0
  79.     @classmethod
  80.     def __subclasshook__(cls, C):
  81.         if cls is Hashable:
  82.             return _check_methods(C, "__hash__")
  83.         return NotImplemented
  84. class Awaitable(metaclass=ABCMeta):
  85.     __slots__ = ()
  86.     @abstractmethod
  87.     def __await__(self):
  88.         yield
  89.     @classmethod
  90.     def __subclasshook__(cls, C):
  91.         if cls is Awaitable:
  92.             return _check_methods(C, "__await__")
  93.         return NotImplemented
  94. class Coroutine(Awaitable):
  95.     __slots__ = ()
  96.     @abstractmethod
  97.     def send(self, value):
  98.         """Send a value into the coroutine.
  99.         Return next yielded value or raise StopIteration.
  100.         """
  101.         raise StopIteration
  102.     @abstractmethod
  103.     def throw(self, typ, val=None, tb=None):
  104.         """Raise an exception in the coroutine.
  105.         Return next yielded value or raise StopIteration.
  106.         """
  107.         if val is None:
  108.             if tb is None:
  109.                 raise typ
  110.             val = typ()
  111.         if tb is not None:
  112.             val = val.with_traceback(tb)
  113.         raise val
  114.     def close(self):
  115.         """Raise GeneratorExit inside coroutine.
  116.         """
  117.         try:
  118.             self.throw(GeneratorExit)
  119.         except (GeneratorExit, StopIteration):
  120.             pass
  121.         else:
  122.             raise RuntimeError("coroutine ignored GeneratorExit")
  123.     @classmethod
  124.     def __subclasshook__(cls, C):
  125.         if cls is Coroutine:
  126.             return _check_methods(C, '__await__', 'send', 'throw', 'close')
  127.         return NotImplemented
  128. Coroutine.register(coroutine)
  129. class AsyncIterable(metaclass=ABCMeta):
  130.     __slots__ = ()
  131.     @abstractmethod
  132.     def __aiter__(self):
  133.         return AsyncIterator()
  134.     @classmethod
  135.     def __subclasshook__(cls, C):
  136.         if cls is AsyncIterable:
  137.             return _check_methods(C, "__aiter__")
  138.         return NotImplemented
  139. class AsyncIterator(AsyncIterable):
  140.     __slots__ = ()
  141.     @abstractmethod
  142.     async def __anext__(self):
  143.         """Return the next item or raise StopAsyncIteration when exhausted."""
  144.         raise StopAsyncIteration
  145.     def __aiter__(self):
  146.         return self
  147.     @classmethod
  148.     def __subclasshook__(cls, C):
  149.         if cls is AsyncIterator:
  150.             return _check_methods(C, "__anext__", "__aiter__")
  151.         return NotImplemented
  152. class AsyncGenerator(AsyncIterator):
  153.     __slots__ = ()
  154.     async def __anext__(self):
  155.         """Return the next item from the asynchronous generator.
  156.         When exhausted, raise StopAsyncIteration.
  157.         """
  158.         return await self.asend(None)
  159.     @abstractmethod
  160.     async def asend(self, value):
  161.         """Send a value into the asynchronous generator.
  162.         Return next yielded value or raise StopAsyncIteration.
  163.         """
  164.         raise StopAsyncIteration
  165.     @abstractmethod
  166.     async def athrow(self, typ, val=None, tb=None):
  167.         """Raise an exception in the asynchronous generator.
  168.         Return next yielded value or raise StopAsyncIteration.
  169.         """
  170.         if val is None:
  171.             if tb is None:
  172.                 raise typ
  173.             val = typ()
  174.         if tb is not None:
  175.             val = val.with_traceback(tb)
  176.         raise val
  177.     async def aclose(self):
  178.         """Raise GeneratorExit inside coroutine.
  179.         """
  180.         try:
  181.             await self.athrow(GeneratorExit)
  182.         except (GeneratorExit, StopAsyncIteration):
  183.             pass
  184.         else:
  185.             raise RuntimeError("asynchronous generator ignored GeneratorExit")
  186.     @classmethod
  187.     def __subclasshook__(cls, C):
  188.         if cls is AsyncGenerator:
  189.             return _check_methods(C, '__aiter__', '__anext__',
  190.                                   'asend', 'athrow', 'aclose')
  191.         return NotImplemented
  192. AsyncGenerator.register(async_generator)
  193. class Iterable(metaclass=ABCMeta):
  194.     __slots__ = ()
  195.     @abstractmethod
  196.     def __iter__(self):
  197.         while False:
  198.             yield None
  199.     @classmethod
  200.     def __subclasshook__(cls, C):
  201.         if cls is Iterable:
  202.             return _check_methods(C, "__iter__")
  203.         return NotImplemented
  204. class Iterator(Iterable):
  205.     __slots__ = ()
  206.     @abstractmethod
  207.     def __next__(self):
  208.         'Return the next item from the iterator. When exhausted, raise StopIteration'
  209.         raise StopIteration
  210.     def __iter__(self):
  211.         return self
  212.     @classmethod
  213.     def __subclasshook__(cls, C):
  214.         if cls is Iterator:
  215.             return _check_methods(C, '__iter__', '__next__')
  216.         return NotImplemented
  217. Iterator.register(bytes_iterator)
  218. Iterator.register(bytearray_iterator)
  219. #Iterator.register(callable_iterator)
  220. Iterator.register(dict_keyiterator)
  221. Iterator.register(dict_valueiterator)
  222. Iterator.register(dict_itemiterator)
  223. Iterator.register(list_iterator)
  224. Iterator.register(list_reverseiterator)
  225. Iterator.register(range_iterator)
  226. Iterator.register(longrange_iterator)
  227. Iterator.register(set_iterator)
  228. Iterator.register(str_iterator)
  229. Iterator.register(tuple_iterator)
  230. Iterator.register(zip_iterator)
  231. class Reversible(Iterable):
  232.     __slots__ = ()
  233.     @abstractmethod
  234.     def __reversed__(self):
  235.         while False:
  236.             yield None
  237.     @classmethod
  238.     def __subclasshook__(cls, C):
  239.         if cls is Reversible:
  240.             return _check_methods(C, "__reversed__", "__iter__")
  241.         return NotImplemented
  242. class Generator(Iterator):
  243.     __slots__ = ()
  244.     def __next__(self):
  245.         """Return the next item from the generator.
  246.         When exhausted, raise StopIteration.
  247.         """
  248.         return self.send(None)
  249.     @abstractmethod
  250.     def send(self, value):
  251.         """Send a value into the generator.
  252.         Return next yielded value or raise StopIteration.
  253.         """
  254.         raise StopIteration
  255.     @abstractmethod
  256.     def throw(self, typ, val=None, tb=None):
  257.         """Raise an exception in the generator.
  258.         Return next yielded value or raise StopIteration.
  259.         """
  260.         if val is None:
  261.             if tb is None:
  262.                 raise typ
  263.             val = typ()
  264.         if tb is not None:
  265.             val = val.with_traceback(tb)
  266.         raise val
  267.     def close(self):
  268.         """Raise GeneratorExit inside generator.
  269.         """
  270.         try:
  271.             self.throw(GeneratorExit)
  272.         except (GeneratorExit, StopIteration):
  273.             pass
  274.         else:
  275.             raise RuntimeError("generator ignored GeneratorExit")
  276.     @classmethod
  277.     def __subclasshook__(cls, C):
  278.         if cls is Generator:
  279.             return _check_methods(C, '__iter__', '__next__',
  280.                                   'send', 'throw', 'close')
  281.         return NotImplemented
  282. Generator.register(generator)
  283. class Sized(metaclass=ABCMeta):
  284.     __slots__ = ()
  285.     @abstractmethod
  286.     def __len__(self):
  287.         return 0
  288.     @classmethod
  289.     def __subclasshook__(cls, C):
  290.         if cls is Sized:
  291.             return _check_methods(C, "__len__")
  292.         return NotImplemented
  293. class Container(metaclass=ABCMeta):
  294.     __slots__ = ()
  295.     @abstractmethod
  296.     def __contains__(self, x):
  297.         return False
  298.     @classmethod
  299.     def __subclasshook__(cls, C):
  300.         if cls is Container:
  301.             return _check_methods(C, "__contains__")
  302.         return NotImplemented
  303. class Collection(Sized, Iterable, Container):
  304.     __slots__ = ()
  305.     @classmethod
  306.     def __subclasshook__(cls, C):
  307.         if cls is Collection:
  308.             return _check_methods(C,  "__len__", "__iter__", "__contains__")
  309.         return NotImplemented
  310. class Callable(metaclass=ABCMeta):
  311.     __slots__ = ()
  312.     @abstractmethod
  313.     def __call__(self, *args, **kwds):
  314.         return False
  315.     @classmethod
  316.     def __subclasshook__(cls, C):
  317.         if cls is Callable:
  318.             return _check_methods(C, "__call__")
  319.         return NotImplemented
  320. ### SETS ###
  321. class Set(Collection):
  322.     """A set is a finite, iterable container.
  323.     This class provides concrete generic implementations of all
  324.     methods except for __contains__, __iter__ and __len__.
  325.     To override the comparisons (presumably for speed, as the
  326.     semantics are fixed), redefine __le__ and __ge__,
  327.     then the other operations will automatically follow suit.
  328.     """
  329.     __slots__ = ()
  330.     def __le__(self, other):
  331.         if not isinstance(other, Set):
  332.             return NotImplemented
  333.         if len(self) > len(other):
  334.             return False
  335.         for elem in self:
  336.             if elem not in other:
  337.                 return False
  338.         return True
  339.     def __lt__(self, other):
  340.         if not isinstance(other, Set):
  341.             return NotImplemented
  342.         return len(self) < len(other) and self.__le__(other)
  343.     def __gt__(self, other):
  344.         if not isinstance(other, Set):
  345.             return NotImplemented
  346.         return len(self) > len(other) and self.__ge__(other)
  347.     def __ge__(self, other):
  348.         if not isinstance(other, Set):
  349.             return NotImplemented
  350.         if len(self) < len(other):
  351.             return False
  352.         for elem in other:
  353.             if elem not in self:
  354.                 return False
  355.         return True
  356.     def __eq__(self, other):
  357.         if not isinstance(other, Set):
  358.             return NotImplemented
  359.         return len(self) == len(other) and self.__le__(other)
  360.     @classmethod
  361.     def _from_iterable(cls, it):
  362.         '''Construct an instance of the class from any iterable input.
  363.         Must override this method if the class constructor signature
  364.         does not accept an iterable for an input.
  365.         '''
  366.         return cls(it)
  367.     def __and__(self, other):
  368.         if not isinstance(other, Iterable):
  369.             return NotImplemented
  370.         return self._from_iterable(value for value in other if value in self)
  371.     __rand__ = __and__
  372.     def isdisjoint(self, other):
  373.         'Return True if two sets have a null intersection.'
  374.         for value in other:
  375.             if value in self:
  376.                 return False
  377.         return True
  378.     def __or__(self, other):
  379.         if not isinstance(other, Iterable):
  380.             return NotImplemented
  381.         chain = (e for s in (self, other) for e in s)
  382.         return self._from_iterable(chain)
  383.     __ror__ = __or__
  384.     def __sub__(self, other):
  385.         if not isinstance(other, Set):
  386.             if not isinstance(other, Iterable):
  387.                 return NotImplemented
  388.             other = self._from_iterable(other)
  389.         return self._from_iterable(value for value in self
  390.                                    if value not in other)
  391.     def __rsub__(self, other):
  392.         if not isinstance(other, Set):
  393.             if not isinstance(other, Iterable):
  394.                 return NotImplemented
  395.             other = self._from_iterable(other)
  396.         return self._from_iterable(value for value in other
  397.                                    if value not in self)
  398.     def __xor__(self, other):
  399.         if not isinstance(other, Set):
  400.             if not isinstance(other, Iterable):
  401.                 return NotImplemented
  402.             other = self._from_iterable(other)
  403.         return (self - other) | (other - self)
  404.     __rxor__ = __xor__
  405.     def _hash(self):
  406.         """Compute the hash value of a set.
  407.         Note that we don't define __hash__: not all sets are hashable.
  408.         But if you define a hashable set type, its __hash__ should
  409.         call this function.
  410.         This must be compatible __eq__.
  411.         All sets ought to compare equal if they contain the same
  412.         elements, regardless of how they are implemented, and
  413.         regardless of the order of the elements; so there's not much
  414.         freedom for __eq__ or __hash__.  We match the algorithm used
  415.         by the built-in frozenset type.
  416.         """
  417.         MAX = sys.maxsize
  418.         MASK = 2 * MAX + 1
  419.         n = len(self)
  420.         h = 1927868237 * (n + 1)
  421.         h &= MASK
  422.         for x in self:
  423.             hx = hash(x)
  424.             h ^= (hx ^ (hx << 16) ^ 89869747)  * 3644798167
  425.             h &= MASK
  426.         h = h * 69069 + 907133923
  427.         h &= MASK
  428.         if h > MAX:
  429.             h -= MASK + 1
  430.         if h == -1:
  431.             h = 590923713
  432.         return h
  433. Set.register(frozenset)
  434. class MutableSet(Set):
  435.     """A mutable set is a finite, iterable container.
  436.     This class provides concrete generic implementations of all
  437.     methods except for __contains__, __iter__, __len__,
  438.     add(), and discard().
  439.     To override the comparisons (presumably for speed, as the
  440.     semantics are fixed), all you have to do is redefine __le__ and
  441.     then the other operations will automatically follow suit.
  442.     """
  443.     __slots__ = ()
  444.     @abstractmethod
  445.     def add(self, value):
  446.         """Add an element."""
  447.         raise NotImplementedError
  448.     @abstractmethod
  449.     def discard(self, value):
  450.         """Remove an element.  Do not raise an exception if absent."""
  451.         raise NotImplementedError
  452.     def remove(self, value):
  453.         """Remove an element. If not a member, raise a KeyError."""
  454.         if value not in self:
  455.             raise KeyError(value)
  456.         self.discard(value)
  457.     def pop(self):
  458.         """Return the popped value.  Raise KeyError if empty."""
  459.         it = iter(self)
  460.         try:
  461.             value = next(it)
  462.         except StopIteration:
  463.             raise KeyError
  464.         self.discard(value)
  465.         return value
  466.     def clear(self):
  467.         """This is slow (creates N new iterators!) but effective."""
  468.         try:
  469.             while True:
  470.                 self.pop()
  471.         except KeyError:
  472.             pass
  473.     def __ior__(self, it):
  474.         for value in it:
  475.             self.add(value)
  476.         return self
  477.     def __iand__(self, it):
  478.         for value in (self - it):
  479.             self.discard(value)
  480.         return self
  481.     def __ixor__(self, it):
  482.         if it is self:
  483.             self.clear()
  484.         else:
  485.             if not isinstance(it, Set):
  486.                 it = self._from_iterable(it)
  487.             for value in it:
  488.                 if value in self:
  489.                     self.discard(value)
  490.                 else:
  491.                     self.add(value)
  492.         return self
  493.     def __isub__(self, it):
  494.         if it is self:
  495.             self.clear()
  496.         else:
  497.             for value in it:
  498.                 self.discard(value)
  499.         return self
  500. MutableSet.register(set)
  501. ### MAPPINGS ###
  502. class Mapping(Collection):
  503.     __slots__ = ()
  504.     """A Mapping is a generic container for associating key/value
  505.     pairs.
  506.     This class provides concrete generic implementations of all
  507.     methods except for __getitem__, __iter__, and __len__.
  508.     """
  509.     @abstractmethod
  510.     def __getitem__(self, key):
  511.         raise KeyError
  512.     def get(self, key, default=None):
  513.         'D.get(k[,d]) -> D[k] if k in D, else d.  d defaults to None.'
  514.         try:
  515.             return self[key]
  516.         except KeyError:
  517.             return default
  518.     def __contains__(self, key):
  519.         try:
  520.             self[key]
  521.         except KeyError:
  522.             return False
  523.         else:
  524.             return True
  525.     def keys(self):
  526.         "D.keys() -> a set-like object providing a view on D's keys"
  527.         return KeysView(self)
  528.     def items(self):
  529.         "D.items() -> a set-like object providing a view on D's items"
  530.         return ItemsView(self)
  531.     def values(self):
  532.         "D.values() -> an object providing a view on D's values"
  533.         return ValuesView(self)
  534.     def __eq__(self, other):
  535.         if not isinstance(other, Mapping):
  536.             return NotImplemented
  537.         return dict(self.items()) == dict(other.items())
  538.     __reversed__ = None
  539. Mapping.register(mappingproxy)
  540. class MappingView(Sized):
  541.     __slots__ = '_mapping',
  542.     def __init__(self, mapping):
  543.         self._mapping = mapping
  544.     def __len__(self):
  545.         return len(self._mapping)
  546.     def __repr__(self):
  547.         return '{0.__class__.__name__}({0._mapping!r})'.format(self)
  548. class KeysView(MappingView, Set):
  549.     __slots__ = ()
  550.     @classmethod
  551.     def _from_iterable(self, it):
  552.         return set(it)
  553.     def __contains__(self, key):
  554.         return key in self._mapping
  555.     def __iter__(self):
  556.         yield from self._mapping
  557. KeysView.register(dict_keys)
  558. class ItemsView(MappingView, Set):
  559.     __slots__ = ()
  560.     @classmethod
  561.     def _from_iterable(self, it):
  562.         return set(it)
  563.     def __contains__(self, item):
  564.         key, value = item
  565.         try:
  566.             v = self._mapping[key]
  567.         except KeyError:
  568.             return False
  569.         else:
  570.             return v is value or v == value
  571.     def __iter__(self):
  572.         for key in self._mapping:
  573.             yield (key, self._mapping[key])
  574. ItemsView.register(dict_items)
  575. class ValuesView(MappingView):
  576.     __slots__ = ()
  577.     def __contains__(self, value):
  578.         for key in self._mapping:
  579.             v = self._mapping[key]
  580.             if v is value or v == value:
  581.                 return True
  582.         return False
  583.     def __iter__(self):
  584.         for key in self._mapping:
  585.             yield self._mapping[key]
  586. ValuesView.register(dict_values)
  587. class MutableMapping(Mapping):
  588.     __slots__ = ()
  589.     """A MutableMapping is a generic container for associating
  590.     key/value pairs.
  591.     This class provides concrete generic implementations of all
  592.     methods except for __getitem__, __setitem__, __delitem__,
  593.     __iter__, and __len__.
  594.     """
  595.     @abstractmethod
  596.     def __setitem__(self, key, value):
  597.         raise KeyError
  598.     @abstractmethod
  599.     def __delitem__(self, key):
  600.         raise KeyError
  601.     __marker = object()
  602.     def pop(self, key, default=__marker):
  603.         '''D.pop(k[,d]) -> v, remove specified key and return the corresponding value.
  604.           If key is not found, d is returned if given, otherwise KeyError is raised.
  605.         '''
  606.         try:
  607.             value = self[key]
  608.         except KeyError:
  609.             if default is self.__marker:
  610.                 raise
  611.             return default
  612.         else:
  613.             del self[key]
  614.             return value
  615.     def popitem(self):
  616.         '''D.popitem() -> (k, v), remove and return some (key, value) pair
  617.            as a 2-tuple; but raise KeyError if D is empty.
  618.         '''
  619.         try:
  620.             key = next(iter(self))
  621.         except StopIteration:
  622.             raise KeyError
  623.         value = self[key]
  624.         del self[key]
  625.         return key, value
  626.     def clear(self):
  627.         'D.clear() -> None.  Remove all items from D.'
  628.         try:
  629.             while True:
  630.                 self.popitem()
  631.         except KeyError:
  632.             pass
  633.     def update(*args, **kwds):
  634.         ''' D.update([E, ]**F) -> None.  Update D from mapping/iterable E and F.
  635.             If E present and has a .keys() method, does:     for k in E: D[k] = E[k]
  636.             If E present and lacks .keys() method, does:     for (k, v) in E: D[k] = v
  637.             In either case, this is followed by: for k, v in F.items(): D[k] = v
  638.         '''
  639.         if not args:
  640.             raise TypeError("descriptor 'update' of 'MutableMapping' object "
  641.                             "needs an argument")
  642.         self, *args = args
  643.         if len(args) > 1:
  644.             raise TypeError('update expected at most 1 arguments, got %d' %
  645.                             len(args))
  646.         if args:
  647.             other = args[0]
  648.             if isinstance(other, Mapping):
  649.                 for key in other:
  650.                     self[key] = other[key]
  651.             elif hasattr(other, "keys"):
  652.                 for key in other.keys():
  653.                     self[key] = other[key]
  654.             else:
  655.                 for key, value in other:
  656.                     self[key] = value
  657.         for key, value in kwds.items():
  658.             self[key] = value
  659.     def setdefault(self, key, default=None):
  660.         'D.setdefault(k[,d]) -> D.get(k,d), also set D[k]=d if k not in D'
  661.         try:
  662.             return self[key]
  663.         except KeyError:
  664.             self[key] = default
  665.         return default
  666. MutableMapping.register(dict)
  667. ### SEQUENCES ###
  668. class Sequence(Reversible, Collection):
  669.     """All the operations on a read-only sequence.
  670.     Concrete subclasses must override __new__ or __init__,
  671.     __getitem__, and __len__.
  672.     """
  673.     __slots__ = ()
  674.     @abstractmethod
  675.     def __getitem__(self, index):
  676.         raise IndexError
  677.     def __iter__(self):
  678.         i = 0
  679.         try:
  680.             while True:
  681.                 v = self[i]
  682.                 yield v
  683.                 i += 1
  684.         except IndexError:
  685.             return
  686.     def __contains__(self, value):
  687.         for v in self:
  688.             if v is value or v == value:
  689.                 return True
  690.         return False
  691.     def __reversed__(self):
  692.         for i in reversed(range(len(self))):
  693.             yield self[i]
  694.     def index(self, value, start=0, stop=None):
  695.         '''S.index(value, [start, [stop]]) -> integer -- return first index of value.
  696.            Raises ValueError if the value is not present.
  697.         '''
  698.         if start is not None and start < 0:
  699.             start = max(len(self) + start, 0)
  700.         if stop is not None and stop < 0:
  701.             stop += len(self)
  702.         i = start
  703.         while stop is None or i < stop:
  704.             try:
  705.                 v = self[i]
  706.                 if v is value or v == value:
  707.                     return i
  708.             except IndexError:
  709.                 break
  710.             i += 1
  711.         raise ValueError
  712.     def count(self, value):
  713.         'S.count(value) -> integer -- return number of occurrences of value'
  714.         return sum(1 for v in self if v is value or v == value)
  715. Sequence.register(tuple)
  716. Sequence.register(str)
  717. Sequence.register(range)
  718. Sequence.register(memoryview)
  719. class ByteString(Sequence):
  720.     """This unifies bytes and bytearray.
  721.     XXX Should add all their methods.
  722.     """
  723.     __slots__ = ()
  724. ByteString.register(bytes)
  725. ByteString.register(bytearray)
  726. class MutableSequence(Sequence):
  727.     __slots__ = ()
  728.     """All the operations on a read-write sequence.
  729.     Concrete subclasses must provide __new__ or __init__,
  730.     __getitem__, __setitem__, __delitem__, __len__, and insert().
  731.     """
  732.     @abstractmethod
  733.     def __setitem__(self, index, value):
  734.         raise IndexError
  735.     @abstractmethod
  736.     def __delitem__(self, index):
  737.         raise IndexError
  738.     @abstractmethod
  739.     def insert(self, index, value):
  740.         'S.insert(index, value) -- insert value before index'
  741.         raise IndexError
  742.     def append(self, value):
  743.         'S.append(value) -- append value to the end of the sequence'
  744.         self.insert(len(self), value)
  745.     def clear(self):
  746.         'S.clear() -> None -- remove all items from S'
  747.         try:
  748.             while True:
  749.                 self.pop()
  750.         except IndexError:
  751.             pass
  752.     def reverse(self):
  753.         'S.reverse() -- reverse *IN PLACE*'
  754.         n = len(self)
  755.         for i in range(n//2):
  756.             self[i], self[n-i-1] = self[n-i-1], self[i]
  757.     def extend(self, values):
  758.         'S.extend(iterable) -- extend sequence by appending elements from the iterable'
  759.         for v in values:
  760.             self.append(v)
  761.     def pop(self, index=-1):
  762.         '''S.pop([index]) -> item -- remove and return item at index (default last).
  763.            Raise IndexError if list is empty or index is out of range.
  764.         '''
  765.         v = self[index]
  766.         del self[index]
  767.         return v
  768.     def remove(self, value):
  769.         '''S.remove(value) -- remove first occurrence of value.
  770.            Raise ValueError if the value is not present.
  771.         '''
  772.         del self[self.index(value)]
  773.     def __iadd__(self, values):
  774.         self.extend(values)
  775.         return self
  776. MutableSequence.register(list)
  777. MutableSequence.register(bytearray)  # Multiply inheriting, see ByteString
复制代码

1.3.使用isinstance而不是type

(1)语法
  1. isinstance(object, classinfo)
复制代码
其中,object 是变量,classinfo 是类型即 (tuple,dict,int,float,list,bool等) 和 class类
若参数 object 是 classinfo 类的实例,或者 object 是 classinfo 类的子类的一个实例, 返回 True。
若 object 不是一个给定类型的的对象, 则返回结果总是False。
若 classinfo 不是一种数据类型或者由数据类型构成的元组,将引发一个 TypeError 异常。
(2)isinstance简单用法
  1. >>> isinstance(1,int)
  2. True
  3. >>>
  4. >>> isinstance('1',str)
  5. True
  6. >>>
  7. >>> isinstance(1,list)
  8. False
复制代码
(3)type()与isinstance()的区别:


  • 共同点两者都可以判断对象类型
  • 不同点对于一个 class 类的子类对象类型判断,type就不行了,而 isinstance 可以。
  1. class A:
  2.     pass
  3. class B(A):
  4.     pass
  5. b = B()
  6. #判断b是不是B的类型
  7. print(isinstance(b,B))        #True
  8. # b是不是A的类型呢,也是的
  9. #因为B继承A,isinstance内部会去检查继承链
  10. print(isinstance(b,A))        #True
  11. print(type(b) is B)           #True
  12. #b指向了B()对象,虽然A是B的父类,但是A是另外一个对象,它们的id是不相等的
  13. print(type(b) is A)           #False
复制代码
1.4.类变量和实例变量

python的类变量和实例变量,顾名思义,类变量是指跟类的变量,而实例变量,指跟类的具体实例相关联的变量
  1. class A:
  2.     #类变量
  3.     bb = 11
  4.     def __init__(self,x,y):
  5.         #实例变量
  6.         self.x = x
  7.         self.y = y
  8. a = A(2,3)
  9. A.bb = 111111
  10. print(a.x,a.y,a.bb)    # 2 3 111111
  11. print(A.bb)            # 111111
  12. a.bb = 2222     #实际上会在实例对象a里面新建一个属性bb
  13. print(a.bb)          # 2222
  14. print(A.bb)          # 111111
复制代码
1.5.类和实例属性的查找顺序

  1. class D:
  2.     pass
  3. class C(D):
  4.     pass
  5. class B(D):
  6.     pass
  7. class A(B,C):
  8.     pass
  9. #顺序:A,B,C,D
  10. #__mro__,类的属性查找顺序
  11. print(A.__mro__)      #(<class '__main__.A'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.D'>, <class 'object'>)
复制代码
  1. class D:
  2.     pass
  3. class E:
  4.     pass
  5. class C(E):
  6.     pass
  7. class B(D):
  8.     pass
  9. class A(B,C):
  10.     pass
  11. #顺序:A,B,D,C,E
  12. #__mro__,类的属性查找顺序
  13. print(A.__mro__)      
  14. #(<class '__main__.A'>, <class '__main__.B'>, <class '__main__.D'>, <class '__main__.C'>, <class '__main__.E'>, <class 'object'>)
复制代码
1.6.类方法,静态方法,和实例方法

实例:
  1. class Date():
  2.     #构造函数
  3.     def __init__(self,year,month,day):
  4.         self.year = year
  5.         self.month = month
  6.         self.day = day
  7.     #实例方法
  8.     def tomorrow(self):
  9.         self.day += 1
  10.     # 静态方法不用写self
  11.     @staticmethod
  12.     def parse_from_string(date_str):
  13.         year, month, day = tuple(date_str.split("-"))
  14.         # 静态方法不好的地方是采用硬编码,如果用类方法的话就不会了
  15.         return Date(int(year), int(month), int(day))
  16.     #类方法
  17.     @classmethod
  18.     def from_string(cls, date_str):
  19.         year, month, day = tuple(date_str.split("-"))
  20.         # cls:传进来的类,而不是像静态方法把类写死了
  21.         return cls(int(year), int(month), int(day))
  22.     def __str__(self):
  23.         return '%s/%s/%s'%(self.year,self.month,self.day)
  24. if __name__ == "__main__":
  25.     new_day = Date(2018,5,9)
  26.     #实例方法
  27.     new_day.tomorrow()
  28.     print(new_day)       #2018/5/10
  29.     #静态方法
  30.     date_str = '2018-05-09'
  31.     new_day = Date.parse_from_string(date_str)
  32.     print(new_day)       #2018/5/9
  33.     # 类方法
  34.     date_str = '2018-05-09'
  35.     new_day = Date.from_string(date_str)
  36.     print(new_day)  # 2018/5/9
复制代码
1.7.python对象的自省机制

在计算机编程中,自省是指一种能力:检查某些事物以确定它是什么、它知道什么以及它能做什么。自省向程序员提供了极大的灵活性和控制力。
  1. class Person:
  2.     '''人类'''
  3.     name = "user"
  4. class Student(Person):
  5.     def __init__(self,school_name):
  6.         self.school_name = school_name
  7. if __name__ == "__main__":
  8.     user = Student('仙剑')
  9.     #通过 __dict__ 查询有哪些属性
  10.     print(user.__dict__)        #{'school_name': '仙剑'}
  11.     print(Person.__dict__)      #{'__module__': '__main__', '__doc__': '人类', 'name': 'user', '__dict__': <attribute '__dict__' of 'Person' objects>, '__weakref__': <attribute '__weakref__' of 'Person' objects>}
  12.     print(Person.__doc__)       #人类
  13.     #可以添加属性
  14.     user.__dict__['school_addr'] = '北京'
  15.     print(user.school_addr)     #北京
  16.     #dir也可以查看属性,比__dict__功能更强大
  17.     print(dir(user))
  18. #['__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'name', 'school_addr', 'school_name']
复制代码
1.8.super函数

super执行的顺序
  1. class A:
  2.     def __init__(self):
  3.         print('A')
  4. class B(A):
  5.     def __init__(self):
  6.         print('B')
  7.         super().__init__()
  8. class C(A):
  9.     def __init__(self):
  10.         print('C')
  11.         super().__init__()
  12. class D(B,C):
  13.     def __init__(self):
  14.         print('D')
  15.         super(D, self).__init__()
  16. if __name__ == '__main__':
  17.     print(D.__mro__)          #(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>)
  18.     d = D()
  19.    
  20.    
  21. #执行结果
  22. D
  23. B
  24. C
  25. A
复制代码
1.9.with语句(上下文管理器)
  1. #上下文管理器
  2. class Sample:
  3.     def __enter__(self):
  4.         print('enter')
  5.         #获取资源
  6.         return self
  7.     def __exit__(self, exc_type, exc_val, exc_tb):
  8.         #释放资源
  9.         print('exit')
  10.     def do_something(self):
  11.         print('doing something')
  12. #会自动执行enter和exit方法
  13. with Sample() as sample:
  14.     sample.do_something()
  15. # 运行结果
  16. enter
  17. doing something
  18. exit
复制代码
3.python元类编程

1.1.propety动态属性

在面向对象编程中,我们一般把名词性的东西映射成属性,动词性的东西映射成方法。在python中他们对应的分别是属性self.xxx和类方法。但有时我们需要的属性需要根据其他属性动态的计算,此时如果直接使用属性方法处理,会导致数据不同步。下面介绍@property方法来动态创建类属性。
  1. from datetime import datetime,date
  2. class User:
  3.     def __init__(self,name,birthday):
  4.         self.name = name
  5.         self.birthday = birthday
  6.         self._age = 0
  7.     @property
  8.     def age(self):
  9.         return datetime.now().year - self.birthday.year
  10.     @age.setter
  11.     def age(self,value):
  12.         self._age = value
  13. if __name__ == '__main__':
  14.     user = User("derek",date(year=1994,month=11,day=11))
  15.     user.age = 23
  16.     print(user._age)   # 23
  17.     print(user.age)    # 24 ,动态计算出来的
复制代码
1.2.__getattr__和__getattribute__的区别

object.__getattr__(self, name)
找不到attribute的时候,会调用getattr,返回一个值或AttributeError异常。
object.__getattribute__(self, name)
无条件被调用,通过实例访问属性。如果class中定义了__getattr__(),则__getattr__()不会被调用(除非显示调用或引发AttributeError异常)
(1)调用一个不存在的属性
  1. class User:
  2.     def __init__(self,info={}):
  3.         self.info = info
  4.     # def __getattr__(self, item):
  5.     #     return self.info[item]
  6. if __name__ == '__main__':
  7.     user = User(info={"name":"derek","age":24})
  8.     print(user.name)
复制代码
会报错

(2)加了__getattr__之后就可以调用了
  1. class User:
  2.     def __init__(self,info={}):
  3.         self.info = info
  4.     #__getattr__是在查找不到属性的时候调用
  5.     def __getattr__(self, item):
  6.         return self.info[item]
  7. if __name__ == '__main__':
  8.     user = User(info={"name":"derek","age":24})
  9.     print(user.name)    #derek
复制代码
(3)__getattribute__
  1. class User:
  2.     def __init__(self,info={}):
  3.         self.info = info
  4.     #__getattr__是在查找不到属性的时候调用
  5.     def __getattr__(self, item):
  6.         return self.info[item]
  7.     #__getattribute不管属性存不存在,都访问这个
  8.     def __getattribute__(self, item):
  9.         return "zhang_derek"
  10. if __name__ == '__main__':
  11.     user = User(info={"name":"derek","age":24})
  12.     #不管属性存不存在,都走__getattribute__
  13.     print(user.name)    #zhang_derek     #即使属性存在也走__getattribute__
  14.     print(user.test)     #zhang_derek    #不存在的属性也能打印
  15.     print(user.company)   #zhang_derek   #不存在的属性也能打印
复制代码
1.3.属性描述符

验证赋值的时候是不是int类型
  1. #属性描述符
  2. import numbers
  3. #只要一个类实现了下面三种魔法函数中的一种,这个类就是属性描述符
  4. class IntField:
  5.     def __get__(self, instance, owner):
  6.         return self.value
  7.     def __set__(self, instance, value):
  8.         if not isinstance(value,numbers.Integral):
  9.             raise ValueError("必须为int")
  10.         self.value = value
  11.     def __delete__(self, instance):
  12.         pass
  13. class User:
  14.     age = IntField()
  15. if __name__ == '__main__':
  16.     user = User()
  17.     user.age = 24
  18.     print(user.age)
复制代码
如果user.age=24,值是int,可以正常打印
如果user.age='test',传一个字符串,则会报错

1.4.__new__和__init__的区别

(1)__new__方法如果不返回对象,不会执行init方法
  1. class User:
  2.     def __new__(cls, *args, **kwargs):
  3.         print("in new")
  4.     def __init__(self,name):
  5.         print("in init")
  6.         self.name = name
  7. # new是用用来控制对象的生成过程,在对象生成之前
  8. # init是用来完善对象的
  9. # 如果new方法不返回对象,则不会调用init函数
  10. if __name__ == '__main__':
  11.     user = User("derek")
复制代码
运行结果:没有调用init方法

(2)返回对象就会执行init方法
  1. class User:
  2.     def __new__(cls, *args, **kwargs):
  3.         print("in new")         #in new
  4.         print(cls)              #cls是当前class对象    <class '__main__.User'>
  5.         print(type(cls))        #<class 'type'>
  6.         return super().__new__(cls)   #必须返回class对象,才会调用__init__方法
  7.     def __init__(self,name):
  8.         print("in init")        #in init
  9.         print(self)             #self是class的实例对象      <__main__.User object at 0x00000000021B8780>
  10.         print(type(self))       #<class '__main__.User'>
  11.         self.name = name
  12. # new是用用来控制对象的生成过程,在对象生成之前
  13. # init是用来完善对象的
  14. # 如果new方法不返回对象,则不会调用init函数
  15. if __name__ == '__main__':
  16.     user = User(name="derek")
  17. #总结
  18. # __new__ 用来创建实例,在返回的实例上执行__init__,如果不返回实例那么__init__将不会执行
  19. # __init__ 用来初始化实例,设置属性什么的
复制代码
1.5.自定义元类

(1)前戏:通过传入不同的字符串动态的创建不同的类
  1. def create_class(name):
  2.     if name == 'user':
  3.         class User:
  4.             def __str__(self):
  5.                 return "user"
  6.         return User
  7.     elif name == "company":
  8.         class Company:
  9.             def __str__(self):
  10.                 return "company"
  11.         return Company
  12. if __name__ == '__main__':
  13.     Myclass = create_class("user")
  14.     my_obj = Myclass()
  15.     print(my_obj)    #user
  16.     print(type(my_obj))     #<class '__main__.create_class.<locals>.User'>
复制代码
(2)用type创建
虽然上面的方法能够创建,但很麻烦,下面是type创建类的一个简单实例
  1. # 一个简单type创建类的例子
  2. #type(object_or_name, bases, dict)
  3. #type里面有三个参数,第一个类名,第二个基类名,第三个是属性
  4. User = type("User",(),{"name":"derek"})
  5. my_obj = User()
  6. print(my_obj.name)    #derek
复制代码
(3)不但可以定义属性,还可以定义方法
  1. def say(self):     #必须加self
  2.     return "i am derek"
  3. User = type("User",(),{"name":"derek","say":say})
  4. my_obj = User()
  5. print(my_obj.name)     #derek
  6. print(my_obj.say())    #i am derek
复制代码
(4)让type创建的类继承一个基类
  1. def say(self):     #必须加self
  2.     return "i am derek"
  3. class BaseClass:
  4.     def answer(self):
  5.         return "i am baseclass"
  6. #type里面有三个参数,第一个类名,第二个基类名,第三个是属性
  7. User = type("User",(BaseClass,),{"name":"derek","say":say})
  8. if __name__ == '__main__':
  9.     my_obj = User()
  10.     print(my_obj.name)          # derek
  11.     print(my_obj.say())         # i am derek
  12.     print(my_obj.answer())      # i am baseclass
复制代码
1.6.什么是元类?

元类就是创建类的类,比如上面的type
在实际编码中,我们一般不直接用type去创建类,而是用元类的写法,自定义一个元类metaclass去创建
  1. # 把User类创建的过程委托给元类去做,这样代码的分离性比较好
  2. class MetaClass(type):
  3.     def __new__(cls, *args, **kwargs):
  4.         return super().__new__(cls,*args, **kwargs)
  5. class User(metaclass=MetaClass):
  6.     def __init__(self,name):
  7.         self.name = name
  8.     def __str__(self):
  9.         return "test"
  10. if __name__ == '__main__':
  11.     #python中类的实例化过程,会首先寻找metaclass,通过metaclass去创建User类
  12.     my_obj = User(name="derek")
  13.     print(my_obj)    #test
复制代码
4.自定义序列类

1.1.序列类型的分类


1.2.序列的+和+=,extend和append的区别
  1. from collections import abc
  2. a = [1,2,]
  3. c = a + [3,4]
  4. print(c)         #[1, 2, 3, 4]
  5. #如果 + 元祖则会报错, not tuple
  6. # c = a + (3,4)     #TypeError: can only concatenate list (not "tuple") to list
  7. # + 是新生产一个list, += 是就地加,不会新生成list
  8. #用+= 则可以是元祖,后面只要是可迭代的就行
  9. #其原理是python内部抽象基类MutableSequence里面有个魔法函数__iadd__来实现的
  10. a += (3,4)       #[1, 2, 3, 4]
  11. print(a)
  12. a.extend((5,6))
  13. print(a)        #[1, 2, 3, 4, 5, 6]
  14. a.append((7,8))
  15. print(a)        #[1, 2, 3, 4, 5, 6, (7, 8)]
  16. #可以看到extend和append结果并不一样,append是把里面当一个值传进去,extend是迭代的传进去
复制代码
1.3.实现可切片的对象

(1)切片的用法
  1. #模式[start:end:step]
  2. """
  3.     其中,第一个数字start表示切片开始位置,默认为0;
  4.     第二个数字end表示切片截止(但不包含)位置(默认为列表长度);
  5.     第三个数字step表示切片的步长(默认为1)。
  6.     当start为0时可以省略,当end为列表长度时可以省略,
  7.     当step为1时可以省略,并且省略步长时可以同时省略最后一个冒号。
  8.     另外,当step为负整数时,表示反向切片,这时start应该比end的值要大才行。
  9. """
  10. aList = [3, 4, 5, 6, 7, 9, 11, 13, 15, 17]
  11. #切片返回的是一个新元素,不会改变原有的list
  12. print (aList[::])  # 返回包含原列表中所有元素的新列表            [3, 4, 5, 6, 7, 9, 11, 13, 15, 17]
  13. print (aList[::-1])  # 返回包含原列表中所有元素的逆序列表        [17, 15, 13, 11, 9, 7, 6, 5, 4, 3]
  14. print (aList[::2])  # 隔一个取一个,获取偶数位置的元素           [3, 5, 7, 11, 15]
  15. print (aList[1::2])  # 隔一个取一个,获取奇数位置的元素          [4, 6, 9, 13, 17]
  16. print (aList[3:6])  # 指定切片的开始和结束位置                   [6, 7, 9]
  17. print(aList[0:100])  # 切片结束位置大于列表长度时,从列表尾部截断      [3, 4, 5, 6, 7, 9, 11, 13, 15, 17]
  18. print(aList[100:])  # 切片开始位置大于列表长度时,返回空列表           []
  19. # aList[len(aList):] = [9]  # 在列表尾部增加元素
  20. # aList[:0] = [1, 2]  # 在列表头部插入元素
  21. # aList[3:3] = [4]  # 在列表中间位置插入元素
  22. # aList[:3] = [1, 2]  # 替换列表元素,等号两边的列表长度相等
  23. # aList[3:] = [4, 5, 6]  # 等号两边的列表长度也可以不相等
  24. # aList[::2] = [0] * 3  # 隔一个修改一个
  25. # print (aList)
  26. # aList[::2] = ['a', 'b', 'c']  # 隔一个修改一个
  27. # aList[::2] = [1,2]  # 左侧切片不连续,等号两边列表长度必须相等  #会报错
  28. # aList[:3] = []  # 删除列表中前3个元素
  29. #删除
  30. # del aList[:3]  # 切片元素连续
  31. # del aList[::2]  # 切片元素不连续,隔一个删一个
复制代码
(2)实现对象支持切片操作
  1. from collections import abc
  2. #Sequence协议
  3. import numbers
  4. class Group:
  5.     #支持切片操作
  6.     def __init__(self, group_name, company_name, staffs):
  7.         self.group_name = group_name
  8.         self.company_name = company_name
  9.         self.staffs = staffs
  10.     def __reversed__(self):
  11.         self.staffs.reverse()
  12.     def __getitem__(self, item):
  13.         #当前的类
  14.         cls = type(self)
  15.         #判断类是不是可切片的对象
  16.         if isinstance(item, slice):
  17.             return cls(group_name=self.group_name, company_name=self.company_name, staffs=self.staffs[item])
  18.         elif isinstance(item, numbers.Integral):
  19.             return cls(group_name=self.group_name, company_name=self.company_name, staffs=[self.staffs[item]])
  20.     def __len__(self):
  21.         return len(self.staffs)
  22.     def __iter__(self):
  23.         return iter(self.staffs)
  24.     def __contains__(self, item):
  25.         if item in self.staffs:
  26.             return True
  27.         else:
  28.             return False
  29. staffs = ["derek1", "derek2", "derek3", "derek4"]
  30. group = Group(company_name="alibaba", group_name="user", staffs=staffs)
  31. #现在对象就成可切片的对象了
  32. #__getitem__
  33. for user in group:
  34.     print(user)
  35. #运行结果
  36. # derek1
  37. # derek2
  38. # derek3
  39. # derek4
  40. #__contains__
  41. if 'derek1' in group:
  42.     print('yes')
复制代码
1.4.列表生成式,字典推导式
  1. # odd_list = []
  2. # for i in range(21):
  3. #     if i%2 == 1:
  4. #         odd_list.append(i)
  5. # print(odd_list)      #[1, 3, 5, 7, 9, 11, 13, 15, 17, 19]
  6. #列表生成式
  7. #1.取出1-20之间的基数
  8. odd_list = [i for i in range(21) if i %2 == 1]
  9. print(odd_list)     #[1, 3, 5, 7, 9, 11, 13, 15, 17, 19]
  10. #2.取出1-20之间的基数的平方
  11. def hadle_item(item):
  12.     return item * item
  13. odd_list = [hadle_item(i) for i in range(21) if i %2 == 1]
  14. print(odd_list)     #[1, 9, 25, 49, 81, 121, 169, 225, 289, 361]
复制代码
利用字典推导式把字典的key和value做转换:{key:value}变成{value:key}的形式
  1. # 字典推导式的用法
  2. my_dict = {'derek1':11,'derek2':22,'derek3':33}
  3. reversed_dict = {value:key for key,value in my_dict.items()}
  4. print(reversed_dict)    #{11: 'derek1', 22: 'derek2', 33: 'derek3'}
复制代码
https://www.cnblogs.com/derek1184405959/p/8579428.html

来源:https://www.cnblogs.com/geoffreygao/p/17271010.html
免责声明:由于采集信息均来自互联网,如果侵犯了您的权益,请联系我们【E-Mail:cb@itdo.tech】 我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x

举报 回复 使用道具