Python编程基础:面向对象编程

简介: 巩固Python面向对象编程一些基础知识,学习有关类的创建使用,以及类的属性方法使用等一些基础操作。

一、前言

本期博客,我们将了解学习Python面向对象编程的相关知识,学习如何编写类和使用类等一系列操作。

二、我的环境

  • 电脑系统:Windows 11
  • 语言版本:Python 3.10.4
  • 编译器:VSCode

三、创建和使用类

我们在使用类时几乎可以模拟现实世界中的任何东西,接下来我们创建一个简单的Dog类来表示所有的小狗,并为它定义属性和行为。

1、创建Dog类

classDog:
"""模拟小狗"""def__init__(self, name, age):
"""初始化小狗属性"""self.name=nameself.age=agedefsit(self):
"""模拟小狗蹲下"""print(f"小狗{self.name}已经蹲下了。")
defroll_over(self):
"""模拟小狗打滚"""print(f"小狗{self.name}正在打滚。")

其中__init__()方法是一个特殊的方法,后续每当我们使用该类创建新实例的时候Python会自动运行该方法,在这个方法中形参self必不可少,而且必须位于其他形参的前面,每个与实例相关联的方法调用都会自动传递实参self,它是一个指向实例本身的引用,让实例能够访问类中的属性和方法。

2、根据类创建实例

接下来我们将根据之前创建的类创建一个实例:

my_dog=Dog("哈里", 3)
print(f"我的小狗叫{my_dog.name}。")
print(f"我的小狗今年{my_dog.age}岁了。")
my_dog.sit()
my_dog.roll_over()

它运行的结果是:

我的小狗叫哈里。我的小狗今年3岁了。小狗哈里已经蹲下了。小狗哈里正在打滚。

如果我们要访问类的属性需要使用“实例.属性”格式,例如上面我们访问小狗的名字my_dog.name。

如果我们要调用类的方法需要使用“实例.方法”格式,例如上面我们让小狗蹲下my_dog.sit()。

我们也可以创建多个实例,并且每个实例都有自己的一组属性,能够执行相同的操作:

my_dog=Dog("哈里", 3)
print(f"我的小狗叫{my_dog.name}。")
print(f"我的小狗今年{my_dog.age}岁了。")
my_dog.sit()
your_dog=Dog("卢斯", 4)
print(f"你的小狗叫{your_dog.name}。")
print(f"你的小狗今年{your_dog.age}岁了。")
your_dog.sit()

它运行的结果是:

我的小狗叫哈里。我的小狗今年3岁了。小狗哈里已经蹲下了。你的小狗叫卢斯。你的小狗今年4岁了。小狗卢斯已经蹲下了。

四、使用类和实例

我们还可以使用类来模拟现实世界中的很多情景,当我们的类编写好之后,我们就可以将时间花在根据类创建的实例上。

1、创建Car类

classCar:
"""模拟汽车"""def__init__(self, make, model, year):
"""初始化描述汽车的属性"""self.make=makeself.model=modelself.year=yeardefget_descriptive_name(self):
"""返回整洁的描述信息"""long_name=f"{self.year}{self.make}{self.model}"returnlong_name.title()
my_new_car=Car("audi", 'a4', 2022)
print(my_new_car.get_descriptive_name())

它运行的结果是:

2022AudiA4

2、给属性指定默认值

我们在创建实例时,有些属性无须通过形参来定义,我们可以指定默认值,例如我们添加一个odometer_reading的属性,其初始值为0,然后我们在添加一个read_odometer()方法,用于读取汽车的里程表:

classCar:
"""模拟汽车"""def__init__(self, make, model, year):
"""初始化描述汽车的属性"""self.make=makeself.model=modelself.year=yearself.odometer_reading=0defget_descriptive_name(self):
"""返回整洁的描述信息"""long_name=f"{self.year}{self.make}{self.model}"returnlong_name.title()
defread_odometer(self):
"""打印汽车里程表信息"""print(f"这辆车已经跑了{self.odometer_reading}公里了。")
my_new_car=Car("audi", 'a4', 2022)
print(my_new_car.get_descriptive_name())
my_new_car.read_odometer()

