Python基础-13面向对象

13 面向对象

13.1 基础概念

  • 类:

类是具有一系列共同特征和行为的事件抽象概念集合。类所描述的概念和现实生活中的类概念非常相似。例如生物有很多种类,食物也有不同的种类,商品也有很多不同的种类。而一个类通过细分又可以划分各种小的分类,如动物就可以分为陆生动物、水生动物和两栖动物等等,被划分到同一个类,其都有着相似的特征和行为方式。

  • 实例化对象

  实例对象通常被定义某一个类事物的具体个体,是该类事物的一个具体表现。如人是一个类,”张三“就是人这个类的实例对象,因为他代表了一个具体类个体。

  • 类变量

  类变量一般是定义类中,但不在类方法中的变量,可以在整个实例化对象中公用。

  • 类属性

  类变量或实例变量,常用于描述实例对象的特征,如张三的姓名、身高等

  • 方法:

  以不严谨的方式来讲,就是定义在类中的函数称之为方法,常用于描述实例对象的行为,一般分为构造函数和常规函数。

  • 继承

  就是子类可以从父类获取父类定义的属性和方法等。

  • 方法重写

  如果父类的方法不能满足子类的要求,可以定义一个方法名与父类相同名字的方法,这个过程称之方法覆写(如Java中@Override)或方法重写

  • 实例变量

  类是一个非常抽象的概念,而实例则代表一个具体的对象,而与对象绑定的变量可称之为实例变量。

13.2 类定义

    在Python中一个简单的类定义格式如下所示:

class Person(object):
    """
    定义一个人类
    """

    # 类变量
    classVariable="sample"
    # 构造函数,也称实例化对象方法
    def __init__(self,name,age):
        self.name=name
        self.age=age
    # 类中定义的方法
    def getPersonInfo(self):
        return f"name is {self.name},age is {self.age}"

  详细解释如下所示:

  • 定义一个类的关键字为class
  • Person:定义的类名称,首字母必须大写
  • (object):表示定义的类所继承的父类,这里为object
  • 三引号中的字符为文档字符串,用于描述类的功能和作用
  • classVariable:类变量,所有实例对象可共享该变量
  • _init_:构造函数,用于实例化一个对象前,做一些的特征描述
  • name,age:实例化对象本身所具有的特征
  • getPersonInfo:类中定义的方法,用于描述类的行为,一般常以动词做为开头
  • self:代表实例化对象本身,相当于Java/C#中的this

13.3 面向对象三大特性

    面向常见的三大特性:封装继承多态。分别如下所示:

13.3.1 封装

    封装主要将类的属性和方法打包为一个整体,对外部而言,仿佛是不可见的。其主要特点如下所示:

  • 1.对象的行为对于外部来说是不可见的,即对象的状态信息是私密的。
  • 2.客户端不能通过直接操作来改变对象内部状态。相反,客户端需要通过发送消息来请求对象改变其内部状态。对象根据请求的类型,通过特定的方法改变内部状态,以做出响应
  • 3.在Python中,封装(属性和方法的隐藏)的概念不是隐式的,因为它没有提供诸如C系列语言的关键字public/private/protected。

    单下划线和双下划线在Python变量和方法名中都有其特殊的含义,一部分仅仅作为一种约定,一部分则含有特殊意义,详细如下所示:

  • 前置下划线:

    前置下划线通常代表一种约定,表示仅在内部使用,通过针对编写程序员,可以简单理解为其他语言的private关键字。

  • 后置下划线

    通常用于解决命名冲突,在Python中,str是一个将其他类型转换为字符串的函数,那如果写str_,则可以代表重新定义一个变量或方法,在PyQT中,这种用法比较多。

  • 前置双下划线

    在类环境中使用时会触发名称改写,对Python解释器有特殊含义

  • 前后双置下划线

    前后双置下划线的方法被称之为魔法方法,表示则Python语言本身定义的特殊方法,在自定义的属性中要避免使用这种命名方式。

  • 单下线

    通常用作临时或无意义变量名称,另外也可以表示Python REPL会话中上一个表达式的结果。

