Python面向对象:封装和多态(python面向对象程序)
off999 2024-10-25 13:44 33 浏览 0 评论
一、封装
封装是隐藏对象的属性和实现细节,仅对外公开接口,控制在程序中属性的读取和修改的访问级别。
封装就是将抽象得到的数据和行为(或功能)相结合,形成一个有机的整体,也就是将数据与操作数据的源代码进行有机的结合,形成“类”,其中数据和函数都是类的成员。
1、简单理解封装
顾名思义,封装属性就是把已有的属性封装到一个类里面去:
class Person():
def __init__(self, name, age, sex):
self.name = name
self.age = age
self.sex = sex
jack = Person('jack', 18, '男')
#将jack、 18、 男 封装到jack对象(self)的name、age、sex中
#name、age、sex又封装在了Person类中
print(jack.__dict__)
#{'name': 'jack', 'age': 18, 'sex': '男'}
分析:self是一个形式参数,创建什么对象,它就代表那个对象
2、调用封装的属性
通过对象直接调用:
class Person():
def __init__(self, name, age, sex):
self.name = name
self.age = age
self.sex = sex
jack = Person('jack', 18, '男')
print(jack.name, jack.sex, jack.age)
#jack 男 18
分析:在这里,我们可以用对象名随意调用到自身的属性,并进行属性修改,从安全的角度来看,这样是很不安全的,所以需要将属性隐藏起来,也就是私有化。
私有化属性的方法:类中定义私有的,只有类内部使用,外部无法访问(比如_(杠) __(杠杠) )
class Person():
def __init__(self, name, age, sex):
self.__name = name
self.__age = age
self.__sex = sex
jack = Person('jack', 18, '男')
print(jack._Person__name)#jack
print(jack.name)
#Error:'Person' object has no attribute 'name'
分析:
1、通过使用__(杠杠)的方法使得类Person属性name、age、sex成功私有化,子类无法直接调用,但是通过jack._Person__name的方式可以调用到私有化的属性,并且能对其修改,说明python在设置私有属性的时候,只是把属性的名字换成了其他的名字。
2、类中以_或者__的属性,都是私有属性,禁止外部调用。虽然可以通过特殊的手段获取到,并且赋值,但是这么做不觉的很蛋疼么,本来就是设置私有属性,还非要去强制修改。
私有化属性设置好了,不可能是存在那里谁都不让使用的,要不然设置私有化属性就失去了本身的意义,我们只是不想让私有化属性直接被随意的修改,而不是拒绝访问,所以还需要给私有化属性提供查找和修改的接口,我们只需要通过对接口的控制,就能有效的控制私有化属性的数据安全,比如对接口进行设置,就不会出现age被赋值为负数。
class Person(object):
def __init__(self, name, age, sex):
self.__name = name
self.__age = age
self.__sex = sex
def get_age(self):
return self.__age
def set_age(self, age):
if age > 150 or age < 0:
print('年龄必须大于0,小于150')
else:
self.__age = age
jack = Person('jack', 18, '男')
#访问age属性
print(jack.get_age())#18
#修改age属性
jack.set_age(100)
print(jack.get_age())#100
#非法修改age属性
jack.set_age(-20)#年龄必须大于0,小于150
print(jack.get_age())#100
分析:这样就完美了,我们既可以访问到实例化对象内部的属性,也可以在数据安全的情况下(禁止非法数据修改),修改对象的属性
3、python自带的调用私有化数据的方法
前面,我们用set和get的方式来调用或修改对象本身的私有化属性,达到了数据安全的目的,其实python中提供了一种直接用obj.属性名的方式调用类的私有化属性,也能保证数据安全。
class Person(object):
def __init__(self, name, age, sex):
self.__name = name
self.__age = age
self.__sex = sex
@property
def age(self):
return self.__age
@age.setter
def age(self, age):
if age > 150 or age < 0:
print('年龄必须大于0,小于150')
else:
self.__age = age
jack = Person('jack', 18, '男')
#访问age属性
print(jack.age)#18
#修改age属性
jack.age = 100
print(jack.age)#100
#非法修改age属性
jack.age = -20#年龄必须大于0,小于150
print(jack.age)#100
分析:
1、使用 @property 装饰器时,接口名不必与属性名相同。
2、凡是赋值语句,就会触发set方法。获取属性值,会触发get方法。
3、我们可以使用@property装饰器来创建只读属性,@property装饰器会将方法转换为相同名称的只读属性,可以与所定义的属性配合使用,这样可以防止属性被修改。
二、多态
接口的多种不同的实现方式即为多态。
多态最核心的思想就是,父类的引用可以指向子类的对象,或者接口类型的引用可以指向实现该接口的类的实例。
多态是一种运行期的行为,不是编译期行为!在编译期间它只知道是一个引用,只有到了执行期,引用才知道指向的是谁。这就是所谓的“软绑定”。
多态是一项让程序员“将改变的事物和未改变的事物分离开来”重要技术。
1、多态性
多态性是指指具有不同功能的函数可以使用相同的函数名,这样就可以用一个函数名调用不同内容的函数。
在面向对象方法中一般是这样表述多态性:向不同的对象发送同一条消息,不同的对象在接收时会产生不同的行为。
不同的行为就是指不同的实现,即执行不同的函数。
class Animals(object):
def talk(self):
pass
class Person(Animals):
def talk(self):
print('高级语言')
class Cat(Animals):
def talk(self):
print('喵喵喵')
class Dog(Animals):
def talk(self):
print('汪汪汪')
per = Person()
cat = Cat()
dog = Dog()
# 定义一个统一的接口来访问
def fun(obj):
obj.talk()
fun(per)#高级语言
fun(cat)#喵喵喵
fun(dog)#汪汪汪
分析:
1、per对象、cat对象、dog对象是通过Animals类实现的三种不同形态,这就是多态的体现。
2、per、cat、dog对象都是通过fun(obj)的同一种方式调用,实现了不同的效果,这就是多态性的体现,所以多态性可以说是一个接口,多种实现
3、多态性的优点:
3.1、增加了程序的灵活性:以不变应万变,不论对象千变万化,使用者都是同一种形式去调用,如fun(obj)
3.2、增加了程序额可扩展性:通过继承Animal类派生新的类(Person类、Cat类、Dog类),使用者无需更改自己的代码,还是用fun(obj)去调用
2、鸭子类型
调用不同的子类将会产生不同的行为,而无须明确知道这个子类实际上是什么,这是多态的重要应用场景。而在python中,因为鸭子类型(duck typing)使得其多态不是那么酷,原因是python是强类型的动态脚本语言,不使用显示数据类型声明,且确定一个变量的类型是在第一次给它赋值的时候。
鸭子类型是动态类型的一种风格。在这种风格中,一个对象有效的语义,不是由继承自特定的类或实现特定的接口,而是由"当前方法和属性的集合"决定。这个概念的名字来源于由James Whitcomb Riley提出的鸭子测试,“鸭子测试”可以这样表述:“当看到一只鸟走起来像鸭子、游泳起来像鸭子、叫起来也像鸭子,那么这只鸟就可以被称为鸭子。”
在鸭子类型中,关注的不是对象的类型本身,而是它是如何使用的。例如,在不使用鸭子类型的语言中,我们可以编写一个函数,它接受一个类型为"鸭子"的对象,并调用它的"走"和"叫"方法。在使用鸭子类型的语言中,这样的一个函数可以接受一个任意类型的对象,并调用它的"走"和"叫"方法。如果这些需要被调用的方法不存在,那么将引发一个运行时错误。任何拥有这样的正确的"走"和"叫"方法的对象都可被函数接受的这种行为引出了以上表述,这种决定类型的方式因此得名。
鸭子类型通常得益于不测试方法和函数中参数的类型,而是依赖文档、清晰的代码和测试来确保正确使用。
class Duck(object):
def walk(self):
print('I walk like a duck')
def swim(self):
print('I swim like a duck')
class Person():
def walk(self):
print('this one walk like a duck')
def swim(self):
print('this man swim like a duck')
def fun(obj):
obj.walk()
obj.swim()
fun(Duck())
# I walk like a duck
# I swim like a duck
fun(Person())
#this one walk like a duck
#this man swim like a duck
分析:可以看出Pseron类拥有和Duck类一样的方法,当程序调用Duck类,并利用了其中的walk和swim方法时,我们传入Person类也一样可以运行,程序并不会检查类型是不是Duck,只要他拥有 walk()和swim()方法,就能被正确地调用。
再举例,如果一个对象实现了__getitem__方法,那python的解释器就会把它当做一个collection,就可以在这个对象上使用切片,获取子项等方法;
如果一个对象实现了__iter__和next方法,python就会认为它是一个iterator,就可以在这个对象上通过循环来获取各个子项。
class Foo:
def __iter__(self):
pass
def __next__(self):
pass
from collections import Iterable
from collections import Iterator
print(isinstance(Foo(), Iterable)) # True
print(isinstance(Foo(), Iterator)) # True
因此,这就诠释了“当看到一只鸟走起来像鸭子、游泳起来像鸭子、叫起来也像鸭子,那么这只鸟就可以被称为鸭子。”这句话,那只鸟是不是鸭子不重要,重要的是它有和鸭子一样的方法,把它当鸭子调用,程序就不会报错。
需要源码01私信小编
相关推荐
- 面试官:来,讲一下枚举类型在开发时中实际应用场景!
-
一.基本介绍枚举是JDK1.5新增的数据类型,使用枚举我们可以很好的描述一些特定的业务场景,比如一年中的春、夏、秋、冬,还有每周的周一到周天,还有各种颜色,以及可以用它来描述一些状态信息,比如错...
- 一日一技:11个基本Python技巧和窍门
-
1.两个数字的交换.x,y=10,20print(x,y)x,y=y,xprint(x,y)输出:102020102.Python字符串取反a="Ge...
- Python Enum 技巧,让代码更简洁、更安全、更易维护
-
如果你是一名Python开发人员,你很可能使用过enum.Enum来创建可读性和可维护性代码。今天发现一个强大的技巧,可以让Enum的境界更进一层,这个技巧不仅能提高可读性,还能以最小的代价增...
- Python元组编程指导教程(python元组的概念)
-
1.元组基础概念1.1什么是元组元组(Tuple)是Python中一种不可变的序列类型,用于存储多个有序的元素。元组与列表(list)类似,但元组一旦创建就不能修改(不可变),这使得元组在某些场景...
- 你可能不知道的实用 Python 功能(python有哪些用)
-
1.超越文件处理的内容管理器大多数开发人员都熟悉使用with语句进行文件操作:withopen('file.txt','r')asfile:co...
- Python 2至3.13新特性总结(python 3.10新特性)
-
以下是Python2到Python3.13的主要新特性总结,按版本分类整理:Python2到Python3的重大变化Python3是一个不向后兼容的版本,主要改进包括:pri...
- Python中for循环访问索引值的方法
-
技术背景在Python编程中,我们经常需要在循环中访问元素的索引值。例如,在处理列表、元组等可迭代对象时,除了要获取元素本身,还需要知道元素的位置。Python提供了多种方式来实现这一需求,下面将详细...
- Python enumerate核心应用解析:索引遍历的高效实践方案
-
喜欢的条友记得关注、点赞、转发、收藏,你们的支持就是我最大的动力源泉。根据GitHub代码分析统计,使用enumerate替代range(len())写法可减少38%的索引错误概率。本文通过12个生产...
- Python入门到脱坑经典案例—列表去重
-
列表去重是Python编程中常见的操作,下面我将介绍多种实现列表去重的方法,从基础到进阶,帮助初学者全面掌握这一技能。方法一:使用集合(set)去重(最简单)pythondefremove_dupl...
- Python枚举类工程实践:常量管理的标准化解决方案
-
本文通过7个生产案例,系统解析枚举类在工程实践中的应用,覆盖状态管理、配置选项、错误代码等场景,适用于Web服务开发、自动化测试及系统集成领域。一、基础概念与语法演进1.1传统常量与枚举类对比#传...
- 让Python枚举更强大!教你玩转Enum扩展
-
为什么你需要关注Enum?在日常开发中,你是否经常遇到这样的代码?ifstatus==1:print("开始处理")elifstatus==2:pri...
- Python枚举(Enum)技巧,你值得了解
-
枚举(Enum)提供了更清晰、结构化的方式来定义常量。通过为枚举添加行为、自动分配值和存储额外数据,可以提升代码的可读性、可维护性,并与数据库结合使用时,使用字符串代替数字能简化调试和查询。Pytho...
- 78行Python代码帮你复现微信撤回消息!
-
来源:悟空智能科技本文约700字,建议阅读5分钟。本文基于python的微信开源库itchat,教你如何收集私聊撤回的信息。[导读]Python曾经对我说:"时日不多,赶紧用Python"。于是看...
- 登录人人都是产品经理即可获得以下权益
-
文章介绍如何利用Cursor自动开发Playwright网页自动化脚本,实现从选题、写文、生图的全流程自动化,并将其打包成API供工作流调用,提高工作效率。虽然我前面文章介绍了很多AI工作流,但它们...
- Python常用小知识-第二弹(python常用方法总结)
-
一、Python中使用JsonPath提取字典中的值JsonPath是解析Json字符串用的,如果有一个多层嵌套的复杂字典,想要根据key和下标来批量提取value,这是比较困难的,使用jsonpat...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python自定义函数 (53)
- python进度条 (67)
- python吧 (67)
- python字典遍历 (54)
- python的for循环 (65)
- python格式化字符串 (61)
- python串口编程 (60)
- python读取文件夹下所有文件 (59)
- java调用python脚本 (56)
- python操作mysql数据库 (66)
- python字典增加键值对 (53)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python人脸识别 (54)
- python多态 (60)
- python命令行参数 (53)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)