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月前
|
Python
python基础篇:面向对象编程
python基础篇:面向对象编程
27 0
|
1月前
|
Python
Python编程作业五:面向对象编程
Python编程作业五:面向对象编程
49 1
|
1月前
|
Python
【Python进阶(三)】——面向对象编程
【Python进阶(三)】——面向对象编程
|
1月前
|
Python
Python中的面向对象编程与继承
本文将深入探讨Python中面向对象编程的核心概念,重点讨论继承的实现原理以及在实际开发中的应用。通过详细的示例和解释,读者将能够全面理解Python中继承的使用方式和优势,为提高代码的复用性和可维护性提供有效的技术支持。
|
16天前
|
Python
别再被Python的面向对象编程绕晕了!一篇文章带你理清思路!
【6月更文挑战第12天】了解Python的面向对象编程(OOP)至关重要,它基于“万物皆对象”的理念,通过类(Class)定义对象的属性和方法。对象是类的实例,具有状态和行为。OOP包括类、对象、属性和方法四个基本概念。例如,`class Dog`定义了一个有`name`和`age`属性及`bark`方法的类。OOP提供代码重用、模块化、可扩展性和易理解性等优势,是现代编程的常用范式。通过学习,你可以更好地掌握这一核心技能。
|
1月前
|
存储 程序员 数据安全/隐私保护
Python面向对象编程:核心概念与实践
Python面向对象编程:核心概念与实践
|
1月前
|
算法 Java 程序员
Python面向对象编程
Python面向对象编程
24 1
|
29天前
|
存储 Java 程序员
Python中的面向对象编程(OOP)详解
Python中的面向对象编程(OOP)详解
20 0
|
1月前
|
数据安全/隐私保护 Python
Python从入门到精通——2.2.1深入学习面向对象编程:类和对象的定义
Python从入门到精通——2.2.1深入学习面向对象编程:类和对象的定义
|
1月前
|
设计模式 算法 程序员
Python从入门到精通:2.1.3深入学习面向对象编程——设计模式的学习与实践
Python从入门到精通:2.1.3深入学习面向对象编程——设计模式的学习与实践

热门文章

最新文章