13.3.2 继承

    继承通常是指子类从父类中获取父类的数据和方法等等。其主要特点如下所示:

  • 1.继承表示一个类可以继承父类的大部分功能
  • 2.继承可以复用父类已经定义的功能并可以对父类原有功能进行扩展
  • 3.继承可以利用不同类的对象之间建立层次结构,在Python支持多重继承。

13.3.3 多态

    多态是指不同的对象可以执行相同的动作,但却要通过自己的实现代码来执行。国粹京剧之前都是子承父业,代代相传的艺术。假设有一对父子,父亲是非常有名的京剧艺术家,儿子长大成人,模仿父亲的戏也惟妙惟肖。有一天父亲生病,导致无法上台,而父亲的戏票已经卖出,如果退票则损失太大。因为京剧都是需要化妆后才上台,那是不是儿子可以代替父亲上台?在这里需要注意以下几点:

  • 子类以父类身份出现

    儿子代替父亲出演,化妆上台后就是父亲身份了。

  • 子类工作时必须以自己的方式实现

    儿子模仿再像也是模仿,只能以自己的理解来出演父亲的作品

  • 子类以父类的身份出现时,子类特有的属性和方法不可以使用

    儿子经过多年学习,其实已经有了自己独特的表演模式和行为习惯,但代替父亲出演时,是不能以自己的表演方式出演的。必须按父亲的风格去表演。

13.4 创建类

    我们先来创建一个类,示例如下所示:

class Person:
    
    def __init__(self,name,age):
        self.name=name
        self.age=age

    def getPersonInfo(self):
        return f"name is {self.name},age is {self.age}"
    
    def run(self):
        return f"{self.name} is running"

13.5 创建类的实例对象

    创建类的实例对象就是根据类创建具体的个体,如下所示:

person=Person("Surpass",28)

    以上就完成了根据Person类实例化一个具体的人,其名字为Surpass,年龄为28,在创建实例后,我们就可以访问类的属性和方法,通过方式为:

实例对象.[属性或方法]

13.6 使用类和实例

13.6.1 访问属性

person=Person("Surpass",28)
print(f"name is {person.name},age is {person.age}")

13.6.2 访问方法

person=Person("Surpass",28)
print(f"person info is {person.getPersonInfo()}")

13.6.3 创建多个实例

    既然类是一种事件的抽象集合,那当然也可以根据创建n个实例对象了,如下所示:

personA=Person("Surpass",28)
personB=Person("Kevin",38)
personB=Person("Lisa",20)

13.6.4 为属性指定默认值

    类中的每个属性必须有初始值,对于一些经常无需要每次赋值的属性可以指定默认值,如下所示:

class Person:

    def __init__(self,name,age=28):
        self.name=name
        self.age=age

    def getPersonInfo(self):
        return f"name is {self.name},age is {self.age}"

    def run(self):
        return f"{self.name} is running"
        
# 实例化对象
personA=Person("Surpass")

    看看上面的用法是不是跟函数使用很像。

13.6.5 修改属性值

    如果需要修改属性的值,可以使用以下方法进行修改

  • 1.通过实例进行修改
  • 2.通过调用方法进行修改

1.通过实例进行修改

    示例代码如下所示:

class Person:

    def __init__(self,name,age=28):
        self.name=name
        self.age=age

    def getPersonInfo(self):
        return f"name is {self.name},age is {self.age}"

    def run(self):
        return f"{self.name} is running"

person=Person("Surpass",28)
print(f"Before modify property {person.name} {person.age}")
person.name="Lisa"
person.age=20
print(f"After modify property {person.name} {person.age}")

输出结果如下所示:

Before modify property Surpass 28
After modify property Lisa 20

2.通过调用方法进行修改

    通过实例修改直接修改属性值非常方便,如下所示:

class Person:

    def __init__(self,name,age=28):
        self.name=name
        self.age=age

    def getName(self):
        return self.name

    def setName(self,name):
        self.name=name

    def getAge(self):
        return self.age

    def setAge(self,age):
        self.age=age

    def getPersonInfo(self):
        return f"name is {self.name},age is {self.age}"

    def run(self):
        return f"{self.name} is running"

person=Person("Surpass",28)
print(f"Before modify property {person.name} {person.age}")
person.setName("Lisa")
person.setAge(20)
print(f"After modify property {person.name} {person.age}")

