5 Python的面向对象编程

概述

        在上一节,我们介绍了Python的函数,包括:函数的定义、函数的调用、参数的传递、lambda函数等内容。在本节中,我们将介绍Python的面向对象编程。面向对象编程(Object-Oriented Programming, 即OOP)是一种编程范型,它以对象为基础,将数据和操作封装在一个类(Class)中。在Python中,类是一种定义对象结构和行为的模板,而对象则是类的实例。类定义了一个新的类型,用于创建具有特定属性和方法的对象。类是面向对象编程的核心,它允许程序员使用对象来组织代码和复用代码。

类的定义

在Python中,类的基本语法如下:

class ClassName:# class attributeclass_variable = "class variable"def __init__(self, arg1, arg2):# instance variableself.instance_variable = arg1 + arg2# instance methoddef instance_method(self):print("I am an instance method")# class method@classmethoddef class_method(cls):print("I am a class method")# static method@staticmethoddef static_method():print("I am a static method")

        下面逐一介绍上面示例代码中的各个元素。

        class ClassName:这是类定义的开始,以class关键字作为开头,ClassName是要定义的类的名称;最后面是冒号,冒号后面的内容需要缩进。

        class_variable = "class variable":这是类变量,它是一个在类中定义的全局变量,所有实例共享同一个变量。

        def __init__(self, arg1, arg2):这是类的构造函数,当一个类实例被创建时会自动调用。在这个例子中,构造函数接受两个参数:arg1和arg2。self是对当前实例的引用,调用时不需要写,由系统自动填入。构造函数可以不带参数,也可以带一个或多个参数。

        self.instance_variable = arg1 + arg2:这是一个实例变量,每个实例都有自己独立的实例变量。在这个例子中,实例变量是arg1和arg2的和。

        def instance_method(self):这是一个实例方法,它需要一个实例作为其第一个参数(通常命名为self,也可以使用其他名称)。self是对当前实例的引用,调用时不需要写,由系统自动填入。

        @classmethod:这是一个类方法装饰器,标识后面是一个类方法。它不需要实例作为其第一个参数,而是使用类名本身作为第一个参数(通常命名为cls,也可以使用其他名称)。cls是对当前类的引用,调用时不需要写,由系统自动填入。

        @staticmethod:这是一个静态方法装饰器,标识后面是一个静态方法。它不需要实例或类作为其参数。

类的使用

        定义好类之后,我们就可以实例化该类的对象,并调用其属性和方法了。

class Person:def __init__(self, name, age):self.name = nameself.age = agedef show_info(self):print(f'name is {self.name}, {self.age} years old')@staticmethoddef show_skill():print('walk, run, swim')# 创建一个Person类的实例
person1 = Person('xiaoxiao', 20)
# 访问实例变量,输出:xiaoxiao 20
print(person1.name, person1.age)
# 访问实例方法,输出:name is xiaoxiao, 20 years old
person1.show_info()
# 访问静态方法,输出:walk, run, swim
person1.show_skill()
Person.show_skill()

        可以看到,访问实例变量和实例方法时,必须使用实例对象。访问静态方法时,既可以使用实例对象,也可以使用类名。

        类的__init__方法是类的构造函数,当一个类实例被创建时会自动调用。同样的,类的__del__方法是类的析构函数,在释放实例对象时,也会被自动调用。

class Animal:def __init__(self, name):self.name = nameprint(f'animal {name} constructed')def __del__(self):print(f'animal {self.name} destructed')def show(self):print(f'{self.name} is preset')def test():animal = Animal('seagull')animal.show()''' test作用域结束时,animal会销毁,故依次输出:
animal seagull constructed
seagull is preset
animal seagull destructed
'''
test()

类变量和实例变量

        类变量是在类的所有实例中共享的变量,这就意味着,如果你改变了一个类变量的值,那么这个改变将影响到类的所有实例。类变量在类定义时声明,通常在类的方法中使用。

class MyClass:  # 类变量  class_variable = 0  def __init__(self, instance_variable):  self.instance_variable = instance_variable  MyClass.class_variable = self.instance_variable + 100  instance1 = MyClass(66)
# 输出:166
print(MyClass.class_variable)instance2 = MyClass(88)
# 均输出:188
print(MyClass.class_variable)
print(instance1.class_variable)
print(instance2.class_variable)

        在上面的示例代码中,class_variable是一个类变量,它被所有实例共享。每次创建一个新的实例时,都会改变class_variable的值。因此,当创建了instance2并给它一个值为88的参数时,class_variable的值就从166变成了188。

        实例变量是在类的每个实例中单独存储的变量,这就意味着,如果你改变了一个实例的状态,那么这个改变只影响那个特定的实例。实例变量在类的__init__方法中声明,通常在实例的方法中使用。

