Python面向对象:封装和多态(python面向对象程序)
off999 2024-10-25 13:44 61 浏览 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私信小编
相关推荐
- apisix动态修改路由的原理_动态路由协议rip的配置
-
ApacheAPISIX能够实现动态修改路由(DynamicRouting)的核心原理,是它将传统的静态Nginx配置彻底解耦,通过中心化配置存储(如etcd)+OpenRest...
- 使用 Docker 部署 OpenResty Manager 搭建可视化反向代理系统
-
在之前的文章中,xiaoz推荐过可视化Nginx反向代理工具NginxProxyManager,最近xiaoz还发现一款功能更加强大,界面更加漂亮的OpenRestyManager,完全可以替代...
- OpenResty 入门指南:从基础到动态路由实战
-
一、引言1.1OpenResty简介OpenResty是一款基于Nginx的高性能Web平台,通过集成Lua脚本和丰富的模块,将Nginx从静态反向代理转变为可动态编程的应用平台...
- OpenResty 的 Lua 动态能力_openresty 动态upstream
-
OpenResty的Lua动态能力是其最核心的优势,它将LuaJIT嵌入到Nginx的每一个请求处理阶段,使得开发者可以用Lua脚本动态控制请求的生命周期,而无需重新编译或rel...
- LVS和Nginx_lvs和nginx的区别
-
LVS(LinuxVirtualServer)和Nginx都是常用的负载均衡解决方案,广泛应用于大型网站和分布式系统中,以提高系统的性能、可用性和可扩展性。一、基本概念1.LVS(Linux...
- 外网连接到内网服务器需要端口映射吗,如何操作?
-
外网访问内网服务器通常需要端口映射(或内网穿透),这是跨越公网与私网边界的关键技术。操作方式取决于网络环境,以下分场景详解。一、端口映射的核心原理内网服务器位于私有IP地址段(如192.168.x.x...
- Nginx如何解决C10K问题(1万个并发连接)?
-
关注△mikechen△,十余年BAT架构经验倾囊相授!大家好,我是mikechen。Nginx是大型架构的必备中间件,下面我就全面来详解NginxC10k问题@mikechen文章来源:mikec...
- 炸场!Spring Boot 9 大内置过滤器实战手册:从坑到神
-
炸场!SpringBoot9大内置过滤器实战手册:从坑到神在Java开发圈摸爬滚打十年,见过太多团队重复造轮子——明明SpringBoot自带的过滤器就能解决的问题,偏偏要手写几十...
- WordPress和Typecho xmlrpc漏洞_wordpress主题漏洞
-
一般大家都关注WordPress,毕竟用户量巨大,而国内的Typecho作为轻量级的博客系统就关注的人并不多。Typecho有很多借鉴WordPress的,包括兼容的xmlrpc接口,而WordPre...
- Linux Shell 入门教程(六):重定向、管道与命令替换
-
在前几篇中,我们学习了函数、流程控制等Shell编程的基础内容。现在我们来探索更高级的功能:如何控制数据流向、将命令链接在一起、让命令间通信变得可能。一、输入输出重定向(>、>>...
- Nginx的location匹配规则,90%的人都没完全搞懂,一张图让你秒懂
-
刚配完nginx网站就崩了?运维和开发都头疼的location匹配规则优先级,弄错顺序直接导致500错误。核心在于nginx处理location时顺序严格:先精确匹配=,然后前缀匹配^~,接着按顺序正...
- liunx服务器查看故障命令有那些?_linux查看服务器性能命令
-
在Linux服务器上排查故障时,需要使用一系列命令来检查系统状态、日志文件、资源利用情况以及网络状况。以下是常用的故障排查命令,按照不同场景分类说明。1.系统资源相关命令1.1查看CPU使...
- 服务器被入侵的常见迹象有哪些?_服务器入侵可以被完全操纵吗
-
服务器被入侵可能会导致数据泄露、服务异常或完全失控。及时发现入侵迹象能够帮助你尽早采取措施,减少损失。以下是服务器被入侵的常见迹象以及相关的分析与处理建议。1.服务器被入侵的常见迹象1.1系统性能...
- 前端错误可观测最佳实践_前端错误提示
-
场景解析对于前端项目,生产环境的代码通常经过压缩、混淆和打包处理,当代码在运行过程中产生错误时,通常难以还原原始代码从而定位问题,对于深度混淆尤其如此,因此Mozilla自2011年开始发起并...
- 8个能让你的Kubernetes集群“瞬间崩溃”的配置错误
-
错误一:livenessProbe探针“自杀式”配置——30秒内让Pod重启20次现象:Pod状态在Running→Terminating→CrashLoopBackOff之间循环,重启间隔仅...
你 发表评论:
欢迎- 一周热门
- 最近发表
-
- apisix动态修改路由的原理_动态路由协议rip的配置
- 使用 Docker 部署 OpenResty Manager 搭建可视化反向代理系统
- OpenResty 入门指南:从基础到动态路由实战
- OpenResty 的 Lua 动态能力_openresty 动态upstream
- LVS和Nginx_lvs和nginx的区别
- 外网连接到内网服务器需要端口映射吗,如何操作?
- Nginx如何解决C10K问题(1万个并发连接)?
- 炸场!Spring Boot 9 大内置过滤器实战手册:从坑到神
- WordPress和Typecho xmlrpc漏洞_wordpress主题漏洞
- Linux Shell 入门教程(六):重定向、管道与命令替换
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python进度条 (67)
- python吧 (67)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python列表切片 (59)
- python面向对象编程 (60)
- python 代码加密 (65)
- python串口编程 (77)
- python封装 (57)
- python写入txt (66)
- python读取文件夹下所有文件 (59)
- python操作mysql数据库 (66)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)