输出结果如下所示:

Before modify property Surpass 28
After modify property Lisa 20

    熟悉Java的小伙伴同学,一眼就看出来这种跟Java很像,而更地道的Python写法如下所示:

class Person:

    def __init__(self,name,age=28):
        self._name=name
        self._age=age

    @property
    def name(self):
        return self._name

    @name.setter
    def name(self,value):
        if not isinstance(value,(str,)):
            raise TypeError("Expect str")
        else:
            self._name=value

    @name.deleter
    def name(self):
        raise AttributeError("Can not delete attribute")

    @property
    def age(self):
        return self._age

    @age.setter
    def age(self,value):
        if not isinstance(value,(int,)):
            raise TypeError("Expect a numer")
        elif value<0:
            self._age=0
        else:
            self._age=value

    @age.deleter
    def age(self):
        raise AttributeError("Can not delete attribute")

    def getPersonInfo(self):
        return f"name is {self.name},age is {self.age}"

    def run(self):
        return f"{self.name} is running"

person=Person("Surpass",28)
print(f"Before modify property {person.getPersonInfo()} ")
person.name="Lisa"
person.age=-123
print(f"After modify property {person.getPersonInfo()}")
print(f"测试异常情况")
person.name=98
person.age="098"

输出结果如下所示:

Before modify property name is Surpass,age is 28 
After modify property name is Lisa,age is 0
测试异常情况
Traceback (most recent call last):
  File "C:/Users/Surpass/Documents/PycharmProjects/SADCI/TempCode/basicCode.py", line 52, in <module>
    person.name=98
  File "C:/Users/Surpass/Documents/PycharmProjects/SADCI/TempCode/basicCode.py", line 15, in name
    raise TypeError("Expect str")
TypeError: Expect str

    以上这种方法,必须getter、setter、deleter三个方法的名称必须一样,且必须先创建getter,后面的setter、deleter才能访问和使用,那如果已经事先定义好了getter、setter、deleter的方法,还有没有办法实现以上这种做法?答案当然是肯定的,如下所示:

class Person:

    def __init__(self,name,age=28):
        self._name=name
        self._age=age

    def getName(self):
        return self._name

    def setName(self,value):
        if not isinstance(value,(str,)):
            raise TypeError("Expect str")
        else:
            self._name=value

    def delName(self):
        raise AttributeError("Can not delete attribute")

    def getAge(self):
        return self._age

    def setAge(self,value):
        if not isinstance(value,(int,)):
            raise TypeError("Expect a numer")
        elif value<0:
            self._age=0
        else:
            self._age=value

    def delAge(self):
        raise AttributeError("Can not delete attribute")

    ageProperty=property(getAge,setAge,delAge)
    nameProperty=property(getName,setName,delName)

    def getPersonInfo(self):
        return f"name is {self._name},age is {self._age}"

    def run(self):
        return f"{self._name} is running"

person=Person("Surpass",28)
print(f"Before modify property {person.getPersonInfo()} ")
person.setName("Lisa")
person.setAge(-123)
print(f"After modify property {person.getPersonInfo()}")
print(f"测试异常情况")
person.setName(98)

输出结果如下所示:

Before modify property name is Surpass,age is 28 
After modify property name is Lisa,age is 0
测试异常情况
Traceback (most recent call last):
  File "C:/Users/Surpass/Documents/PycharmProjects/SADCI/TempCode/basicCode.py", line 49, in <module>
    person.setName(98)
  File "C:/Users/Surpass/Documents/PycharmProjects/SADCI/TempCode/basicCode.py", line 13, in setName
    raise TypeError("Expect str")
TypeError: Expect str

    以上通过方法来修改属性的思想就是面向对象中的封装思想。但在Python,如果一个类并不存在某个属性,通过对象.属性名可以添加一个新的属性,但这种做法破坏了面向对象中的封装特性,因此不建议使用。如下所示:

class Person:

    def __init__(self,name,age=28):
        self._name=name
        self._age=age

    @property
    def name(self):
        return self._name

    @name.setter
    def name(self,value):
        if not isinstance(value,(str,)):
            raise TypeError("Expect str")
        else:
            self._name=value

    @name.deleter
    def name(self):
        raise AttributeError("Can not delete attribute")

    @property
    def age(self):
        return self._age

    @age.setter
    def age(self,value):
        if not isinstance(value,(int,)):
            raise TypeError("Expect a numer")
        elif value<0:
            self._age=0
        else:
            self._age=value

    @age.deleter
    def age(self):
        raise AttributeError("Can not delete attribute")

    def getPersonInfo(self):
        return f"name is {self._name},age is {self._age}"

    def run(self):
        return f"{self._name} is running"

person=Person("Surpass",28)
# 通过实例直接增加一个新的属性,不推荐
person.country="China"
print(f"add new person property : {person.country}")

输出结果如下所示:

add new person property : China

13.7 属性

    属性常用于描述类的特征,可以通过类中定义的方法访问,也可以通过类或实例进行访问。因因此属性又可分为类属性实例属性

13.7.1 实例属性

    实例属性,通俗来讲是通过self.变量名称定义的属性,有时也称实例变量,属于特定的实例对象。示例如下代码如下所示:

class Person:

    def __init__(self,name,age):
        # 实例属性赋值
        self._name=name
        self._age=age

    # 获取实例属性值
    def getName(self):
        return self._name


# 实例属性需要与实例对象进行绑定后才能访问
person=Person("Surpass",28)
print(f"{person.getName()}")

13.7.2 类属性

    Python允许声明属于类本身的变量称之为类属性或类变量,类属性无需实例化对象即可,如下所示:

class Person:
    # 类属性
    classVar="类属性"
    def __init__(self,name,age):
        # 实例属性赋值
        self._name=name
        self._age=age

    # 获取实例属性值
    def getName(self):
        return self._name


# 实例属性需要与实例对象进行绑定后才能访问
person=Person("Surpass",28)
# 访问实例属性-通过实例
print(f"访问实例属性-通过实例:{person.getName()}")
# 访问类属性-通实例
print(f"访问类属性-通实例:{person.classVar}")
# 访问类属性
print(f"访问类属性:{Person.classVar}")

输出结果如下所示:

访问实例属性-通过实例:Surpass
访问类属性-通实例:类属性
访问类属性:类属性

13.7.3 私有属性

    Python没有Java/C#语言的权限控制字段,如public/protected/private等等,通常的约定是以单下划线开头定义的属性为私有属性,如下所示:

class Person:
    # 类私有属性
    _classVar="类属性"
    def __init__(self,name,age):
        # 实例属性赋值
        self._name=name
        self._age=age

13.7.4 特殊属性

    Python对象中包含很多开始和结尾以双下划线的开始和结束的方法,称为特殊属性,常见的特殊属性如下所示:

特殊方法 含义 示例
obj. __dict__ 对象的属性字典 int.__dict__
instance.__class__ 对象所属的类 int.__class__
class.__bases__ 类的基类元组 int.__bases__
class.__base__ 类的基类 int.__base__

13.7.5 小结

  • 1.实例属性只能通过实例对象进行访问
  • 2.类属性属于类和实例对象共享的变量,无需要实例化即可访问,在实例化对象也可以通过实例对象进行访问
  • 3.如果要定义一个私有,只需要在变量加一个下划线即可
  • 4.面向对象编程的封装特性原则上要求不直接访问类中的属性,Python中可以通过定义私有属性,然后定义相应的方法来访问和修改私有属性,也可以使用装饰器@property来装饰这些函数,达到程序将函数当做属性访问。

13.8 继承

    在创建一个类,并不需要每次都新建。如果已经现成的类,其已有功能能满足你的要求,那么则可以使用继承来达到目的。一个类继承另一个类时,将自动获取另一个类的属性和方法。原有的类称为父类,新的类称为子类。子类除了继承父类的属性和方法,还可以定义自己的属性和方法,也可以重写父类的方法。

13.8.1 构造方法_init_()

    创建子类的实例时,Python首先需要完成的任务是给父类的所有属性赋值。因此子类的方法_init_()需要父类施以援手。如下所示:

1.显式调用父类构造方法