class MyClass:def __init__(self, instance_variable):# 实例变量self.instance_variable = instance_variable  instance1 = MyClass(66)
# 输出:66
print(instance1.instance_variable)# 创建实例2
instance2 = MyClass(88)
# 输出:88
print(instance2.instance_variable)
# 输出:66
print(instance1.instance_variable)# 改变实例1的实例变量值
instance1.instance_variable = 100
# 输出:100
print(instance1.instance_variable)
# 输出:88
print(instance2.instance_variable)

        在上面的示例代码中,instance_variable是一个实例变量,每个实例都有它自己的副本。当你改变instance1的 instance_variable值时,instance2的值不会受到影响。

        在Python中,两个下划线开头的属性被声明为私有属性,不能在类的外部被使用或直接访问。否则,运行时会报AttributeError的错误信息。

class MyNumber:def __init__(self, value):self.__value = valuenum = MyNumber(66)
# 访问私有属性,运行时报错:'MyNumber' object has no attribute '__value'
print(num.__value)

实例方法、类方法和静态方法

        实例方法是定义在类中的普通函数,它需要一个实例作为第一个参数(通常命名为self)。实例方法只能通过类的实例来调用,可以访问类变量和实例变量。

        类方法是定义在类中的普通函数,它需要一个类作为第一个参数(通常命名为cls)。类方法可以通过类和类的实例来调用,只能访问类变量,不能访问实例变量。在定义类方法时,可以使用@classmethod装饰器进行声明。

        静态方法是定义在类中的普通函数,它不需要任何参数(包括 self)。静态方法可以通过类和类的实例来调用,不可以访问类变量和实例变量。在定义静态方法时,可以使用@staticmethod装饰器进行声明。

class Person:# 类变量skills: ['walk', 'run', 'swim']def __init__(self, name, age):# 示例变量self.name = nameself.age = age# 实例方法,可以访问类变量和实例变量def show_info(self):print(f'name is {self.name}, {self.age} years old')print(self.skills)# 类方法,可以访问类变量@classmethoddef show_skill(cls):print(cls.skills)# 静态方法,不可以访问类变量和实例变量@staticmethoddef show_basic():print('a person here')

        在Python中,两个下划线开头的方法被声明为私有方法,不能在类的外部被使用或直接访问。否则,运行时会报AttributeError的错误信息。

class MyNumber:def __init__(self, value):self.__value = valuedef __show(self):print(self.__value)num = MyNumber(66)
# 访问私有属性,运行时报错:'MyNumber' object has no attribute '__show'
num.__show()

        与C++、Java等语言不同,Python不支持函数重载。当类的定义中有多个同名的函数时,将以最后一个声明的函数为准。

class MyNumber:def __init__(self, value):self.__value = valuedef show(self):print(self.__value)def show(self, a):print(self.__value, a)num = MyNumber(100)
num.show(66)
# 运行时报错:MyNumber.show() missing 1 required positional argument: 'a'
num.show()

类的运算符重载

        可以通过定义特定方法来重载类对象的运算符,以下是一些常见的运算符重载方法。

        __add__(self, other):重载加法运算符 +,用于实现两个对象的相加。

        __sub__(self, other):重载减法运算符 -,用于实现两个对象的相减。

        __mul__(self, other):重载乘法运算符 *,用于实现两个对象的相乘。

        __truediv__(self, other):重载除法运算符 /,用于实现两个对象的相除。

        __floordiv__(self, other):重载整数除法运算符 //,用于实现两个对象的整数相除。

        __mod__(self, other):重载取模运算符 %,用于实现两个对象的取模运算。

        __pow__(self, other):重载幂运算运算符 **,用于实现两个对象的幂运算。

        __eq__(self, other):重载相等运算符 ==,用于判断两个对象是否相等。

        __ne__(self, other):重载不等运算符 !=,用于判断两个对象是否不相等。

        __lt__(self, other):重载小于运算符 <,用于判断两个对象是否小于。

        __le__(self, other):重载小于等于运算符 <=,用于判断两个对象是否小于等于。

        __gt__(self, other):重载大于运算符 >,用于判断两个对象是否大于。

        __ge__(self, other):重载大于等于运算符 >=,用于判断两个对象是否大于等于。

        假如我们定义了一个名为MyNumber的类,可以按照以下方式重载加法运算符。

class MyNumber:def __init__(self, value):self.value = valuedef __add__(self, other):if isinstance(other, MyNumber):return MyNumber(self.value + other.value)return MyNumber(self.value + other)num1 = MyNumber(100)
num2 = MyNumber(200)
num = num1 + num2
# 输出:300
print(num.value)

