当前位置: 首页 > article >正文

类与对象—python

一、类的含义

1.1类的作用(理解)

收集学生信息时,如果让同学们自主填写,信息的顺序、格式不一,内容混乱。如果发给同学们既定的表格,同学们按照规定的顺序、格式进行填写,那信息就会一目了然,如下图所示:

在程序中如果简单地使用变量来记录学生信息,不仅在创建变量,赋值这一步繁琐,在使用学生信息相关变量的时候,也非常不便,如下列代码所示:

#学生小明的信息
student_name1='小明'
student_ID1=10234049980
student_age1=19
student_year1=2021
student_class1='会计7班'
#学生小刚的信息
student_name2='小刚'
student_ID2=10234057760
student_age2=20
student_year2=2020
student_class2='土木3班'

类的作用就相当于上图中的表格,类相当于一个范式,在程序中也可以做到和生活中一样,以设计表格、生产表格、填写表格的形式组织信息,示例如下(此处看懂即可,语法在1.2中细讲):

#在程序中设计表格,我们称之为:设计类( class )
class Student_info:
    name=None      #学生姓名
    ID=None        #学生学号
    age=None       #学生年龄
    year=None      #学生年级
    classes=None   #学生班级

#在程序中打印生产表格,我们称之为:创建对象
stu1=Student_info()
stu2=Student_info()

#在程序中填写表格,我们称之为:对象属性赋值
stu1.name='小明'
stu1.ID=10234049980
stu1.age=19
stu1.year=2021
stu1.classes='会计7班'

这样写代码只是为了让大家理解类的角色,继续往下学习,可以将代码写得更简洁高效。

1.2类的语法

类由两部分组成,包括类的属性和类的行为。

语法:

class 类名称:

    类的属性

    类的行为

  • 类的属性:即定义在类中的变量(成员变量)
  • 类的行为:即定义在类中的函数(成员方法)

创建类对象的语法:

对象=类名称()

成员方法的定义语法:

def 方法名 (self,形参1,形参2,……):

    方法体

在方法定义的列表中,有一个self关键字,self关键字是成员方法定义时必须写的。

  • 它用来表示类对象自身的意思
  • 当我们使用类对象调用方法时, self 会自动被 python 传入
  • 在方法内部,想要访问类的成员变量,必须使用 self

我们利用类来描述书本信息(书名、作者、售价),示例如下:

#设计类( class )
class Book:
    #定义成员变量
    name=None   #书名
    author=None #作者
    price=None  #价格
    #定义成员方法
    def instru(self):
        print(f'《{self.name}》作者为{self.author},价格为{self.price}元')
#创建对象
book1=Book()
#赋值
book1.name='往昔之光'
book1.author='星河'
book1.price=20
#调用成员方法
book1.instru() #定义参数列表中只有self,调用时无须传参

输出:

《往昔之光》作者为星河,价格为20元

二、类和对象

类有属性和行为,现实世界的事物也有属性和行为,比如打扫卫生这件事,属性是:何人、何时、何地,行为是:擦桌子、扫地、拖地,或者又比如饮水机这个物体,属性是:颜色、体积、重量,行为是:烧水、保温。使用程序中的类,可以完美的描述现实世界的事物,那我们为什么还需要创建对象呢?
因为类只是一种程序内的“设计图纸”,需要基于图纸生产实体(对象),才能正常工作,这种套路,称之为:面向对象编程。

例如我们要设计一款闹钟,代码如下所示:

#设计闹钟图纸——设计类
class Clock:
    ID=None    #序列号
    price=None #价格
    #响铃
    def ring(self):
        import winsound
        winsound.Beep(2000,3000)#声音频率,持续时间
#根据图纸进行生产——创建对象
clock1=Clock()
#使用对象的属性和行为(完成具体工作)
clock1.ID=123456
clock1.price=39
print(f'闹钟的ID为{clock1.ID},价格为{clock1.price}元')
clock1.ring()

运行代码后,会得到如下输出,且听到铃声。

闹钟的ID为123456,价格为39元

面向对象编程:设计类,基于类创建对象,由对象做具体的工作。

三、构造方法

Python 类内置的方法有很多,这些内置的类方法各自有各自特殊的功能,被称之为:魔术方法,我们主要了解以下5个方法即可。

3.1__init__( ) 方法

在给类对象赋值时,我们可以依次为对象的属性赋值,但是当属性较多时,就会比较麻烦。我们也可以只用一行代码进行赋值,高效地进行这个过程,此时就需要用到__init__( ) 方法。 __init__( ) 方法是Python 类中的构造方法之一,可以实现:

  • 在创建类对象(构造类)的时候,会自动执行。
  • 在创建类对象(构造类)的时候,将传入参数自动传递给 __init__ ( )方法使用。

借此特性可以给成员变量赋值所以我们才能完成只用一行代码进行赋值。

