Python_面向对象进阶
isinstance和issubclass
isinstance(obj,cls)检查是否obj是否是类 cls 的对象
- class Foo(object):
- pass
- obj = Foo()
- isinstance(obj, Foo)
issubclass(sub, super)检查sub类是否是 super 类的派生类
- class Foo(object):
- pass
- class Bar(Foo):
- pass
- issubclass(Bar, Foo)
反射
1 什么是反射
反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问、检测和修改它本身状态或行为的一种能力(自省)。这一概念的提出很快引发了计算机科学领域关于应用反射性的研究。它首先被程序语言的设计领域所采用,并在Lisp和面向对象方面取得了成绩。
2 python面向对象中的反射:通过字符串的形式操作对象相关的属性。python中的一切事物都是对象(都可以使用反射)
四个可以实现自省的函数
下列方法适用于类和对象(一切皆对象,类本身也是一个对象)
- def hasattr(*args, **kwargs): # real signature unknown
- """
- Return whether the object has an attribute with the given name.
- This is done by calling getattr(obj, name) and catching AttributeError.
- """
- pass
hasattr
- def getattr(object, name, default=None): # known special case of getattr
- """
- getattr(object, name[, default]) -> value
- Get a named attribute from an object; getattr(x, 'y') is equivalent to x.y.
- When a default argument is given, it is returned when the attribute doesn't
- exist; without it, an exception is raised in that case.
- """
- pass
getattr
- def setattr(x, y, v): # real signature unknown; restored from __doc__
- """
- Sets the named attribute on the given object to the specified value.
- setattr(x, 'y', v) is equivalent to ``x.y = v''
- """
- pass
setattr
- def delattr(x, y): # real signature unknown; restored from __doc__
- """
- Deletes the named attribute from the given object.
- delattr(x, 'y') is equivalent to ``del x.y''
- """
- pass
delattr
- class Foo:
- f = '类的静态变量'
- def __init__(self,name,age):
- self.name=name
- self.age=age
- def say_hi(self):
- print('hi,%s'%self.name)
- obj=Foo('egon',73)
- #检测是否含有某属性
- print(hasattr(obj,'name'))
- print(hasattr(obj,'say_hi'))
- #获取属性
- n=getattr(obj,'name')
- print(n)
- func=getattr(obj,'say_hi')
- func()
- print(getattr(obj,'aaaaaaaa','不存在啊')) #报错
- #设置属性
- setattr(obj,'sb',True)
- setattr(obj,'show_name',lambda self:self.name+'sb')
- print(obj.__dict__)
- print(obj.show_name(obj))
- #删除属性
- delattr(obj,'age')
- delattr(obj,'show_name')
- delattr(obj,'show_name111')#不存在,则报错
- print(obj.__dict__)
四个方法的使用演示
- class Foo(object):
- staticField = "old boy"
- def __init__(self):
- self.name = 'wupeiqi'
- def func(self):
- return 'func'
- @staticmethod
- def bar():
- return 'bar'
- print getattr(Foo, 'staticField')
- print getattr(Foo, 'func')
- print getattr(Foo, 'bar')
类也是对象
- #!/usr/bin/env python
- # -*- coding:utf-8 -*-
- import sys
- def s1():
- print 's1'
- def s2():
- print 's2'
- this_module = sys.modules[__name__]
- hasattr(this_module, 's1')
- getattr(this_module, 's2')
反射当前模块成员
导入其他模块,利用反射查找该模块是否存在某个方法
- #!/usr/bin/env python
- # -*- coding:utf-8 -*-
- def test():
- print('from the test')
- #!/usr/bin/env python
- # -*- coding:utf-8 -*-
- """
- 程序目录:
- module_test.py
- index.py
- 当前文件:
- index.py
- """
- import module_test as obj
- #obj.test()
- print(hasattr(obj,'test'))
- getattr(obj,'test')()
__str__和__repr__
改变对象的字符串显示__str__,__repr__
自定制格式化字符串__format__
- #_*_coding:utf-8_*_
- format_dict={
- 'nat':'{obj.name}-{obj.addr}-{obj.type}',#学校名-学校地址-学校类型
- 'tna':'{obj.type}:{obj.name}:{obj.addr}',#学校类型:学校名:学校地址
- 'tan':'{obj.type}/{obj.addr}/{obj.name}',#学校类型/学校地址/学校名
- }
- class School:
- def __init__(self,name,addr,type):
- self.name=name
- self.addr=addr
- self.type=type
- def __repr__(self):
- return 'School(%s,%s)' %(self.name,self.addr)
- def __str__(self):
- return '(%s,%s)' %(self.name,self.addr)
- def __format__(self, format_spec):
- # if format_spec
- if not format_spec or format_spec not in format_dict:
- format_spec='nat'
- fmt=format_dict[format_spec]
- return fmt.format(obj=self)
- s1=School('oldboy1','北京','私立')
- print('from repr: ',repr(s1))
- print('from str: ',str(s1))
- print(s1)
- '''
- str函数或者print函数--->obj.__str__()
- repr或者交互式解释器--->obj.__repr__()
- 如果__str__没有被定义,那么就会使用__repr__来代替输出
- 注意:这俩方法的返回值必须是字符串,否则抛出异常
- '''
- print(format(s1,'nat'))
- print(format(s1,'tna'))
- print(format(s1,'tan'))
- print(format(s1,'asfdasdffd'))
- class B:
- def __str__(self):
- return 'str : class B'
- def __repr__(self):
- return 'repr : class B'
- b=B()
- print('%s'%b)
- print('%r'%b)
%s和%r
item系列
__getitem__\__setitem__\__delitem__
- class Foo:
- def __init__(self,name):
- self.name=name
- def __getitem__(self, item):
- print(self.__dict__[item])
- def __setitem__(self, key, value):
- self.__dict__[key]=value
- def __delitem__(self, key):
- print('del obj[key]时,我执行')
- self.__dict__.pop(key)
- def __delattr__(self, item):
- print('del obj.key时,我执行')
- self.__dict__.pop(item)
- f1=Foo('sb')
- f1['age']=18
- f1['age1']=19
- del f1.age1
- del f1['age']
- f1['name']='alex'
- print(f1.__dict__)
__del__
析构方法,当对象在内存中被释放时,自动触发执行。
注:此方法一般无须定义,因为Python是一门高级语言,程序员在使用时无需关心内存的分配和释放,因为此工作都是交给Python解释器来执行,所以,析构函数的调用是由解释器在进行垃圾回收时自动触发执行的。
- class Foo:
- def __del__(self):
- print('执行我啦')
- f1=Foo()
- del f1
- print('------->')
- #输出结果
- 执行我啦
- ------->
简单示范
__new__
- class A:
- def __init__(self):
- self.x = 1
- print('in init function')
- def __new__(cls, *args, **kwargs):
- print('in new function')
- return object.__new__(A)
- a = A()
- print(a.x)
- class Singleton:
- def __new__(cls, *args, **kw):
- if not hasattr(cls, '_instance'):
- cls._instance = object.__new__(cls)
- return cls._instance
- one = Singleton()
- two = Singleton()
- two.a = 3
- print(one.a)
- #
- # one和two完全相同,可以用id(), ==, is检测
- print(id(one))
- #
- print(id(two))
- #
- print(one == two)
- # True
- print(one is two)
单例模式
__call__
对象后面加括号,触发执行。
注:构造方法的执行是由创建对象触发的,即:对象 = 类名() ;而对于 __call__ 方法的执行是由对象后加括号触发的,即:对象() 或者 类()()
- class Foo:
- def __init__(self):
- pass
- def __call__(self, *args, **kwargs):
- print('__call__')
- obj = Foo() # 执行 __init__
- obj() # 执行 __call__
with和__enter__,__exit__
- class A:
- def __enter__(self):
- print('before')
- def __exit__(self, exc_type, exc_val, exc_tb):
- print('after')
- with A() as a:
- print('')
with语句
- class A:
- def __init__(self):
- print('init')
- def __enter__(self):
- print('before')
- def __exit__(self, exc_type, exc_val, exc_tb):
- print('after')
- with A() as a:
- print('')
with语句和init
- class Myfile:
- def __init__(self,path,mode='r',encoding = 'utf-8'):
- self.path = path
- self.mode = mode
- self.encoding = encoding
- def __enter__(self):
- self.f = open(self.path, mode=self.mode, encoding=self.encoding)
- return self.f
- def __exit__(self, exc_type, exc_val, exc_tb):
- self.f.close()
- with Myfile('file',mode='w') as f:
- f.write('wahaha')
with和文件操作
- import pickle
- class MyPickledump:
- def __init__(self,path):
- self.path = path
- def __enter__(self):
- self.f = open(self.path, mode='ab')
- return self
- def dump(self,content):
- pickle.dump(content,self.f)
- def __exit__(self, exc_type, exc_val, exc_tb):
- self.f.close()
- class Mypickleload:
- def __init__(self,path):
- self.path = path
- def __enter__(self):
- self.f = open(self.path, mode='rb')
- return self
- def __exit__(self, exc_type, exc_val, exc_tb):
- self.f.close()
- def load(self):
- return pickle.load(self.f)
- def loaditer(self):
- while True:
- try:
- yield self.load()
- except EOFError:
- break
- # with MyPickledump('file') as f:
- # f.dump({1,2,3,4})
- with Mypickleload('file') as f:
- for item in f.loaditer():
- print(item)
with和pickle
- import pickle
- class MyPickledump:
- def __init__(self,path):
- self.path = path
- def __enter__(self):
- self.f = open(self.path, mode='ab')
- return self
- def dump(self,content):
- pickle.dump(content,self.f)
- def __exit__(self, exc_type, exc_val, exc_tb):
- self.f.close()
- class Mypickleload:
- def __init__(self,path):
- self.path = path
- def __enter__(self):
- self.f = open(self.path, mode='rb')
- return self
- def __exit__(self, exc_type, exc_val, exc_tb):
- self.f.close()
- def __iter__(self):
- while True:
- try:
- yield pickle.load(self.f)
- except EOFError:
- break
- # with MyPickledump('file') as f:
- # f.dump({1,2,3,4})
- with Mypickleload('file') as f:
- for item in f:
- print(item)
with和pickle和iter
__len__
- class A:
- def __init__(self):
- self.a = 1
- self.b = 2
- def __len__(self):
- return len(self.__dict__)
- a = A()
- print(len(a))
__hash__
- class A:
- def __init__(self):
- self.a = 1
- self.b = 2
- def __hash__(self):
- return hash(str(self.a)+str(self.b))
- a = A()
- print(hash(a))
__eq__
- class A:
- def __init__(self):
- self.a = 1
- self.b = 2
- def __eq__(self,obj):
- if self.a == obj.a and self.b == obj.b:
- return True
- a = A()
- b = A()
- print(a == b)
- class FranchDeck:
- ranks = [str(n) for n in range(2,11)] + list('JQKA')
- suits = ['红心','方板','梅花','黑桃']
- def __init__(self):
- self._cards = [Card(rank,suit) for rank in FranchDeck.ranks
- for suit in FranchDeck.suits]
- def __len__(self):
- return len(self._cards)
- def __getitem__(self, item):
- return self._cards[item]
- deck = FranchDeck()
- print(deck[0])
- from random import choice
- print(choice(deck))
- print(choice(deck))
纸牌游戏1
- class FranchDeck:
- ranks = [str(n) for n in range(2,11)] + list('JQKA')
- suits = ['红心','方板','梅花','黑桃']
- def __init__(self):
- self._cards = [Card(rank,suit) for rank in FranchDeck.ranks
- for suit in FranchDeck.suits]
- def __len__(self):
- return len(self._cards)
- def __getitem__(self, item):
- return self._cards[item]
- def __setitem__(self, key, value):
- self._cards[key] = value
- deck = FranchDeck()
- print(deck[0])
- from random import choice
- print(choice(deck))
- print(choice(deck))
- from random import shuffle
- shuffle(deck)
- print(deck[:5])
纸牌游戏2
- class Person:
- def __init__(self,name,age,sex):
- self.name = name
- self.age = age
- self.sex = sex
- def __hash__(self):
- return hash(self.name+self.sex)
- def __eq__(self, other):
- if self.name == other.name and self.sex == other.sex:return True
- p_lst = []
- for i in range(84):
- p_lst.append(Person('egon',i,'male'))
- print(p_lst)
- print(set(p_lst))
一道面试题
Python_面向对象进阶的更多相关文章
- python_面向对象进阶(7)
第1章 面向对象特性—继承(补充) 1.1 接口类.抽象类介绍 1.2 接口类 1.3 接口类应用过程 1.3.1 第一版:完成多种支付方式接口 1.3.2 第二版: 归一化设计,统一支付方式 1.3 ...
- Python全栈开发【面向对象进阶】
Python全栈开发[面向对象进阶] 本节内容: isinstance(obj,cls)和issubclass(sub,super) 反射 __setattr__,__delattr__,__geta ...
- day26、面向对象进阶:多态、封装、反射
一.多态 什么是多态: 类的继承有两层意义:1.改变 2.扩展 多态就是类的这两层意义的一个具体的实现机. 即:调用不同类实例化的对象,下的相同的方法,实现的过程不一样 python中的标准类型就是多 ...
- Python开发【第七篇】:面向对象 和 python面向对象进阶篇(下)
Python开发[第七篇]:面向对象 详见:<Python之路[第五篇]:面向对象及相关> python 面向对象(进阶篇) 上一篇<Python 面向对象(初级篇)> ...
- python基础——面向对象进阶下
python基础--面向对象进阶下 1 __setitem__,__getitem,__delitem__ 把对象操作属性模拟成字典的格式 想对比__getattr__(), __setattr__( ...
- python基础——面向对象进阶
python基础--面向对象进阶 1.isinstance(obj,cls)和issubclass(sub,super) isinstance(obj,cls)检查是否obj是否是类 cls 的对象 ...
- 周末班:Python基础之面向对象进阶
面向对象进阶 类型判断 issubclass 首先,我们先看issubclass() 这个内置函数可以帮我们判断x类是否是y类型的子类. class Base: pass class Foo(Base ...
- Python面向对象进阶和socket网络编程-day08
写在前面 上课第八天,打卡: 为什么坚持?想一想当初: 一.面向对象进阶 - 1.反射补充 - 通过字符串去操作一个对象的属性,称之为反射: - 示例1: class Chinese: def __i ...
- 铁乐学python_day24_面向对象进阶1_内置方法
铁乐学python_day24_面向对象进阶1_内置方法 题外话1: 学习方法[wwwh] what where why how 是什么,用在哪里,为什么,怎么用 学习到一个新知识点的时候,多问问上面 ...
随机推荐
- CSS的一些文本操作有哪些
一.文本的对齐方式 text-align:center/left/right text-align:justify; font-style:normal/italic/oblique/inherit ...
- 开发分支管理模型之阿里AoneFlow
说到分支管理模型,令人最为熟悉的莫过于TrunkBased 和 GitFlow. TrunkBased 模型是持续集成思想所崇尚的工作方式,它由单个master分支和许多release分支组成,每个r ...
- Fabric智能合约(base)
这里的智能合约仅包含Init函数和Invoke函数. 为什么一定是这两个方法? 因为在源码中的智能合约模块有这样的接口,如果要完成智能合约的相关编程,就需要实现源码中定义的接口,接口中定义了这两个方法 ...
- java开发相关工具安装包分享
链接:https://pan.baidu.com/s/19rSlXhrZ9AtNdai64tErGQ 提取码:04up
- npm常用模块汇总
npm常用模块汇总: 点击插件名字,查看使用文档 npm常用模块汇总 node常用模块汇总 gulp常用插件汇总 npx 使用教程:npx使用教程 bable:bable这是JavaScript编译器 ...
- 关于全球唯一标识GUID的生成
1.c#生成GUID的几种方式 (1)生成标准的标志符 (36位标准) var strguid = Guid.NewGuid().ToString(); 结果:B2A5AB40-EE29-4791-9 ...
- (1)-Android学习笔记之:初识Android系统架构和项目结构
Android系统架构 Android程序结构 创建一个Android项目,为初学便于理解,将程序项目结构切换为Project模式,项目结构如下 .gradle和.idea:这两个目录下放的都是And ...
- 关于整合ssh中的细节03
关于spring中提供的一些工具类和监听介绍 一.spring提供了一个HibernateTemplate类 ①HibernateTemplate类: 用于操作PO对象,类似Hibernate Ses ...
- [CF1034B] Longest Palindrome - 贪心
如果自己是回文串可以做中心 如果一个串和另一个串的转置相等则可以凑一对 优先配对 #include <bits/stdc++.h> using namespace std; int n,m ...
- Springboot 中的配置文件
Spring Boot提供了两种常用的配置文件,分别是properties文件和yml文件.他们的作用都是修改Spring Boot自动配置的默认值. 技术:yaml.properties语法,Con ...