它运行的结果是:

2022AudiA4这辆车已经跑了0公里了。

3、修改属性值

新车的里程为0很正常,但随着开车,里程不可能一直为0,所以我们还需要定义一个可以修改属性的方法,下面介绍三种方法。

  • 直接修改属性的值
    这是最简单的方法在调用方法前指定属性值为多少。
my_new_car=Car("audi", 'a4', 2022)
print(my_new_car.get_descriptive_name())
my_new_car.odometer_reading=23my_new_car.read_odometer()
  • 它运行的结果是:
2022AudiA4这辆车已经跑了23公里了。
  • 这种方法简单,但非常不实用,因为我们可以随便定义里程,这不符合实际情况。
  • 通过方法修改属性的值
    定义一个方法,让其在内部更新,并且为其定义逻辑禁止将里程表读数往回调。
classCar:
--snip--defupdate_odometer(self, mileage):
"""将里程表读数设置为指定的值"""ifmileage>=self.odometer_reading:
self.odometer_reading=mileageelse:
print("你不能输入比之前里程数更低的数!")
--snip--my_new_car=Car("audi", 'a4', 2022)
print(my_new_car.get_descriptive_name())
my_new_car.update_odometer(23)
my_new_car.read_odometer()
  • 结果跟之前的一样,但添加了新逻辑判断。
  • 通过方法对属性的值进行递增
    有时候需要将属性值递增特定的量,而不是将其设置为全新的值。假设我们购买了一辆二手车,且从购买到登记期间增加了100英里的里程。下面的方法让我们能够传递这个增量,并相应地增大里程表读数。
classCar:
--snip--defupdate_odometer(self, mileage):
"""将里程表读数设置为指定的值"""ifmileage>=self.odometer_reading:
self.odometer_reading=mileageelse:
print("你不能输入比之前里程数更低的数!")
defincrement_odometer(self,miles):
self.odometer_reading+=miles--snip--my_used_car=Car("audi", 'a4', 2022)
print(my_used_car.get_descriptive_name())
my_used_car.update_odometer(23_500)
my_used_car.read_odometer()
my_used_car.increment_odometer(100)
my_used_car.read_odometer()
  • 运行的结果是:
2022AudiA4这辆车已经跑了23500公里了。这辆车已经跑了23600公里了。

五、继承

编写类时,并非总是要从空白开始。如果要编写的类是另一个现成类的特殊版本,可使用继承。一个类继承另一个类时,将自动获得另一个类的所有属性和方法。原有的类称为父类,而新类称为子类。子类继承了父类的所有属性和方法,同时还可以定义自己的属性和方法。

1、子类的方法__init__()

在既有类的基础上编写新类时,通常要调用父类的方法__init__(),从而让子类也包含这些属性。

下面我们在之前的Car类下再创建一个ElectricCar类,它具备Car类的所有功能。

classCar:
"""模拟汽车"""def__init__(self, make, model, year):
"""初始化描述汽车的属性"""self.make=makeself.model=modelself.year=yearself.odometer_reading=0defget_descriptive_name(self):
"""返回整洁的描述信息"""long_name=f"{self.year}{self.make}{self.model}"returnlong_name.title()
defupdate_odometer(self, mileage):
"""将里程表读数设置为指定的值"""ifmileage>=self.odometer_reading:
self.odometer_reading=mileageelse:
print("你不能输入比之前里程数更低的数!")
defincrement_odometer(self,miles):
self.odometer_reading+=milesdefread_odometer(self):
"""打印汽车里程表信息"""print(f"这辆车已经跑了{self.odometer_reading}公里了。")
classElectricCar(Car):
""""电动车的独特之处"""def__init__(self, make, model, year):
""""初始化父类的属性"""super().__init__(make, model, year)
my_tesla=ElectricCar('tesla','models', '2022')
print(my_tesla.get_descriptive_name())


