Python面向对象编程-进阶篇

简介: 面向对象三大特性:封装、继承、多态,面向对象中的变量:类变量、成员变量、局部变量,类中的私有方法和私有属性,类的三类方法:实例方法、类方法、静态方法。

前言

在上一篇《Python面向对象编程-初级篇》中,主要介绍了面向对象相关概念、面向对象相关术语、获取或添加对象属性、魔法方法以及Python的内置属性,本篇内容则继续介绍面向对象进阶部分的内容:

  • 面向对象的三大特性:封装、继承、多态
  • 类中的三类变量:类变量、成员变量、局部变量
  • 类中的私有方法和私有属性
  • 类的三类方法:实例方法、类方法、静态方法

一、面向对象的三大特性:封装、继承、多态

1.封装

封装就是把内容封装到某个地方,后面再从某处调用被封装的内容

函数式编程的封装

defwork(name, age, work):
print(f"我叫{name},我今年{age}岁了,我的工作是{work}")
definterest(name, age, interest):
print(f"我叫{name},我今年{age}岁了,我的爱好是{interest}")
defcity(name, age, city):
print(f"我叫{name},我今年{age}岁了,我的家乡是{city}")
work('小明', 28, '司机')
interest('小明', 28, '滑雪')
city('小明', 28, '北京')

面向对象编程的封装

classIntroduction(object):
def__init__(self, name, age, city, work, interest)
self.name=nameself.age=ageself.city=cityself.work=workself.interest=interestdefintro(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)子类继承父类

如果在子类中需要父类的构造方法就需要显式地调用父类的构造方法,或者不重写父类的构造方法

classParentObject(object):
def__init__(self, height):
self.name="当当"self.age=5defparent_func(self):
print("这是父类中的方法")
classChildObject(ParentObject):
defchild_func(self):
print("这是子类中的方法")
child=ChildObject()
child.parent_func()  # 这是父类中的方法child.child_func()  # 这是子类中的方法print(ChildObject.__bases__)  # (<class '__main__.ParentObject'>,)

2)子类继承父类中的构造方法

如下案例:子类ChildObject继承了父类ParentObject,如果想要在子类的构造方法中继承父类构造方法中的属性,可以有以下几种写法:

  1. ParentObject.__init__(self,height='115cm')
  2. super().__init__(height='115cm')
  3. super(ChildObject, self).__init__(height='115cm')
classParentObject(object):
def__init__(self, height):
self.name="当当"self.age=5self.height=heightdefparent_func(self):
print("这是父类中的方法")
classChildObject(ParentObject):
def__init__(self):
# ParentObject.__init__(self,height='115cm')  # 子类继承父类的构造方法,写法一super().__init__(height='115cm')  # 子类继承父类的构造方法,写法二# super(ChildObject, self).__init__('115cm')  # 子类继承父类的构造方法,写法三

3)类的多继承:深度优先和广度优先

Python中一个子类可以继承多个父类,寻找方法有两种,分别是:深度优先(Python2)和广度优先(Python3)

在Python2中,经典类遵循的是深度优先的原则,新式类遵循的是广度优先的原则;而在Python3中,无论是经典类还是新式类,都遵循广度优先

classA(object):
name="Asia"def__init__(self):
print("class A")
classB(A):
def__init__(self):
print("class B")
classC(A):
def__init__(self):
print("class C")
classD(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)子类重写父类方法

在子类中,使用与父类中相同的变量名或方法名,或重写父类的属性或方法

classParent:
def__init__(self):
self.name='Lucy'deffun_a(self):
print("this is a function in class Parent")
classSon(Parent):
def__init__(self):
super().__init__()
self.name='Tom'# 子类重写父类属性deffun_a(self):  # 子类重写父类方法print("this is a function in class Son")
son=Son()
print(son.name)  # Tomson.fun_a()  # this is a function in class Son

3.多态

不同的子类对象,调用相同的父类方法,产生不同的结果,一种事物的多种体现形式,函数的重写其实就是多态的一种体现

