前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Python 面向对象 高阶-描述符与设计模式#学习猿地

Python 面向对象 高阶-描述符与设计模式#学习猿地

作者头像
学习猿地
发布2020-03-19 16:55:32
2940
发布2020-03-19 16:55:32
举报
文章被收录于专栏:学习猿地学习猿地

## 描述符

> 当一个类中,包含了三个魔术方法(`__get__,__set__,__delete__`)之一,或者全部时,那么这个类就称为描述符类

### 作用

> 描述符的作用就是对一个类中的某个成员进行一个详细的管理操作(获取,赋值,删除)

> 描述符就是代理了一个类中的成员的操作,描述符属于类,只能定义为类的属性

### 三个魔术方法

```python

'''

__get__(self, instance, owner)

触发机制:在访问对象成员属性时自动触发(当该成员已经交给描述符管理时)

作用:设置当前属性获取的值

参数:1. self 描述符对象 2.被管理成员的类的对象。3.被管理成员的类

返回值:返回值作为成员属性获取的值

注意事项:无

__set__(self, instance, value)

触发机制:在设置对象成员属性时自动触发(当该成员已经交给描述符管理时)

作用:对成员的赋值进行管理

参数:1. self 描述符对象 2.被管理成员的类的对象。3.要设置的值

返回值:无

注意事项:无

__delete__(self, instance)

触发机制:在删除对象成员属性时自动触发(当该成员已经交给描述符管理时)

作用:对成员属性的删除进行管理

参数:1. self 描述符对象 2.被管理成员的类的对象。

返回值:无

注意事项:无

'''

```

### 数据描述符:(完整)

> 同时具备三个魔术方法的类就是 数据描述符

### 非数据描述符:(不完整)

> 没有同时具备三个魔术方法的类就是 非描述符类

### 基本使用格式

> 把当前的描述符类赋值给一个需要代理的类中的成员属性

代码示例:

```python

# 定义描述符类

class PersonName():

__name = 'abc'

def __get__(self, instance, owner):

# print(self,instance,owner)

return self.__name

def __set__(self, instance, value):

# print(self,instance,value)

self.__name = value

def __delete__(self, instance):

# print(self,instance)

# del self.__name

print('不允许删除')

# 定义的普通类

class Person():

# 把类中的一个成员属性交给一个描述符类来实现

# 一个类中的成员的值是另一个描述符类的对象()

# 那么当对这个类中得成员进行操作时,可以理解为就是对另一个对象的操作

name = PersonName()

# 实例化对象

zs = Person()

print(zs.name)

zs.name = '张三'

print(zs.name)

del zs.name

print(zs.name)

```

### 描述符应用解析

```python

#定义一个学生类,需要记录 学员的id,名字,分数

class Student():

def __init__(self,id,name,score):

self.id = id

self.name = name

self.score = score

def returnMe(self):

info = f'''

学员编号:{self.id}

学员姓名:{self.name}

学员分数:{self.score}

'''

print(info)

'''

# 要求:学员的分数只能在0-100范围中

解决方法:

1。在__init__方法中检测当前分数范围

# 检测分数范围

if score >= 0 and score <= 100:

self.score = score

这个解决方案只能在对象初始化时有效。

2。 定义一个setattr魔术方法检测

检测如果给score分数进行赋值时,进行分数的检测判断

def __setattr__(self, key, value):

# 检测是否是给score进行赋值操作

if key == 'score':

# 检测分数范围

if value >= 0 and value <= 100:

object.__setattr__(self, key, value)

else:

print('当前分数不符合要求')

else:

object.__setattr__(self,key,value)

假如 学员的分数不止一个时怎么办,比如 语文分数,数学分数,英语分数

另外就是当前这个类中的代码是否就比较多了呢?

3。可以思考使用描述符来代理我们的分数这个属性

1.定义Score描述符类

2.把学生类中的score这个成员交给描述符类进行代理

3.只要在代理的描述符类中对分数进行赋值和获取就ok了

'''

#定义描述符类 代理分数的管理

class Score():

def __get__(self, instance, owner):

return self.__score

def __set__(self, instance, value):

if value >= 0 and value <= 100:

self.__score = value

else:

print('分数不符合要求')

# 使用描述符类代理score分数属性

class Student():

score = Score()

def __init__(self,id,name,score):

self.id = id

self.name = name

self.score = score

def returnMe(self):

info = f'''

学员编号:{self.id}

学员姓名:{self.name}

学员分数:{self.score}

'''

print(info)

# 实例化对象

zs = Student(1011,'张三疯',99)

zs.returnMe()

zs.score = -20

zs.score = 88

zs.returnMe()

```