class Person:

    def __init__(self,name,age):
        self._name=name
        self._age=age

    def getPersonInfo(self):
        return f"name is {self._name},age is {self._age}"

    def run(self):
        return f"{self._name} is running"

class Student(Person):

    def __init__(self,name,age,):
        super().__init__(name,age)

student=Student("Surpass",25)
print(student.getPersonInfo())

输出结果如下所示:

name is Surpass,age is 25

    以上代码详解如下所示:

  • super()._init_(name,age):显式声明调用父类构造方法,如果子类没有特殊属性,也可以省略
  • student.getPersonInfo():子类调用父类继承过来方法

2.显式调用父类构造方法并添加自己的属性

    示例如下所示:

class Person:

    def __init__(self,name,age):
        self._name=name
        self._age=age

    def getPersonInfo(self):
        return f"name is {self._name},age is {self._age}"

    def run(self):
        return f"{self._name} is running"

class Student(Person):

    def __init__(self,name,age,classNumber):
        super().__init__(name,age)
        self._classNumber=classNumber

    def getStudentInfo(self):
        return f"name is {self._name},age is {self._age},classNumer is {self._classNumber}"

student=Student("Surpass",25,"Class-NO.1")
print(student.getStudentInfo())

输出结果如下所示:

name is Surpass,age is 25,classNumer is Class-NO.1

3.隐式调用父类构造方法

    示例如下所示:

class Person:

    def __init__(self,name,age):
        self._name=name
        self._age=age

    def getPersonInfo(self):
        return f"name is {self._name},age is {self._age}"

    def run(self):
        return f"{self._name} is running"

class Student(Person):
    pass

student=Student("Surpass",25)
print(student.getPersonInfo())

输出结果如下所示:

name is Surpass,age is 25

    通过以上几种,那可以猜测一下,是不是不管子类有没有声明构造方法,都是先调用父类构造方法呢?来看看以下示例:

1.父类和子类都有定义自己构造函数

class Person:

    def __init__(self,name,age):
        self._name=name
        self._age=age
        print("1.调用父类构造方法")

    def getPersonInfo(self):
        return f"name is {self._name},age is {self._age}"

    def run(self):
        return f"{self._name} is running"

class Student(Person):
    def __init__(self,name,age,classNumber):
        self._name=name
        self._age=age
        self._classNumber=classNumber
        print("2.调用子类构造方法")

    def getStudentInfo(self):
        return f"name is {self._name},age is {self._age},classNumer is {self._classNumber}"

student=Student("Surpass",25,"Class-No.1")

输出结果如下所示:

2.调用子类构造方法

2.父类有定义自己构造函数,子类未定义自己的构造函数

class Person:

    def __init__(self,name,age):
        self._name=name
        self._age=age
        print("1.调用父类构造方法")

    def getPersonInfo(self):
        return f"name is {self._name},age is {self._age}"

    def run(self):
        return f"{self._name} is running"

class Student(Person):
    pass

输出结果如下所示:

1.调用父类构造方法

3.父类和子类都有定义自己构造函数,但子类显式调用父类构造方法

class Person:

    def __init__(self,name,age):
        self._name=name
        self._age=age
        print("1.调用父类构造方法")

    def getPersonInfo(self):
        return f"name is {self._name},age is {self._age}"

    def run(self):
        return f"{self._name} is running"

class Student(Person):

    def __init__(self,name,age,classNumber):
        super().__init__(name,age)
        self._classNumber=classNumber
        print("2.调用子类构造方法")

    def getStudentInfo(self):
        return f"name is {self._name},age is {self._age},classNumer is {self._classNumber}"

student=Student("Surpass",25,"Class-NO.1")

输出结果如下所示:

1.调用父类构造方法
2.调用子类构造方法

    通过以上代码,可以总结出以下结论:

  • 1.在子类未定义构造函数,则隐式调用父类构造函数
  • 2.在子类如果显式声明调用父类构造函数,则先调用父类构造函数,再调用子类的构造函数
  • 3.如果子类自己定义了构造函数,则只调用子类本身的构造函数
  • 4.如果需要显式调用父类构造函数,可以使用super(英文:superclass)关键字
  • 5.擅用super方法,可以简化代码,当要调用父类构造方法或普通方法时,均可以使用该关键字