继承

        继承是一种实现面向对象编程的重要机制,它允许我们基于已有的类创建新的类,从而继承已有类的属性和方法。在Python中,使用class语句定义一个类时,可以在类名后面使用(base_classes)的形式指定该类要继承的父类。base_classes可以为一个类,也可以为多个类。多个类时,各个类之间用逗号进行分隔,属于多重继承的内容。

class Animal:def __init__(self, name):self.name = namedef eat(self):print(self.name + " is eating...")class Dog(Animal):def __init__(self, name, breed):super().__init__(name)self.breed = breeddog = Dog('Sky', 'Corgi')
# 输出:Sky is eating...
dog.eat()
# 输出:Corgi
print(dog.breed)

        在上面的示例代码中,Dog类继承了Animal类,因此Dog类具有了Animal类的属性和方法。在Dog类的定义中,我们可以通过调用super().__init__(name)来调用父类的构造函数,从而初始化Dog类实例的name属性。

        除了继承父类的属性和方法,子类还可以重写父类的方法,从而实现对父类行为的修改。

class Animal:def __init__(self, name):self.name = namedef eat(self):print(self.name + " is eating...")class Dog(Animal):def __init__(self, name, breed):super().__init__(name)self.breed = breeddef eat(self):print(self.breed + " eats more")dog = Dog('Sky', 'Corgi')
# 覆盖父类方法,输出:Corgi eats more
dog.eat()
# 强制调用父类方法,输出:Sky is eating...
super(Dog, dog).eat()

        在上面的示例代码中,我们直接在Dog类中定义了一个与父类同名的eat方法,从而完全覆盖了父类的行为。如果想强制调用父类的方法,可以使用类似super(Dog, dog)的方式获得父类的实例,再调用父类的方法。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/49118.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

系统架构设计师之缓存技术:Redis与Memcache能力比较

系统架构设计师之缓存技术&#xff1a;Redis与Memcache能力比较

OpenCV使用CMake和MinGW-w64的编译安装

OpenCV使用CMake和MinGW-w64的编译安装中的问题 问题&#xff1a;gcc: error: long: No such file or directory** C:\PROGRA~2\Dev-Cpp\MinGW64\bin\windres.exe: preprocessing failed. modules\core\CMakeFiles\opencv_core.dir\build.make:1420: recipe for target ‘modul…

Git如何操作本地分支仓库?

基本使用TortoiseGit 操作本地仓库(分支) 分支的概念 几乎所有的版本控制系统都以某种形式支持分支。 使用分支意味着你可以把你的工作从开发主线上分离开来&#xff0c;避免影响开发主线。多线程开发,可以同时开启多个任务的开发&#xff0c;多个任务之间互不影响。 为何要…

无涯教程-PHP - 全局变量函数

全局变量 与局部变量相反,可以在程序的任何部分访问全局变量。通过将关键字 GLOBAL 放置在应被识别为全局变量的前面,可以很方便地实现这一目标。 <?php$somevar15;function addit() {GLOBAL $somevar;$somevar;print "Somevar is $somevar";}addit(); ?> …

Cohere API 之旅

Cohere API之旅 0. 背景1. Cohere提供的模型和函数2. Cohere 客户端3. Co.Generate API4. Co.Embed API5. Co.Classify API6. Co.Tokenize API7. Co.Detokenize API8. Co.Detect_language9. Co.Summarize API10. Co.Rerank API 0. 背景 大约两个月前有一份新闻稿。 https://w…

合并两个有序数组【leetcode】

1. 题目&#xff1a; 给你两个按 非递减顺序 排列的整数数组 nums1 和 nums2&#xff0c;另有两个整数 m 和 n &#xff0c;分别表示 nums1 和 nums2 中的元素数目。 请你 合并 nums2 到 nums1 中&#xff0c;使合并后的数组同样按 非递减顺序 排列。 注意&#xff1a;最终&…

测试框架pytest教程(9)跳过测试skip和xfail