### 描述符的三种定义格式

```python

# 格式一 通过定义 描述符类来实现 推荐

'''

class ScoreManage():

def __get__(self, instance, owner):

pass

def __set__(self, instance, value):

pass

def __delete__(self, instance):

pass

class Student():

score = ScoreManage()

'''

# 格式二, 使用 property 函数 来实现

'''

class Student():

# 在当前需要被管理的类中 直接定义类似下面三个方法

def getscore(self):

print('getscore')

def setscore(self,value):

print('setscore',value)

def delscore(self):

print('delscore')

# 在 property 函数中指定对应的三个方法,对应的方法 1。__get__,2。__set__,3。__delete__

score = property(getscore,setscore,delscore)

zs = Student()

# print(zs.score)

# zs.score = 200

# del zs.score

'''

# 格式三 使用 @property 装饰器语法来实现

'''

class Student():

__score = None

@property

def score(self):

print('get')

return self.__score

@score.setter

def score(self,value):

print('set')

self.__score = value

@score.deleter

def score(self):

print('delete')

del self.__score

zs = Student()

# print(zs.score)

zs.score = 199

print(zs.score)

del zs.score

'''

```

## 设计模式

> 设计模式是前人为完成某个功能或需求,根据经验和总结,对实现的代码步骤和代码设计进行了总结和归纳,成为了实现某个需求的经典模式。

>

> 设计模式并不是固定的代码格式,而是一种面向对象编程的设计

### 单例(单态)设计模式

> 在当前脚本中,同一个类只能创建出一个对象去使用。这种情况就成为单例(单态)。

```python

'''

实现单例的案例,思考:

单例和婚姻法的关系,特别像,一个人只能有一个结婚对象

在社会中是如何完成一夫一妻制的?

如果要结婚,必须要到 民政局 登记

民政局 需要检测两个人的户口本,看上面是否属于 结婚的状态

如果是已婚,肯定就撵出去了。

如果没有结婚,可以给盖个章了,开始登记。

那么按照这样的思路如何去实现 python中的单例设计模式呢?

1。需要有一个方法,可以去控制当前对象的创建过程?

构造方法 __new__

2。需要有一个标示来存储和表示是否有对象

创建一个私有属性 进行存储,默认值为None

3。在创建对象的方法中去检测和判断是否有对象?

如果没有对象,则创建对象,并且把对象存储起来,返回对象

如果存储的是对象,则直接返回对象,就不需要创建新的对象了

'''

class Demo():

# 2.定义私有属性存储对象,默认值为None

__obj = None

# 1.定义构造方法

def __new__(cls, *args, **kwargs):

# 3。在创建对象的过程中,判断是否有对象

if not cls.__obj:

# 判断如果没有对象,则创建对象,并且存储起来

cls.__obj = object.__new__(cls)

# 直接把存储的对象返回

return cls.__obj

# 实例化对象

a = Demo()

b = Demo()

print(a)

print(b)

'''

<__main__.Demo object at 0x106f4d850>

<__main__.Demo object at 0x106f4d850>

'''

```

### Mixin 混合设计模式

#### Mixin类

+ Mixin 必须是表示一种功能,而不是一个对象。

+ Mixin 的功能必须单一,如果有多个功能,那就多定义Mixin类

+ python 中的Mixin是通过多继承实现的

+ Mixin 这个类通常不单独使用,而是混合到其它类中,去增加功能的

+ Mixin 类不依赖子类的实现,即便子类没有继承这个Mixin,子类也能正常运行,可能就是缺少了一些功能。。

#### 使用Mixin混入类的好处?

1. Mixin 这个混入类的设计模式,在不对类的内容修改的前提下,扩展了类的功能

2. Mixin 混入类为了提高代码的重用性,使得代码结构更加简单清晰

3. 可以根据开发需要任意调整功能(创建新的Mixin混入类)避免设计多层次的复杂的继承关系。

示例:

