python面对对象编程中会用到的装饰器
1、property
用途:用来将对像的某个方法伪装成属性来提高代码的统一性。
class Goods: #商品类
discount = 0.8 #商品折扣
def __init__(self,name,price):
self.name = name
self.price = price
def discountprice(self): #定义goodsprice方法,返回折扣后价格
return self.price*self.discount
ipad = Goods('ipad',2299) #实例化商品
print(ipad.name)
print(ipad.discountprice()) #打印商品的的折扣价格
实时上商品的折扣价格应该跟商品的名字一样更像一个对象的属性应该是按照:对象.属性去访问而不是对象.方法()访问,因为抵扣价格更像是商品的属性。而property就可以将方法伪装成对象的属性:
class Goods:
discount = 0.8
def __init__(self,name,price):
self.name = name
self.price = price
@property #在要装饰的方法前面加上@property,注意装饰的方法是除了self不能接受参数的
def discountprice(self):
return self.price*self.discount ipad = Goods('ipad',2299) print(ipad.name) print(ipad.discountprice) #可以直接用访问属性的方法得到折扣价。
#这样我们就实现了对象访问的一致性。
既然伪装成了一个属性,那么属性不仅可以访问应该还可以修改和删除,但是你发现你如果调用ipad.discountprice=1000,会执行错误,因为他本身还是个函数,只是被装饰器伪装了所以这里引入property的另外两个方法setter和deleter:
class Goods:
discount = 0.8
def __init__(self,name,price):
self.name = name
self.price = price
@property
def discountprice(self):
return self.price*self.discount
@discountprice.setter #装饰函数发送修改的值,必须有一个参数
def discountprice(self,new):
self.price = new
@discountprice.deleter
def discountprice(self): #装饰函数删除属性
del self.price ipad = Goods('ipad',2299)
print(ipad.price) #
print(ipad.discountprice) #1839.2
ipad.discountprice = 3000 #执行被@discountprice.setter装饰的函数new接受3000
print(ipad.discountprice) # del ipad.discountprice #删除属性 执行被@discountprice.deleter装饰的函数
print(ipad.discountprice) #报错没有discountprice #以上就完全将discounsprice伪装成了属性,可以按照属性的查删改去调用discountprice,注意被property、setter、deleter装饰的函数一定要同名
# 一个方法被伪装成属性之后
# 应该可以执行一个属性的增删改查操作
# 那么增加和修改 就对应这被setter装饰的方法 :这个方法又一个必传的参数new,表示赋值的时候等号后面的值
# 删除一个属性 对应着 被deleter装饰的方法,这个方法并不能在执行的时候真的删除这个属性,而是你在代码中执行什么就有什么效果
#classmethod
# 在类中定义一个类方法 、是一个装饰器
# 什么时候用?
# 如果你的整个方法中都没有用到对象命名空间中的名字,且你用到了类的命名空间中的名字(普通方法和property方法除外)
# 类方法的默认参数 : cls 值得是调用这个方法的类
# 类方法的调用方式 : 通过类名调用,本质是方法
2、classmethod
故名思议,类方法装饰器,当执行的方法操作的只有类的静态属性时,这个方法应该是用类本身去调用而不是实例的对象,还是上面的例子我要你用去定义个方法更改类的disount的值你会写出下面的代码
class Goods:
__discount = 0.8 # 静态属性
def __init__(self,price):
self.__price = price # 对象属性
self.name = 'apple' def price(self): return self.__price * Goods.__discount def change_discount(self,new): #方法在对象调用时,函数内部并没有引用对象任何属性
Goods.__discount = new
ipad = Goods('ipad',2000)
ipad.chang_discount(0.7) #这样你虽然实现了更改了类的静态属性,但这在编程上是不规范的:对象通过方法修改类属性,在方法内部没有引用任何对象的属性,
#那么把对象调用就多余了,还有类的属性应该由类本身去修改 #修改如下:
class Goods:
__discount = 0.8 # 静态属性
def __init__(self,price):
self.__price = price # 对象属性
self.name = 'apple' def price(self): return self.__price * Goods.__discount
@classmethod
def change_discount(clf,new): # 类方法,这样这里的clf就如同对象调用方法自动将对象传入self一样会自动将类自己传入
clf.__discount = new
Goods.change_discount(0.7) #这样就是类自己调用类方法修改自己的属性,而且只需要输入new的参数即可,crf会自动接收类本身
3、staticmethod
静态方法,就是通常我们定义的函数,内部既不会引用对象方法属性也不会引用类属性和方法,但是需要你将他放在类里面,这样只是为了纯面向对象编程,外部无函数:
class Foo: @classmethod def class_method(cls):pass @staticmethod def static_method():pass
Foo.static() #通过类调用函数,传参和函数还是一致
# staticmethod
# 将一个普通的函数放到类中来就给这个函数加上一个@staticmethod装饰器
# 这个函数就不需要传默认的参数:self,cls
# 静态方法的调用方式 : 通过类名调用,本质还是函数
python面对对象编程中会用到的装饰器的更多相关文章
- python面对对象编程----2:__init__
面对对象编程估计我们最早接触到的就是__init__了,也就是实例的初始化处理过程: 1:来看看最基础的__init__ class Card(object): #抽象类Card,并不用于实例化 de ...
- python面对对象编程------4:类基本的特殊方法__str__,__repr__,__hash__,__new__,__bool__,6大比较方法
一:string相关:__str__(),__repr__(),__format__() str方法更面向人类阅读,print()使用的就是str repr方法更面对python,目标是希望生成一个放 ...
- python面对对象编程----------7:callable(类调用)与context(上下文)
一:callables callables使类实例能够像函数一样被调用 如果类需要一个函数型接口这时用callable,最好继承自abc.Callable,这样有些检查机制并且一看就知道此类的目的是c ...
- Learning Python 012 函数式编程 2 返回函数 匿名函数 装饰器 偏函数
Python 函数式编程 2 返回函数 返回函数的意思就是:函数作为返回值.(高阶函数除了可以接受函数作为参数外,还可以把函数作为结果值返回.) 举个例子:实现一个可变参数的求和. 正常的函数: de ...
- python面对对象编程---------6:抽象基类
抽象基本类的几大特点: 1:要定义但是并不完整的实现所有方法 2:基本的意思是作为父类 3:父类需要明确表示出那些方法的特征,这样在写子类时更加简单明白 用抽象基本类的地方: 1:用作父类 2:用作检 ...
- python面对对象编程-------5:获取属性的四种办法:@property, __setattr__(__getattr__) ,descriptor
一:最基本的属性操作 class Generic: pass g= Generic() >>> g.attribute= "value" #创建属性并赋值 > ...
- python面对对象编程------3:写集合类的三种方法
写一个集合类的三种方法:wrap,extend,invent 一:包装一个集合类 class Deck: def __init__( self ): self._cards = [card6(r+1, ...
- python面对对象编程----1:BlackJack(21点)
昨天读完了<Mastering Object-oriented Python>的第一部分,做一些总结. 首先,第一部分总过八章,名字叫Pythonic Classes via Specia ...
- Python学习6——再谈抽象(面对对象编程)
1.对象魔法 在面对对象编程中,术语对象大致意味着一系列数据(属性)以及一套访问和操作这些数据的方法. 使用对象而非全局变量以及函数的原因有多个,而最重要的好处不过以下几点: 多态:可对不同类型的对象 ...
随机推荐
- <转>大型分布式网站术语浅析
夜半睡起看书,看到一篇关于分布式网站性能优化术语的文章,个人觉得不错,分享出来... 原文地址:大型分布式网站术语分析 一.I/O优化 1.增加缓存,减少磁盘的访问次数. 2.优化磁盘的管理系统,设计 ...
- lesson 8:小程序
程序源代码: //20163683 蔡金阳 信1605-3 import java.io.*; import java.util.Scanner; public class kaoshi { publ ...
- day90
Vue项目简介 最终效果:Vue通过axios发请求给Django后台,Django返回数据给Vue 创建项目: 创建vue项目: -安装node.js -vue脚手架 -vue create 项目名 ...
- day88
ContentType 场景需求:比方说我们现有两种商品,但是他们对应着一个价格策略表,为了防止数据库的浪费,我们在价格策略中加入一个表名字段,一个表明对应的id字段,这样的设计既优化了数据库,还不怕 ...
- BZOJ4237 JOISC2014 稻草人 CDQ分治、单调栈
传送门 题意:给出平面上$N$个点,求满足以下两个条件的矩形:①左下角与右上角各有一个点:②矩形内部没有点.$N \leq 2 \times 10^5$,所有数字大于等于$0$,保证坐标两两不同 最开 ...
- CF101D Castle 树形DP、贪心
题目传送门 题意:给出一个有$N$个点的树,你最开始在$1$号点,经过第$i$条边需要花费$w_i$的时间.每条边只能被经过$2$次.求出到达除$1$号点外所有点的最早时间的最小平均值.$N \leq ...
- 创建一个宽高成比例的弹性div盒子
这里先提供一种,有更好的方法再补充. demo代码如下: <!DOCTYPE html> <html lang="en"> <head> < ...
- asp.net网站,在没有项目源码情况下的扩展
如果在没有源码的情况下,要扩展asp.net网站,可以自己新增一个类库项目,在里面添加需要扩展的类,代码如下: using System; using System.Collections.Gener ...
- ubuntu 中iptables
ubuntu中启动及关闭iptables 在ubuntu中由于不存在 /etc/init.d/iptales文件,所以无法使用service等命令来启动iptables,需要用modprobe命令. ...
- [UWP 自定义控件]了解模板化控件(7):支持Command
以我的经验来说,要让TemplatedControl支持Command的需求不会很多,大部分情况用附加属性解决这个需求会更便利些,譬如UWPCommunityToolkit的HyperlinkExte ...