它运行的结果是:

2022TeslaModels

在创建子类时,父类必须包含在当前文件中,且位于子类的前面。

子类中的super()是一种特殊的函数,可以让我们能够调用父类的方法。

2、给子类定义属性和方法

让一个类继承另一个类后,就可以添加区分子类和父类所需的新属性和新方法了。

下面我们将给电动车添加新属性以及一个描述该属性的方法。

classElectricCar(Car):
""""电动车的独特之处"""def__init__(self, make, model, year):
""""先初始化父类的属性,再初始化电动车特有属性"""super().__init__(make, model, year)
self.battery_size=75defdescribe_battery(self):
"""打印描述电瓶容量的消息"""print(f"这个电动车电瓶容量为{self.battery_size}kwh。")
my_tesla=ElectricCar('tesla','models', '2022')
print(my_tesla.get_descriptive_name())
my_tesla.describe_battery()

它运行的结果是:

2022TeslaModels这个电动车电瓶容量为75kwh。

模拟电动汽车时,可根据所需的准确程度添加任意数量的属性和方法。如果一个属性或方法是任何汽车都有的,而不是电动汽车特有的,就应将其加入到Car类而非ElectricCar类中。这样,使用Car类的人将获得相应的功能,而ElectricCar类只包含处理电动汽车特有属性和行为的代码。

3、重写父类的方法

对于父类,只要它不符合子类模拟的实物的行为就可以进行重写,我们只需要再子类中定义一个与要重写的父类方法同名的方法即可。

4、将实例用作属性

使用代码模拟实物时,你可能会发现自己给类添加的细节越来越多:属性和方法清单以及文件都越来越长。在这种情况下,可能需要将类的一部分提取出来,作为一个独立的类。可以将大型类拆分成多个协同工作的小类。

例如我们可以专门定义一个Battery类,将针对汽车电瓶的属性和方法提取出来。

classBattery:
"""模拟电动车电瓶"""def__init__(self, battery_size=75):
"""初始化电瓶的属性"""self.battery_size=battery_sizedefdescribe_battery(self):
"""打印描述电瓶容量的消息"""print(f"这个电动车电瓶容量为{self.battery_size}kwh。")
classElectricCar(Car):
""""电动车的独特之处"""def__init__(self, make, model, year):
""""先初始化父类的属性,再初始化电动车特有属性"""super().__init__(make, model, year)
self.battery=Battery()
my_tesla=ElectricCar('tesla','models', '2022')
print(my_tesla.get_descriptive_name())
my_tesla.battery.describe_battery()

它运行的结果是:

2022TeslaModels这个电动车电瓶容量为75kwh。

输出结果和之前一样,我们将电瓶和电动车分开,这样看似很麻烦要做很多额外的工作,但可以让我们更加专注于描述某一类,并且也更好的专门添加功能和修改。

六、最后我想说

本期的Python面向对象编程内容就到此为止了,内容也不是很多,总结的也比较粗略,想要学习更多有关这方面的知识可以多去看看相关书籍或者其他大佬的博客。

有关Python编程基础的内容更新也差不多结束了,后续更新可能就是具体的练习题目或者具体的问题解决。

最后,谢谢大家能阅读完,我也希望能得到大家的支持和肯定,谢谢!