classAnimals(object):
deftalk(self):
print("animals")
classPerson(Animals):
deftalk(self):
print("person")
classCat(Animals):
deftalk(self):
print("cat")
classDog(Animals):
deftalk(self):
print("dog")
Person().talk()  # personCat().talk()  # catDog().talk()  # dog

如上图所示,Person、Dog、Cat分别继承了Animals类,但是分别重写了talk方法,当这三个类分别被调用时会执行自己类中所定义的talk方法,而非父类Animals中的talk方法

二、类变量、成员变量、局部变量

1.类变量

类变量在整个实例化的对象中是公用的。类变量定义在类中且在函数体之外。类变量通常不作为实例变量使用。可以由类名直接调用,也可由对象来调用。

# 类变量classA:
name='Tony'deffun_a(self):
print('this is a test function in class A')
print(A.name)  # Tonyprint(A().name)  # Tony

2.实例变量(成员变量)

在类的声明中,属性是用变量来表示的。这种变量就称为实例变量,是在类声明的内部但是在类的其他成员方法之外声明的,在构造方法中以self. 开头来定义。实例变量只能通过对象来调用,不能通过类名调用。

# 实例变量(成员变量)classB:
def__init__(self):
self.city='suzhou'# 实例变量self.street='松涛街'# 实例变量# 在构造方法中提前声明了一个方法,这个方法中所包含的变量也属于成员变量self.vars()  
defvars(self):
self.home="月亮湾壹号"self.house="1幢一单元108"print(B().city)  # suzhouprint(B().street)  # 松涛街print(B().__dict__)  # {'city': 'suzhou', 'street': '松涛街', 'home': '月亮湾壹号', 'house': '1幢一单元108'}

3.局部变量

定义在方法中的变量,只作用于当前实例的类。如以下方法中的mobile就属于局部变量。

definfo(self):
self.number=227# 局部变量self.phone=15252162666# 局部变量

三、类中私有方法和私有属性

1.类的私有属性

__private_attrs:两个下划线开头,声明该属性为私有,不能在类的外部被使用或直接访问。在类内部的方法中使用时 self.__private_attrs

# 私有属性classParentObject(object):
mobile=15252162666# 类变量__private_mobile=15252162666# 私有变量deffun(self):
print("打印私有变量{}".format(self.__private_mobile))
deffunc(self):
returnself.__fun()
po=ParentObject()
po.fun()  # 打印私有变量15252162666

2.类的私有方法

__private_method:两个下划线开头,声明该方法为私有方法,不能在类的外部调用。在类的内部调用 self.__private_methods

def__fun(self):
print("这是一个私有化方法")
deffunc(self):
returnself.__fun()

3.外部调用类中的私有属性或方法

通常私有属性和私有方法只能在类的内部被调用,外部是不可以调用的。但如果强行调用,也是可以的,相当于Python中开了个后门:

外部调用类的私有属性:对象名._类名__属性名

外部调用类的私有方法:对象名._类名__方法名

# 外部调用类的私有属性:对象名._类名__属性名print(po._ParentObject__private_mobile)
# 外部调用类的私有方法:对象名._类名__方法名po._ParentObject__fun()

四、类的三类方法:实例方法、类方法、静态方法

1.实例方法

第一个参数必须是实例对象,该参数一般约定为self,通过它来传递实例的属性和方法(也可以传类的属性和方法),只能由实例对象调用

# 实例方法classExample:
deffun_a(self):
print("这是一个实例方法")
# 调用实例方法,只能由实例对象调用ex=Example()
ex.fun_a()  # 这是一个实例方法

2.类方法

使用装饰器@classmethod,第一个参数必须是当前类对象,该参数名一般约定为“cls”,通过它来传递类的属性和方法(不能传实例的属性和方法),类和实例对象都可以调用

@classmethoddeffun_b(self):
print("这是一个类方法")
# 调用类方法,实例对象和类名都可以调用,使用类名直接调用时,不会执行类中的构造方法ex.fun_b()  # 这是一个类方法Example.fun_b()  # 这是一个类方法