```python

'''

继承需要有一个必要的前提,继承应该是一个 'is-a' 的关系

例如:

苹果可以去继承水果,因为苹果就是一个水果

苹果不能继承午饭,因为午饭可以有苹果也可以没有

比如 汽车可以继承 交通工具,因为汽车本身就是一个交通工具

交通工具有哪些?

汽车,飞机,直升机,这些都属于 交通工具

那么如何去设计这些类的关系呢?

比如创建一个交通工具类,然后属于交通工具的都来继承,再去实现。。。

但是,飞机和直升机都有飞行的功能,而汽车并没有,那么在交通工具中如果去定义 飞行这个功能,那就不太合适了。。

能不能在飞机和直升机类中分别实现 飞行 这个功能呢?可以,但是代码又无法重用。

怎么办?

单独去定义交通工具类,和 飞行器 这个两个父类,这样飞机和直升机就可以去继承这两个类.

'''

# 交通工具 vehicle

class vehicle():

# 运输货物

def huo(self):

print('运输货物')

# 搭载乘客

def ren(self):

print('搭载乘客')

# 飞行器

class FlyingMixin():

def fly(self):

print('可以起飞了。。。')

# 定义汽车类

class cart(vehicle):

pass

# 定义飞机

class airplane(vehicle,FlyingMixin):

pass

# 定义直升机

class helicopter(vehicle,FlyingMixin):

pass

# 此时去定义一个飞行器的类 Flying,让需要飞行的交通工具,直接继承这个类。可以解决这个问题。

# 但是,1。出现类多继承,违背了'is-a' 2。飞行器这个类很容易被误解

# 解决方案也是使用多继承,但是给飞行器这个类,定义成为一个 Mixin 混合类,

# 此时就是等于把飞行器这个类,作为了一个扩展的功能,来扩展其它类

'''

在上面的代码中,虽然直升机和飞机都使用了多继承,也就是继承了FlyingMixin

但是由于 FlyingMixin 类加了 Minin这个名,就告诉了后面阅读代码的人,这个类是一个Mixin类

'''

```

### 抽象类(了解)

> 抽象类是一个特殊的类:

> 1. 抽象类不能用,不能直接实例化成为一个对象。

> 2. 抽象类中包含了抽象方法,抽象方法就是没有实现代码的方法。

> 3. 抽象类需要子类继承,并重写父类的抽象方法。才可以使用。

```

抽象类,一般应用在程序设计,程序设计中一般是要对功能和需求进行规划,其中有一些需求是明确的并且可以完成的,

但是也可能会有一些需求是不明确的,或者不确定具体需要怎么实现,

那么此时就可以把这个不确定怎么实现或者需要后面再去实现的方法,定义为抽象方法(只定义方法名,不写具体代码)

抽象类的应用:

例如要开发一个框架,这个框架要有一些基本功能和扩展功能。。。。

但是你具体用这个框架开发什么样的产品,开发框架的人并不清楚或者确定。

因此框架就具备一定的功能,并且留下来一些方法的定义,剩下的就是需要自己在方法中具体实现自己业务逻辑。

```

抽象类的定义:

```python

import abc

# 如果要定义为抽象类,那么这个类的 metaclass属性必须是 metaclass=abc.ABCMeta

class WriteCode(metaclass=abc.ABCMeta):

#需要抽象的方法,使用装饰器进行装饰

@abc.abstractmethod

def write_php(self):

pass

def write_java(self):

print('实现了java代码的开发')

def write_python(self):

print('实现了python代码的开发')

# 抽象类不能直接实例化对象

# obj = WriteCode()

# print(obj)

#TypeError: Can't instantiate abstract class WriteCode with abstract methods write_php

# 定义子类,继承抽象类,并实现抽象类中的抽象方法

class Demo(WriteCode):

def write_php(self):

print('实现了php代码的开发')

a = Demo()

print(a)

a.write_java()

a.write_php()

a.write_python()

```

掌握学习方法,不如会弯道超车!

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
对象存储
对象存储(Cloud Object Storage,COS)是由腾讯云推出的无目录层次结构、无数据格式限制,可容纳海量数据且支持 HTTP/HTTPS 协议访问的分布式存储服务。腾讯云 COS 的存储桶空间无容量上限,无需分区管理,适用于 CDN 数据分发、数据万象处理或大数据计算与分析的数据湖等多种场景。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档