目录
相关文章
|
7天前
|
算法 程序员 开发工具
百万级Python讲师又一力作!Python编程轻松进阶,豆瓣评分8.1
在学习Python的旅程中你是否正在“绝望的沙漠”里徘徊? 学完基础教程的你,是否还在为选择什么学习资料犹豫不决,不知从何入手,提高自己?
百万级Python讲师又一力作!Python编程轻松进阶,豆瓣评分8.1
|
3天前
|
存储 Java C#
详解 Python 中的面向对象编程(2)
详解 Python 中的面向对象编程(2)
20 10
|
5天前
|
算法 程序员 开发工具
百万级Python讲师又一力作!Python编程轻松进阶,豆瓣评分8.1
在学习Python的旅程中你是否正在“绝望的沙漠”里徘徊? 学完基础教程的你,是否还在为选择什么学习资料犹豫不决,不知从何入手,提高自己?
|
3天前
|
数据采集 存储 人工智能
掌握Python编程:从基础到进阶的实用指南
【8月更文挑战第17天】 本文旨在通过浅显易懂的语言和实际案例,为初学者和有一定基础的开发者提供一条清晰的Python学习路径。我们将从Python的基本语法入手,逐步深入到面向对象编程、数据科学应用及网络爬虫开发等高级主题。每个部分都配备了代码示例和实操建议,确保读者能够将理论知识转化为实际能力。无论你是编程新手,还是希望提升Python技能的开发者,这篇文章都将为你打开一扇通往高效编程世界的大门。
7 2
|
8天前
|
Python
python Process 多进程编程
python Process 多进程编程
19 1
|
12天前
|
存储 数据挖掘 程序员
揭秘Python:掌握这些基本语法和数据类型,你将拥有编程世界的钥匙!
【8月更文挑战第8天】Python是一种高级、解释型语言,以简洁的语法和强大的功能广受好评。本文从基本语法入手,强调Python独特的缩进规则,展示清晰的代码结构。接着介绍了Python的主要数据类型,包括数值、字符串、列表、元组、集合和字典,并提供了示例代码。通过这些基础知识的学习,你将为深入探索Python及其在文本处理、数据分析等领域的应用打下坚实的基础。
26 3
|
14天前
|
Python
揭秘!Python系统编程里那些让代码自由穿梭的神奇代码行
【8月更文挑战第6天】在Python编程中,一些简洁有力的代码构造让程序更加灵动高效。列表推导式能一行生成列表,如`squares = [x**2 for x in range(10)]`。`with`语句确保资源自动释放,例`with open('example.txt', 'r') as file:`。`lambda`函数便于快速定义小函数,`map(lambda x: x + 1, numbers)`即可完成列表映射。
28 4
|
14天前
|
API C语言 开发者
Python如何成为跨平台编程的超级巨星:系统调用深度探索
【8月更文挑战第6天】Python凭借简洁的语法和强大的库支持,在编程领域中脱颖而出。其跨平台特性是基于CPython等解释器的设计理念,使得Python程序能在不同操作系统上运行而无需修改代码。Python标准库提供的抽象层隐藏了系统间的差异,加之ctypes等扩展机制,使开发者能高效地编写兼容性强且性能优异的应用。例如,在Windows上利用ctypes调用GetSystemTime系统API获取当前时间,展现了Python深入系统底层的强大能力和灵活性。随着技术演进,Python将继续巩固其作为首选编程语言的地位。
19 3
|
15天前
|
存储 程序员 索引
详解 Python 中的面向对象编程(1)
详解 Python 中的面向对象编程(1)
23 2
|
17天前
|
安全 开发者 Python
跨越编程孤岛,构建互联大陆:深入探索Python进程间通信的奥秘,解锁高效协作新纪元!
【8月更文挑战第3天】在编程领域,Python 因其简洁强大而广受欢迎。但随着项目规模扩大,单进程难以应对复杂需求,此时多进程间的协同就显得尤为重要。各进程像孤岛般独立运行,虽提升了稳定性和并发能力,但也带来了沟通障碍。为解决这一问题,Python 提供了多种进程间通信(IPC)方式,如管道、队列和套接字等,它们能有效促进数据交换和任务协作,使各进程像大陆般紧密相连。通过这些机制,我们能轻松搭建起高效的多进程应用系统,实现更加复杂的业务逻辑。
18 2