skip无条件跳过 使用装饰器 pytest.mark.skip(reason"no way of currently testing this") def test_example(faker):print("nihao")print(faker.words()) 方法内部调用 满足条件时跳过 def test_example():a1if a>0:pytest.skip("unsupported …

基于Redis的BitMap实现签到、连续签到统计(含源码)

微信公众号访问地址&#xff1a;基于Redis的BitMap实现签到、连续签到统计(含源码) 推荐文章&#xff1a; 1、springBoot对接kafka,批量、并发、异步获取消息,并动态、批量插入库表; 2、SpringBoot用线程池ThreadPoolTaskExecutor异步处理百万级数据; 3、基于Redis的Geo实现附…

python小脚本——批量将PDF文件转换成图片

语言&#xff1a;python 3 用法&#xff1a;选择PDF文件所在的目录&#xff0c;点击 确定 后&#xff0c;自动将该目录下的所有PDF转换成单个图片&#xff0c;图片名称为: pdf文件名.page_序号.jpg 如运行中报错&#xff0c;需要自行根据报错内容按照缺失的库 例如&#x…

ORB-SLAM系列算法演进

ORB-SLAM算法是特征点法的代表&#xff0c;当前最新发展的ORB-SLAM3已经将相机模型抽象化&#xff0c;适用范围非常广&#xff0c;虽然ORB-SLAM在算法上的创新并不是很丰富&#xff0c;但是它在工程上的创新确实让人耳目一新&#xff0c;也能更好的为AR、机器人的算法实现落地。…

CF 514 C Watto and Mechanism(双哈希)

CF 514 C. Watto and Mechanism(双哈希) Problem - 514C - Codeforces 大意&#xff1a;给出 n 个字典串 &#xff0c;现给出 m 个查询串 &#xff0c; 问是否存在字典串与当前查询串仅差一个字母。每个串仅包含 {‘a’ , ‘b’ , ‘c’ } 三个字母 。 思路&#xff1a;考虑…

Linux编程:在程序中异步的调用其他程序

Linux编程:execv在程序中同步调用其他程序_风静如云的博客-CSDN博客 介绍了同步的调用其他程序的方法。 有的时候我们需要异步的调用其他程序,也就是不用等待其他程序的执行结果,尤其是如果其他程序是作为守护进程运行的,也无法等待其运行的结果。 //ssss程序 #include …

操作教程|通过1Panel开源Linux面板快速安装DataEase

DataEase开源数据可视化分析工具&#xff08;dataease.io&#xff09;的在线安装是通过在服务器命令行执行Linux命令来进行的。但是在实际的安装部署过程中&#xff0c;很多数据分析师或者业务人员经常会因为不熟悉Linux操作系统及命令行操作方式&#xff0c;在安装DataEase的过…

LeetCode42.接雨水

这道题呢可以按列来累加&#xff0c;就是先算第1列的水的高度然后再加上第2列水的高度……一直加到最后就是能加的水的高度&#xff0c;我想到了这里然后就想第i列的水其实就是第i-1列和i1列中最小的高度减去第i列的高度&#xff0c;但是其实并不是&#xff0c;比如示例中的第5…

【蓝桥杯】 [蓝桥杯 2015 省 A] 饮料换购

原题链接&#xff1a;https://www.luogu.com.cn/problem/P8627 1. 题目描述 2. 思路分析 小伙伴们可以看看这篇文章~ https://blog.csdn.net/m0_62531913/article/details/132385341?spm1001.2014.3001.5501 我们这里主要讲下方法二的推导过程&#xff1a; 列方程。 设最…

企业数字化转型需要解决哪些问题?

随着信息技术的不断发展和应用&#xff0c;企业数字化转型已成为提高竞争力和实现可持续发展的必经之路。合沃作为一家专注于为企业提供工业物联网产品与解决方案的企业&#xff0c;拥有丰富的经验和技术实力&#xff0c;正致力于助力企业实现数字化转型。本文将探讨企业数字化…

流的基本概念

流的基本概念 Streaming 101与Streaming 102原文网页 文章目录 流的基本概念术语什么是流无界数据无界数据处理有界数据批处理引擎流引擎低延迟、近似、推测结果正确性时间推理工具 数据处理模式使用经典批处理引擎进行有界数据处理用经典批处理引擎通过临时固定窗口进行无界数…

物联网无线通信方式总结

本文主要内容(一些物联网无线通信方式) 本文将介绍一些物联网无线通信方式的技术特点、底层调制方式和主要应用场景物联网无线通信方式是指利用无线技术实现物体之间的信息交换和网络连接的方式物联网无线通信方式的选择需要考虑多种因素&#xff0c;如传输距离、功耗、数据速…

天翼物联、汕头电信与汕头大学共建新一代信息技术与数字创新(物联网)联合实验室

近日&#xff0c;在工业和信息化部和广东省人民政府共同主办的2023中国数字经济创新发展大会上&#xff0c;天翼物联、汕头电信与汕头大学共建“新一代信息技术与数字创新&#xff08;物联网&#xff09;”联合实验室签约仪式举行。汕头大学校长郝志峰、中国电信广东公司总经理…

字符集与字符编码

目录 一、字符集概述二、Unicode三、Unicode字符集四、Unicode编码规则五、Unicode字符编码规则 UTF-8六、Unicode 字符编码规则 UTF-16七、Unicode 字符编码规则 UTF-32八、UTF 字节顺序标记(BOM)九、Windows 代码页十、Windows(ANSI) 字符集十一、GB2312、GBK、GB18030 一、字…