时间:2021-05-22
导语:本文章记录了本人在学习Python基础之元编程篇的重点知识及个人心得,打算入门Python的朋友们可以来一起学习并交流。
属性:在Python中,数据的属性和处理数据的方法统称属性。
元编程:用元类进行编程,元类→类→对象,元类比类更抽象,生成类的类。
1、使用动态属性访问JSON类数据
第一版:利用json.load(fp)审查数据
from urllib.request import urlopenimport warningsimport osimport jsonURL = 'http:///pub/sc/osconfeed'JSON = 'data/osconfeed.json'def load(): if not os.path.exists(JSON): msg = 'downloading {} to {}'.format(URL, JSON) warnings.warn(msg) #如果需要下载就发出提醒。 with urlopen(URL) as remote, open(JSON, 'wb') as local: #在with语句中使用两个上下文管理器分别用于读取和保存远程文件。 local.write(remote.read()) with open(JSON) as fp: return json.load(fp)#json.load函数解析JSON文件,返回Python原生对象。第二版:使用动态属性访问JSON类数据
第一版查阅深层数据的格式比较冗长,例如feed'Schedule'40,我们希望在读取属性上采用feed.Schedule.events[40].name这类方式来改进。并且第二版的类能递归,自动处理嵌套的映射和列表。
from collections import abcclass FronenJSON(): def __init__(self,mapping): self.__data=dict(mapping)#创建副本,同时确保处理的是字典。 def __getattr__(self, name):#仅当没有指定名称的属性才调用__getattr__方法。 if hasattr(self,name): return getattr(self.__data,name) else: return FronenJSON.build(self.__data[name]) @classmethod def __build__(cls,obj): if isinstance(obj,abc.Mapping):#判断obj是否是映射。 return cls(obj)#创建FrozenJSON对象。 elif isinstance(obj,abc.MutableSequence): return [cls.build(item) for item in obj]#递归调用.build()方法,构建一个列表。 else:#既不是字典也不是列表,则返回元素本身。 return obj分析: FronenJSON类的关键是__getattr__方法。仅当无法使用常规的方式获取属性(即在实例、类或超类中找不到指定的属性),解释器才会调用特殊的__getattr__方法。
2、处理无效属性名
在Python中,由于关键字被保留,名称为关键字的属性是无效的。因此需要对第二版中的__init__进行改进:
def __init__(self,mapping): self.__data={} for key,value in mapping.items(): if keyword.iskeyword(key): key+='_'#与Python关键字重复的key在尾部加上下划线。 self.__data[key]=value3、使用特殊方法__new__
第三版:使用__new__构造方法把一个类转换成一个灵活的对象工厂函数。
from collections import abcclass FronenJSON(): def __new__(cls, arg): # __new__是类方法,第一个参数是类本身cls。 if isinstance(arg, abc.Mapping): return super().__new__(cls) #委托给超类object基类的__new__方法处理。 elif isinstance(arg, abc.MutableSequence): # 余下方法与原先的build方法一致。 return [cls(item) for item in arg] else: return arg def __init__(self,mapping): self.__data={} for key,value in mapping.items(): if keyword.iskeyword(key): key+='_' self.__data[key]=value def __getattr__(self, name): if hasattr(self,name): return getattr(self.__data,name) else: return FronenJSON(self.__data[name])1、类属性、实例属性、私有属性与特性
类属性:类属性在__init__()外初始化,属于类所有,所有实例共享一个属性。
调用方法:类属性在内部用classname.类属性名调用,外部既可以用classname.类属性名又可以用instancename.类属性名来调用。
实例属性:实例属性属于各个实例所有,互不干扰。
私有属性:
特性:是用于管理实例属性的类属性。
特性用途:经常用于把公开的属性变成使用读值方法和设值方法管理的属性,且在不影响客户端代码的前提下实施业务规则。
注意:
这是因为obj.attr不会从实例obj开始寻找attr,而是从obj.__class__开始;而且仅当类中没有名为attr的特性时,Python才会在实例中寻找attr。
简言之,就遮盖层级而言,类特性>实例属性>类属性。
2、使用特性验证属性
使用特性可以验证实例属性的有效性,同时能够根据已知属性和属性之间的关系式调整其他属性,避免硬编码。
案例:假设某商店经营坚果、杂粮等多种有机食物,每位顾客的订单会包含店中的一系列商品,我们需要根据客户的订单计算出总价。
分析:我们不希望顾客订单的商品重量为非正数,需要借助@property装饰器实现值的获取与设置,从而验证实例属性的有效性。代码如下:
class LineItem(): def __init__(self,description,weight,price): self.description=description self.weight=weight self.price=price def subtotal(self): return self.weight*self.price @property#读值。 def weight(self): return self.__weight#真正的值存储在私有属性中。 @weight.setter def weight(self,value): if value >0: self.__weight=value#有效值存入私有属性中。 else: raise ValueError('Value must be > 0')#对于无效的值抛出ValueError。Tips:当我们需要设置只读属性时,只使用@property,无需使用@func.setter。
原理解析:为了更好地理解@property装饰器的原理,我们写一版效果相同但没使用装饰器的代码。
class LineItem: def __init__(self, description, weight, price): self.description = description self.weight = weight self.price = price def subtotal(self): return self.weight * self.price def get_weight(self): #普通读值方法。 return self.__weight def set_weight(self, value): #普通设值方法。 if value > 0: self.__weight = value else: raise ValueError('value must be > 0') weight = property(get_weight, set_weight) #构建property对象,赋值给公开的类特性。property 构造方法的完整签名:
property(fget=None, fset=None, fdel=None, doc=None)3、特性工厂函数
抽象定义特性的方式有两种,一是使用特性工厂函数,二是使用描述符类。
下面我们用特性工厂函数来完成上文中提到的订单结算案例:
4、使用特性删除属性
class BlackKnight: def __init__(self): self.members = ['an arm', 'another arm', 'a leg', 'another leg'] self.phrases = ["'Tis but a scratch.", "It's just a flesh wound.", "I'm invincible!", "All right, we'll call it a draw."] @property def member(self): print('next member is:') return self.members[0] @member.deleter def member(self): text = 'BLACK KNIGHT (loses {})\n-- {}' print(text.format(self.members.pop(0), self.phrases.pop(0)))删除属性只需在主程序中发出指令:del obj.attr
1、特殊属性
2、内置函数
3、特殊方法
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
动态语言与静态语言有很多不同,最大的特性之一就是可以实现动态的对类和实例进行修改,在Python中,我们创建了一个类后可以对实例和类绑定心的方法或者属性,实现动
简介在廖雪峰的python网站上,他是这么说的python是动态语言,它允许程序在执行过程中动态绑定属性或者方法(使用MethodTpye)。某个实例在执行过程
JavaScript中属性和特性是完全不同的两个概念,这里我将根据自己所学,来深入理解JavaScript中的属性和特性。主要内容如下:理解JavaScript
本文主要讲述的是对Python中property属性(特性)的理解,具体如下。定义及作用:在property类中,有三个成员方法和三个装饰器函数。三个成员方法分
C#动态创建button按钮的方法实例详解C#编程中经常需要动态创建,本文主要介绍C#动态创建button按钮的方法,涉及C#按钮属性动态设置的相关技巧,以供借