注:使用类名直接调用时,不会执行类中的构造方法

3.静态方法

使用装饰器@staticmethod修饰,参数随意,没有“self”和“cls”参数,但是方法体中不能使用类或实例的任何属性和方法,一般用于和类本身无太多关联但又会绑定在类中的场景(不可使用类中的属性和方法),如获取时间等等。类和实例对象都可以调用

@staticmethoddeffun_c():
print("这是一个静态方法")
# 调用静态方法,实例对象和类名都可以调用,不能使用类或实例的任何属性和方法ex.fun_c()  # 这是一个静态方法Example.fun_c()  # 这是一个静态方法

小结

类别

调用方式

注意事项

类变量

实例对象和类都可以调用


实例变量(成员变量)

只能通过实例对象调用


局部变量

只能在方法内部调用


类的私有属性:__private_attrs

只能在类的内部调用:self.__private_attrs

外部强行调用类的私有属性:对象名._类名__属性名

类的私有方法:__private_method

只能在类的内部调用:self.__private_methods

外部强行调用类的私有方法:对象名._类名__方法名

实例方法

只能由实例对象调用


静态方法:@staticmethod

相关文章
|
1月前
|
Java C# Python
Python学习七:面向对象编程(中)
这篇文章是关于Python面向对象编程的中级教程,涵盖了析构函数、对象的三大特征(封装、继承、多态)、类属性与实例属性、以及类方法与静态方法的对比。
22 2
|
1月前
|
设计模式 安全 JavaScript
Python学习八:面向对象编程(下):异常、私有等
这篇文章详细介绍了Python面向对象编程中的私有属性、私有方法、异常处理及动态添加属性和方法等关键概念。
20 1
|
6月前
|
Python
Python编程作业五:面向对象编程
Python编程作业五:面向对象编程
71 1
|
2月前
|
存储 Java 程序员
30天拿下Python之面向对象编程
30天拿下Python之面向对象编程
17 3
|
2月前
|
Java Python
全网最适合入门的面向对象编程教程:50 Python函数方法与接口-接口和抽象基类
【9月更文挑战第18天】在 Python 中,虽无明确的 `interface` 关键字,但可通过约定实现类似功能。接口主要规定了需实现的方法,不提供具体实现。抽象基类(ABC)则通过 `@abstractmethod` 装饰器定义抽象方法,子类必须实现这些方法。使用抽象基类可使继承结构更清晰、规范,并确保子类遵循指定的方法实现。然而,其使用应根据实际需求决定,避免过度设计导致代码复杂。
|
1月前
|
Java Python
Python学习六:面向对象编程(上)
这篇文章是关于Python面向对象编程的基础知识,包括类和对象的概念、实例方法、属性、self关键字以及魔法方法等。
16 0
|
2月前
|
Python
全网最适合入门的面向对象编程教程:Python函数方法与接口-函数与方法的区别和lamda匿名函数
【9月更文挑战第15天】在 Python 中,函数与方法有所区别:函数是独立的代码块,可通过函数名直接调用,不依赖特定类或对象;方法则是与类或对象关联的函数,通常在类内部定义并通过对象调用。Lambda 函数是一种简洁的匿名函数定义方式,常用于简单的操作或作为其他函数的参数。根据需求,可选择使用函数、方法或 lambda 函数来实现代码逻辑。
|
3月前
|
Python
Python 中的面向对象编程 (OOP)
【8月更文挑战第29天】
34 7
|
3月前
|
机器学习/深度学习 PHP 开发者
探索PHP中的面向对象编程构建你的首个机器学习模型:以Python和scikit-learn为例
【8月更文挑战第30天】在PHP的世界中,面向对象编程(OOP)是一块基石,它让代码更加模块化、易于管理和维护。本文将深入探讨PHP中面向对象的魔法,从类和对象的定义开始,到继承、多态性、封装等核心概念,再到实战中如何应用这些理念来构建更健壮的应用。我们将通过示例代码,一起见证PHP中OOP的魔力,并理解其背后的设计哲学。
|
3月前
|
Python