13.8.2 子类定义属性和方法

    当一个类继承自父类后,父类的功能无法自身需求后,子类可以定义自己的属性和方法,示例如下所示:

class Person:

    def __init__(self,name,age):
        self._name=name
        self._age=age
        print("1.调用父类构造方法")

    def getPersonInfo(self):
        return f"name is {self._name},age is {self._age}"

    def run(self):
        return f"{self._name} is running"

class Student(Person):

    def __init__(self,name,age,classNumber):
        super().__init__(name,age)
        # 子类自身定义的属性
        self._classNumber=classNumber
        print("2.调用子类构造方法")

    # 子类自身定义的方法,显式调用父类方法
    def getStudentInfo(self):
       return super().getPersonInfo()

    # 子类自身定义的方法
    def getStudentCount(self):
        studentList=[]
        studentList.append(self._name)
        return len(studentList)

输出结果如下所示:

1.调用父类构造方法
2.调用子类构造方法
调用父类方法的结果:name is Surpass,age is 25
调用子类自身的方法:student count is 1

13.8.3 覆写父类方法

    在Python中覆写父类方法非常简单,只需要让子类中方法名称与父类名称一致即可,如下所示:

class Person:

    def __init__(self,name,age):
        self._name=name
        self._age=age
        print("1.调用父类构造方法")

    def getPersonInfo(self):
        return f"name is {self._name},age is {self._age}"

    def run(self):
        return f"{self._name} is running"

class Student(Person):

    def __init__(self,name,age,classNumber):
        super().__init__(name,age)
        # 子类自身定义的属性
        self._classNumber=classNumber
        print("2.调用子类构造方法")

    # 子类自身定义的方法
    def getStudentCount(self):
        studentList=[]
        studentList.append(self._name)
        return len(studentList)

    # 覆写父类已经定义的方法
    def getPersonInfo(self):
        return f"name is {self._name},age is {self._age},class number is {self._classNumber}"

student=Student("Surpass",25,"Class-NO.1")
print(f"调用子类自身的方法:student count is {student.getStudentCount()}")
print(f"调用子类覆写父类的方法:{student.getPersonInfo()}")

输出结果如下所示:

1.调用父类构造方法
2.调用子类构造方法
调用子类自身的方法:student count is 1
调用子类覆写父类的方法:name is Surpass,age is 25,class number is Class-NO.1

13.8.4 多重继承

    像Java/C#等语言是单继承(即一个类同时只能继承一个父类,)和多接口继承(即一个类可继承多个接口),多层继承(因为继承的父类又继承自其他父类,看起来有多个层次),在Python中支持多重继承(即一个类可以同时继承多个类),示例如下所示:

class A:
    def __init__(self):
        print("1.父类构造方法-1")

    def getMethonA(self):
        print(f"1.父类方法{self.getMethonA.__name__}")


class B:
    def __init__(self):
        print("2.父类构造方法-2")

    def getMethonB(self):
        print(f"2.父类方法{self.getMethonB.__name__}")

class C:
    def __init__(self):
        print("3.父类构造方法-3")

    def getMethonC(self):
        print(f"3.父类方法{self.getMethonC.__name__}")


class D(A,B,C):

    def getMethonD(self):
        print(f"4.子类方法{self.getMethonD.__name__}")

d=D()
d.getMethonA()
d.getMethonB()
d.getMethonC()
d.getMethonD()

输出结果如下所示:

1.父类构造方法-1
1.父类方法getMethonA
2.父类方法getMethonB
3.父类方法getMethonC
4.子类方法getMethonD

    使用多重继承有好也有坏

1.优点是可以调用多个父类的方法
2.缺点是如果继承的多个父类又属于同一个父类,则会出现二义性,另外也会造成逻辑上的复杂性增加,因此不推荐使用多重继承

13.9 方法

    方法是与类相关的函数,Python用于定义一个方法格式如下所示:

def 方法名称(self,参数1,参数2...):
    """函数描述"""
    函数体
    return 返回值

    方法的定义也前面的函数是一致,唯一的区别,是参数第一个位置必须是self,在Python方法常被分为构造方法普通方法类方法静态方法