代码示例:

class Book:
    #定义成员变量
    name=None   #书名
    author=None #作者
    price=None  #价格
    # 定义成员方法
    def __init__(self,name,author,price):
        self.name=name     #书名
        self.author=author #作者
        self.price=price   #价格
        print(f'《{self.name}》作者为{self.author},价格为{self.price}元')
#创建对象并赋值
book2=Book('红楼梦','曹雪芹',79)#像传参一样,按顺序赋值

输出:

《红楼梦》作者为曹雪芹,价格为79元

注意:

  • init 前后都有 2 个下划线
  • 构造方法属于成员方法,要在参数列表中加上 self 关键字

我们可以写得更简略,在定义类的时候,定义成员变量这一步也可以省略:

class Book:
    
    # 定义成员方法
    def __init__(self,name,author,price):
        self.name=name     #书名
        self.author=author #作者
        self.price=price   #价格
        print(f'《{self.name}》作者为{self.author},价格为{self.price}元')
#创建对象
book2=Book('红楼梦','曹雪芹',79)

是否使用__init__ ( )方法可以对比一下:

3.2__str__ ( )方法

__str__ ( )字符串方法可以控制类对象转换为字符串。

我们先看这段代码:

#设计类
class Room:
    def __init__(self,desk,chair):
        self.desk = desk  # 桌子的数量
        self.chair=chair #椅子的数量
#创建对象并赋值
room1=Room(12,12)

print(room1)

打印类对象,获得的是其内存地址,输出如下:

<__main__.Room object at 0x0000020A8BB73D30>

使用__str__ ( )字符串方法,控制输出的内容,再次打印类对象,输出就是我们控制的内容:

#设计类
class Room:
    def __init__(self,desk,chair):
        self.desk = desk  # 桌子的数量
        self.chair=chair #椅子的数量

    def __str__(self):
        return f'桌子有{self.desk}张,椅子有{self.chair}个'
#创建对象并赋值
room1=Room(12,12)

print(room1)

输出:

桌子有12张,椅子有12个

3.3__lt__( )方法

直接对 2 个对象进行比较是不可以的,但是在类中实现 __lt__ 方法,即可同时完成:小于符号和 大于符号的比较。

  • 参数: other (另一个类对象)
  • 返回值: True 或 False(实际判断是否与方法中定义一致)

比如我们要比较职工的年龄:

代码示例:

#设计类
class staff():
    def __init__(self,name,age):
        self.name=name
        self.age=age
    def __lt__(self,other):
        return self.age<other.age
#创建对象并赋值
sta1=staff('小明',24)
sta2=staff('小刚',26)
print(sta1<sta2)

输出:

True

3.4__le__( )方法

同__lt__( )方法类似, __le__( )方法用于 <= 和 >= 两种比较运算。

  • 参数: other (另一个类对象)
  • 返回值: True 或 False(实际判断是否与方法中定义一致)

代码示例:

#设计类
class staff():
    def __init__(self,name,age):
        self.name=name
        self.age=age
    def __le__(self,other):
        return self.age<=other.age
#创建对象并赋值
sta1=staff('小明',23)
sta2=staff('小刚',26)
sta3=staff('小蓝',26)
sta4=staff('小雨',22)
print(sta1<=sta2) #小明比小刚年龄小,结果为True
print(sta2>=sta3) #小刚比小蓝年龄一样大小,结果为True
print(sta1<=sta4) #小明比小雨年龄大,结果为False

输出:

True
True
False

3.5__eq__( )方法

 __eq__ 方法,用于比较 2 个对象(指定的部分)是否相等。

  • 参数: other(另一个类对象)
  • 返回值: True 或 False
     

代码示例:

#设计类
class staff():
    def __init__(self,name,age):
        self.name=name
        self.age=age
    def __eq__(self,other):
        return self.age==other.age
#创建对象并赋值
sta1=staff('小明',23)
sta2=staff('小刚',26)
sta3=staff('小蓝',26)
sta4=staff('小雨',22)

print(sta2==sta3) #小刚和小蓝年龄相等,结果为True

输出:

True

代码示例:

#设计类
class staff():
    def __init__(self,name,age):
        self.name=name
        self.age=age
    def __eq__(self,other):
        return self.name==other.name
#创建对象并赋值
sta1=staff('小明',23)
sta2=staff('小刚',26)
sta3=staff('小蓝',26)
sta4=staff('小雨',22)

print(sta2==sta3) #小刚和小蓝姓名不同,结果为False

输出:

False

四、类型注解

类型注解:在代码中涉及数据交互的地方,提供数据类型的注解(显式的说明)。
主要功能:

  • 帮助第三方 IDE 工具(如 PyCharm )对代码进行类型推断,协助做代码提示
  • 帮助开发者自身对变量进行类型注释

