Python面向对象编程-进阶篇(python面向对象教程)
off999 2025-07-03 18:48 20 浏览 0 评论
前言
在上一篇《Python面向对象编程-初级篇》中,主要介绍了面向对象相关概念、面向对象相关术语、获取或添加对象属性、魔法方法以及Python的内置属性,本篇内容则继续介绍面向对象进阶部分的内容:
- 面向对象的三大特性:封装、继承、多态
- 类中的三类变量:类变量、成员变量、局部变量
- 类中的私有方法和私有属性
- 类的三类方法:实例方法、类方法、静态方法
一、面向对象的三大特性:封装、继承、多态
1.封装
封装就是把内容封装到某个地方,后面再从某处调用被封装的内容
函数式编程的封装
def work(name, age, work):
print(f"我叫{name},我今年{age}岁了,我的工作是{work}")
def interest(name, age, interest):
print(f"我叫{name},我今年{age}岁了,我的爱好是{interest}")
def city(name, age, city):
print(f"我叫{name},我今年{age}岁了,我的家乡是{city}")
work('小明', 28, '司机')
interest('小明', 28, '滑雪')
city('小明', 28, '北京')
面向对象编程的封装
class Introduction(object):
def __init__(self, name, age, city, work, interest)
self.name = name
self.age = age
self.city = city
self.work = work
self.interest = interest
def intro(self):
print(f"我叫{self.name},我今年{self.age}岁了,我的家乡是{self.city},工作是{self.work},爱好是{self.interest}")
intro1 = Introduction("小明", 28, "北京", "司机", "滑雪")
intro1.intro()
intro2 = Introduction("小华", 22, "上海", "学生", "篮球")
intro2.intro()
上述对比可以看出,如果使用函数式编程,需要在每次执行函数时传入相同的参数,如果参数较多,则需要多次复制粘贴;而对于面向对象,只需要在创建对象时,将所有需要的参数封装到当前对象中,之后再次使用时,通过self间接去当前对象中取值即可。使用面向对象的思想可以更好地模拟现实生活中的事物。
2.继承
通过继承创建的类称为子类或派生类,被继承的类称为基类、父类或超类,子类可以继承父类的内容,调用父类中的属性或方法。
1)子类继承父类
如果在子类中需要父类的构造方法就需要显式地调用父类的构造方法,或者不重写父类的构造方法
class ParentObject(object):
def __init__(self, height):
self.name = "当当"
self.age = 5
def parent_func(self):
print("这是父类中的方法")
class ChildObject(ParentObject):
def child_func(self):
print("这是子类中的方法")
child = ChildObject()
child.parent_func() # 这是父类中的方法
child.child_func() # 这是子类中的方法
print(ChildObject.__bases__) # (<class '__main__.ParentObject'>,)
2)子类继承父类中的构造方法
如下案例:子类ChildObject继承了父类ParentObject,如果想要在子类的构造方法中继承父类构造方法中的属性,可以有以下几种写法:
- ParentObject.__init__(self,height='115cm')
- super().__init__(height='115cm')
- super(ChildObject, self).__init__(height='115cm')
class ParentObject(object):
def __init__(self, height):
self.name = "当当"
self.age = 5
self.height = height
def parent_func(self):
print("这是父类中的方法")
class ChildObject(ParentObject):
def __init__(self):
# ParentObject.__init__(self,height='115cm') # 子类继承父类的构造方法,写法一
super().__init__(height='115cm') # 子类继承父类的构造方法,写法二
# super(ChildObject, self).__init__('115cm') # 子类继承父类的构造方法,写法三
3)类的多继承:深度优先和广度优先
Python中一个子类可以继承多个父类,寻找方法有两种,分别是:深度优先(Python2)和广度优先(Python3)
在Python2中,经典类遵循的是深度优先的原则,新式类遵循的是广度优先的原则;而在Python3中,无论是经典类还是新式类,都遵循广度优先
class A(object):
name = "Asia"
def __init__(self):
print("class A")
class B(A):
def __init__(self):
print("class B")
class C(A):
def __init__(self):
print("class C")
class D(B, C):
# D类继承了B、C,会先从B类开始查找指定属性,B不存在时再从C开始查找,C也不存在时再从B的父类查找
def __init__(self):
print("class D")
obj = D()
print(obj.name) # Asia
由于D类继承了B、C,会先从B类开始查找name属性,B不存在,所以会再从C开始查找,由于C也不存在,所以会再从B的父类开始查找,最后在B的父类A中找到了name属性,打印结果为Asia。
4)子类重写父类方法
在子类中,使用与父类中相同的变量名或方法名,或重写父类的属性或方法
class Parent:
def __init__(self):
self.name = 'Lucy'
def fun_a(self):
print("this is a function in class Parent")
class Son(Parent):
def __init__(self):
super().__init__()
self.name = 'Tom' # 子类重写父类属性
def fun_a(self): # 子类重写父类方法
print("this is a function in class Son")
son = Son()
print(son.name) # Tom
son.fun_a() # this is a function in class Son
3.多态
不同的子类对象,调用相同的父类方法,产生不同的结果,一种事物的多种体现形式,函数的重写其实就是多态的一种体现
class Animals(object):
def talk(self):
print("animals")
class Person(Animals):
def talk(self):
print("person")
class Cat(Animals):
def talk(self):
print("cat")
class Dog(Animals):
def talk(self):
print("dog")
Person().talk() # person
Cat().talk() # cat
Dog().talk() # dog
如上图所示,Person、Dog、Cat分别继承了Animals类,但是分别重写了talk方法,当这三个类分别被调用时会执行自己类中所定义的talk方法,而非父类Animals中的talk方法
二、类变量、成员变量、局部变量
1.类变量
类变量在整个实例化的对象中是公用的。类变量定义在类中且在函数体之外。类变量通常不作为实例变量使用。可以由类名直接调用,也可由对象来调用。
# 类变量
class A:
name = 'Tony'
def fun_a(self):
print('this is a test function in class A')
print(A.name) # Tony
print(A().name) # Tony
2.实例变量(成员变量)
在类的声明中,属性是用变量来表示的。这种变量就称为实例变量,是在类声明的内部但是在类的其他成员方法之外声明的,在构造方法中以self. 开头来定义。实例变量只能通过对象来调用,不能通过类名调用。
# 实例变量(成员变量)
class B:
def __init__(self):
self.city = 'suzhou' # 实例变量
self.street = '松涛街' # 实例变量
# 在构造方法中提前声明了一个方法,这个方法中所包含的变量也属于成员变量
self.vars()
def vars(self):
self.home = "月亮湾壹号"
self.house = "1幢一单元108"
print(B().city) # suzhou
print(B().street) # 松涛街
print(B().__dict__) # {'city': 'suzhou', 'street': '松涛街', 'home': '月亮湾壹号', 'house': '1幢一单元108'}
3.局部变量
定义在方法中的变量,只作用于当前实例的类。如以下方法中的mobile就属于局部变量。
def info(self):
self.number = 227 # 局部变量
self.phone = 15252162666 # 局部变量
三、类中私有方法和私有属性
1.类的私有属性
__private_attrs:两个下划线开头,声明该属性为私有,不能在类的外部被使用或直接访问。在类内部的方法中使用时 self.__private_attrs
# 私有属性
class ParentObject(object):
mobile = 15252162666 # 类变量
__private_mobile = 15252162666 # 私有变量
def fun(self):
print("打印私有变量{}".format(self.__private_mobile))
def func(self):
return self.__fun()
po = ParentObject()
po.fun() # 打印私有变量15252162666
2.类的私有方法
__private_method:两个下划线开头,声明该方法为私有方法,不能在类的外部调用。在类的内部调用 self.__private_methods
def __fun(self):
print("这是一个私有化方法")
def func(self):
return self.__fun()
3.外部调用类中的私有属性或方法
通常私有属性和私有方法只能在类的内部被调用,外部是不可以调用的。但如果强行调用,也是可以的,相当于Python中开了个后门:
外部调用类的私有属性:对象名._类名__属性名
外部调用类的私有方法:对象名._类名__方法名
# 外部调用类的私有属性:对象名._类名__属性名
print(po._ParentObject__private_mobile)
# 外部调用类的私有方法:对象名._类名__方法名
po._ParentObject__fun()
四、类的三类方法:实例方法、类方法、静态方法
1.实例方法
第一个参数必须是实例对象,该参数一般约定为self,通过它来传递实例的属性和方法(也可以传类的属性和方法),只能由实例对象调用
# 实例方法
class Example:
def fun_a(self):
print("这是一个实例方法")
# 调用实例方法,只能由实例对象调用
ex = Example()
ex.fun_a() # 这是一个实例方法
2.类方法
使用装饰器@classmethod,第一个参数必须是当前类对象,该参数名一般约定为“cls”,通过它来传递类的属性和方法(不能传实例的属性和方法),类和实例对象都可以调用
@classmethod
def fun_b(self):
print("这是一个类方法")
# 调用类方法,实例对象和类名都可以调用,使用类名直接调用时,不会执行类中的构造方法
ex.fun_b() # 这是一个类方法
Example.fun_b() # 这是一个类方法
注:使用类名直接调用时,不会执行类中的构造方法
3.静态方法
使用装饰器@staticmethod修饰,参数随意,没有“self”和“cls”参数,但是方法体中不能使用类或实例的任何属性和方法,一般用于和类本身无太多关联但又会绑定在类中的场景(不可使用类中的属性和方法),如获取时间等等。类和实例对象都可以调用
@staticmethod
def fun_c():
print("这是一个静态方法")
# 调用静态方法,实例对象和类名都可以调用,不能使用类或实例的任何属性和方法
ex.fun_c() # 这是一个静态方法
Example.fun_c() # 这是一个静态方法
小结
类别 | 调用方式 | 注意事项 |
类变量 | 实例对象和类都可以调用 | |
实例变量(成员变量) | 只能通过实例对象调用 | |
局部变量 | 只能在方法内部调用 | |
类的私有属性:__private_attrs | 只能在类的内部调用:self.__private_attrs | 外部强行调用类的私有属性:对象名._类名__属性名 |
类的私有方法:__private_method | 只能在类的内部调用:self.__private_methods | 外部强行调用类的私有方法:对象名._类名__方法名 |
实例方法 | 只能由实例对象调用 | |
静态方法:@staticmethod | 实例对象和类都可以调用 | |
类方法:@classmethod | 实例对象和类都可以调用 | 使用类名调用时,不会执行类中的构造方法 |
相关推荐
- 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)