13.9.1 构造方法

    构造方法通常用于在实例化一个对象前,给对象传递一些必须的属性值,通常以__init__()做为标识符,一般格式如下所示:

class Person:
    # 构造方法
    def __init__(self,name,age):
        self._name=name
        self._age=age

13.9.2 普通方法

    除了构造方法、类方法和静态方法之外的方法称之为普通方法,这种方法一般由用户自行定义的,示例如下所示:

class Person:
    # 构造方法
    def __init__(self,name,age):
        self._name=name
        self._age=age

    # 普通方法
    def getName(self):
        return self._name

person=Person("Surpass",28)
print(f"访问普通方法:{person.getName()}")

输出结果如下所示:

访问普通方法:Surpass

13.9.3 类方法

    类方法与类属性非常相似,是一种属于类的方法,不需要实例化就可以访问的一种方法,但需要使用装饰器@classmethon,其语法格式如下所示,注意与普通方法的区别在于第一个参数:

@classmethod
def 方法名称(cls,参数1,参数2...):
   函数体
   return 返回值

    示例代码如下所示:

class Person:
    className="这是类属性"
    # 构造方法
    def __init__(self,name,age):
        self._name=name
        self._age=age

    # 普通方法
    def getName(self):
        return self._name

    # 类方法,可以访问类属性,但不能访问对象属性
    @classmethod
    def getPersonInfo(cls,hello="Hello,"):
        return f"{hello} {cls.className} "


person=Person("Surpass",28)
print(f"访问普通方法:{person.getName()}")
print(f"访问类方法:{Person.getPersonInfo('Welcome, ')}")

输出结果如下所示:

访问普通方法:Surpass
访问类方法:Welcome,  这是类属性

13.9.4 静态方法

    静态方法即不需要传入self参数、也不需要传入cls参数,从面使得调用静态方法并不能获取得类中定义的属性和其他方法,但并不会影响类对象和实例对象的状态。静态方法有点像附属于类的对象的工具,与普通方法不同,调用静态方法时,只能通过类对象或实例对象,而不能脱离类对象使用,即静态方法被束缚在类对象中。其语法格式如下所示:

@staticmethod
def 方法名称(参数1,参数2...):
   函数体
   return 返回值

    示例代码如下所示:

class Person:
    className="这是类属性"
    # 构造方法
    def __init__(self,name,age,height,weight):
        self._name=name
        self._age=age
        self._height=height
        self._weight=weight

    # 普通方法
    def getName(self):
        return self._name

    # 类方法,可以访问类属性,但不能访问对象属性
    @classmethod
    def getPersonInfo(cls,hello="Hello,"):
        return f"{hello} {cls.className} "

    # 静态方法,不可以访问类属性
    @staticmethod
    def getBMI(h,w):
        return w/((h/100)**2)

    # 普通方法
    def showBMI(self):
        bmi=self.getBMI(self._height,self._weight)
        return bmi

person=Person("Surpass",28,170,59)
print(f"访问普通方法:{person.getName()}")
print(f"访问类方法:{Person.getPersonInfo('Welcome, ')}")
print(f"访问静态方法:{Person.getBMI(170,59)}")
print(f"通过实例对象访问:{person.showBMI()}")

输出结果如下所示:

访问普通方法:Surpass
访问类方法:Welcome,  这是类属性
访问静态方法:20.41522491349481
通过实例对象访问:20.41522491349481

13.9.5 小结

  • 1.类方法和静态方法都是通过装饰器实现,而构造方法和普通方法不昌
  • 2.普通方法需要显式定义self参数,类方法需要定义cls参数,静态方法不需要self和cls参数
  • 3.普通方法只能通过实例对象调用,类方法和静态方法可以通过类对象或实例对象调用,如果是使用实例对象调用的类方法和静态方法,最终都会转而通过类对象调用
  • 4.普通方法使用最多,可以直接处理实例对象的各种逻辑;类方法不需要创建实例对象,直接处理类对象逻辑;静态方法将与类对象相关的部分逻辑抽离出来,可以用于测试,也方便后期维护代码
  • 5.普通方法和类方法,可以改变实例对象或类对象的状态,而静态方法不能。

本文地址:https://www.jianshu.com/p/f88330bc552d

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352