我们在PyCharm中编写代码时,经常会出现如下图中的提示:

这是因为PyCharm能确定list1是列表类型,可以做出提示。

当PyCharm不能确定数据类型时,无法做成提示。如下图,定义一个函数 func ,接收一个参数 data,PyCharm不能确定data的类型,自然也无法提示。

当我们调用方法(内置模块提供的),进行传参的时候,按快捷键 ctrl + p 会弹出提示。

4.1变量的类型注解

为变量设置类型注解有2种方式。

4.1.1变量:类型

为变量设置类型注解的第一种方式,利用基础语法:变量:类型

如下列代码所示:

#基础数据类型注解
var1:int=10
var2:float=3.14
var3:bool=True
var4:str='apple'
#类对象类型注解
class Book:
    name=None   #书名
    author=None #作者
    price=None  #价格
book:Book=Book()
#基础容器类型注解
list1:list=[1,2,3,4]
tuple1:tuple=(1,2,3,4)
set1:set={1,2,3,4}
dict1:dict={'abc':666}
str1:str='apple'
#容器类型详细注解,对容器元素类型进行注解
list2:list[int]=[1,2,3,4]
tuple2:tuple[int,str,bool]=(1,'a',True)#元组类型设置类型详细注解,需要将每一个元素都标记出来
set2:set[int]={1,2,3,4}
dict2:dict[str,int]={'abc':666}#字典类型设置类型详细注解,需要 2 个类型,第一个是 key 第二个是value

注意:

  • 元组类型设置类型详细注解,需要将每一个元素都标记出来
  • 字典类型设置类型详细注解,需要标注 2 个类型,第一个是 key 第二个是value

4.1.2# type: 类型

为变量设置类型注解的第二种方式,在注释中进行类型注解,语法:# type: 类型

一般情况下变量的类型很容易一眼看出,无需注解,而在无法直接看出变量类型之时,最好添加变量的类型注解,如下列代码所示:

import json
import random
class Desk:
    ID=None
    price=None
desk=Desk #type:Desk

var5=random.randint(1,10) #type:int
var6=json.loads(data)     #type:dict[str,int]

4.2函数(方法)的类型注解

函数(方法)可以在两处地方添加注解:

  • 形参的类型注解
  • 返回值的类型注解

语法:

def 函数名(形参:类型,形参:类型,……,形参:类型)->返回值类型:

      函数体

代码示例:

def func1(x:int,y:int)->int:
    return x+y
a=func1(2,3)
print(a)  #a=5

4.3Union类型

列表,元组,集合,字典可以存储不同类型的数据,例如:list1=[1,'a',3.14],list1中有整数元素、字符元素、浮点数元素,那如何对list1进行类型注解呢?形如list1,就需要使用Union 联合类型注解。

使用 Union[ 类型 , ......, 类型 ]可以定义联合类型注解,Union 联合类型注解在变量注解、函数(方法)形参和返回值注解中,均可使用,使用方式如下:

  • 导包: from typing import Union
  • 语法: Union[ 类型 , ......, 类型 ]

代码示例:

from typing import Union
list3:list[Union[int,str,float]]=[1,'a',3.14]
dict3:dict[str,Union[str,int]]={'name':'加菲猫','age':3}

注意:

类型注解只是提示性的,并非决定性的,数据类型和注解类型无法对应也不会导致错误。


http://www.kler.cn/a/324609.html

相关文章:

  • 【LeetCode】--- MySQL刷题集合
  • C语言-----扫雷游戏
  • MyBatis-Plus的插件
  • 蚁群算法 (Ant Colony Optimization) 算法详解及案例分析
  • k8s namespace绑定节点
  • PIC单片机HEX文件格式分析
  • ERP系统委外工单管理
  • java NIO实现UDP通讯
  • - 串口通信
  • 多线程篇八
  • SpringBoot集成阿里easyexcel(二)Excel监听以及常用工具类
  • 阴影的基本原理
  • 梳理一下C语言中的格式说明符
  • uniapp js判断key是否在json中?
  • ArcgisEngine开发中,Ifeatureclass.Addfield 报错0x80040655处理方法
  • 0基础学习CSS(六)字体
  • python-list-comprehension-three-way-partitioning-array-around-given-range
  • iText 5 通过创建 Document 对象,并使用 PdfWriter 将内容写入 PDF 文件
  • ubuntu重新安装clickhouse
  • 前端面试题(九)
  • 【Mybatis】常见面试题汇总 共56题
  • 复试经验分享《一、问答题自测》(408、相关前沿技术)
  • 在Kali Linux VNC服务器上安装RDP服务
  • Android使用RecyclerView仿美团分类界面
  • 【JavaEE】——阻塞队列,生产消费者模型(较难)
  • BACnet协议-(基于ISO 8802-3 UDP)(2)