美文网首页Python新世界
面向对象进阶版!从入门到大牛!Python就是这么牛逼!

面向对象进阶版!从入门到大牛!Python就是这么牛逼!

作者: 919b0c54458f | 来源:发表于2018-09-14 18:42 被阅读12次

    定义类

    在 Python 中,类的定义使用 class 关键字来实现,语法如下:

    class ClassName:

    '''类的帮助信息''' # 类文档字符串

    statement # 类体

    参数说明:

    ClassName :用于指定类名,一般使用大写字母开头,如果类名中包括两个单词,第二个单词的首字母也大写,这种命名方法也称为“驼峰式命名法”,这是惯例。当然,也可根据自己的习惯命名,但是一般推荐按照惯例来命名。

    statement :类体,主要由类变量(或类成员)、方法和属性等定义语句组成。如果在定义类时,没想好类的具体功能,也可以在类体中直接使用 pass 语句代替。

    例如,下面以大雁为例声明一个类,代码如下:

    class Animal:

    '''动物类'''

    pass

    创建类的实例

    class 语句本身并不创建该类的任何实例。所以在类定义完成以后,可以创建类的实例,即实例化该类的对象。创建类的实例的语法如下:

    进群:548377875   即可获取数十套PDF哦!

    ClassName(parameterlist)

    ClassName 是必选参数,用于指定具体的类;

    parameterlist 是可选参数,当创建一个类时,没有创建 __init__() 方法,或者 __init__() 方法只有一个 self 参数时,parameterlist 可以省略。

    例如,创建Geese 类的实例,可以使用下面的代码:

    wildGoose = Animal() # 创建动物类的实例

    print(wildGoose) # 输出实例

    执行上面代码后,将显示类似下面的内容:

    <__main__.Animal object at0x0000000002F47AC8>

    从上面的执行结果中可以看出, wildGoose 是 Geese 类的实例。

    创建类的方法

    类中的方法与函数非常相似,可以理解为方法是在类中定义的函数。创建方法和创建函数类似,语法格式如下:

    def functionName(self,parameterlist):

    block

    参数说明如下:

    functionName:用于指定方法名,一般使用小写字母开头;

    self:必要参数,表示类的实例,其名称可以是self以外的单词,使用self只是一个惯例而已;

    parameterlist:用于指定除self参数以外的参数,各参数间使用逗号“,”进行分隔;

    block:方法体,实现的具体功能。

    说明:实例方法和Python中的函数的主要区别就是,函数实现的是某个独立的功能,而实例方法是实现类中的一个行为,是类的一部分。

    实例方法创建完成后,可以通过类的实例名称和点(.)操作符进行访问。具体的语法格式如下:

    instanceName.functionName(parametervalue)

    instanceName为类的实例名称;

    functionName为要调用的方法名称;

    parametervalue表示为方法指定对应的实际参数。

    创建一个Animal类,并定义eat()、play()和sleep()3个方法。实例化Animal类,并调用这3个方法。下面看一段示例代码。

    class Animal: # 创建动物类

    '''动物类'''

    def eat(self): # 定义吃食物方法

    print('正在吃食物')

    def play(self): # 定义玩耍方法

    print('正在玩耍')

    def sleep(self): # 定义休息方法

    print('正在休息')

    Dog = Animal() # 创建动物类的实例

    Dog.eat() # 调用吃食物的方法

    Dog.play() # 调用玩耍的方法

    Dog.sleep() # 调用休息的方法

    运行结果如下所示。

    正在吃食物

    正在玩耍

    正在休息

    创建__init__()方法

    在创建类后,通常会创建一个__init__()方法。该方法是一个特殊的方法,类似Java语言中的构造方法。每当创建一个类的新实例时,Python都会自动执行它。__init__()方法必须包含一个self参数,并且必须是第一个参数。self参数是一个指向实例本身的引用,用于访问类中的属性和方法。在方法调用时会自动传递实际参数self。因此,当__init__()方法只有一个参数时,在创建类的实例时,就不需要指定实际参数了。

    说明:在__init__()方法的名称中,开头和结尾处是两个下划线(中间没有空格),这是一种约定,旨在区分Python默认方法和普通方法。下面仍然以大雁为例声明一个类,并且创建__init__()方法,下面看一段示例代码。

    class Animal: # 创建动物类

    '''大雁类'''

    def __init__(self):

    print('这是一个动物类')

    def eat(self): # 定义吃食物方法

    print('正在吃食物')

    def play(self): # 定义玩耍方法

    print('正在玩耍')

    def sleep(self): # 定义休息方法

    print('正在大雁休息')

    Dog = Animal() # 创建动物类的实例

    Dog.eat() # 调用吃食物的方法

    Dog.play() # 调用玩耍的方法

    Dog.sleep() # 调用休息的方法

    运行结果如下所示。

    这是一个动物类

    正在吃食物

    正在玩耍

    正在大雁休息

    从上面的运行结果可以看出,在创建大雁类的实例时,虽然没有为__init__()方法指定参数,但是该方法会自动执行。

    在__init__()方法中,除了self参数外,还可以自定义一些参数,参数间使用逗号“,”进行分隔。例如,下面的代码将在创建__init__()方法时,再指定3个参数,分别是name、gender和weight。

    class Animal: # 创建动物类

    '''动物类'''

    def __init__(self,name,gender,weight):

    '''初始化赋值'''

    self.name = name

    self.gender = gender

    self.weight = weight

    def info(self):

    '''输出基本信息'''

    if self.gender == 'male':

    print('这只%s是公的' % self.name)

    else :

    print('这只%s是母的' % self.name)

    print('%s的重量是%s千克' % (self.name , self.weight))

    dog = Animal('腊肠狗','male',10) # 创建动物类的实例

    dog.info() # 调用实例的info方法,输出基本信息

    cat = Animal('波斯猫','female',5) # 创建动物类的实例

    cat.info() # 调用实例的info方法,输出基本信息

    执行上面的代码,运行结果如下:

    这只腊肠狗是公的

    腊肠狗的重量是10千克

    这只波斯猫是母的

    波斯猫的重量是5千克

    上述代码中,__init__()方法的参数包含新创建的实例self和在调用类对象时提供的参数。在__init__()内,通过将属性分配给self来将其保存到实例中。例如,self.name = name表示将name属性保存在实例中。在新创建的实例返回到goods_a和goods_b后,使用点号(.)运算符即可访问这些属性以及类的属性。如goods_a.info()。点号(.)运算符属于属性绑定。访问属性时,首先会检查实例,如果不知道该属性的任何信息,则会对实例的类进行搜索。这是因为类和其所有实例共享其属性的底层机制。

    常见错误:在为类创建__init__()方法时,在开发环境中运行下面代码:

    class Animal:

    '''动物类'''

    def __init__(): # 构造方法

    print("这是动物类!")

    dog = Animal()# 创建动物类的实例

    将显示如下所示的异常信息。该错误的解决方法是在第3行代码的括号中添加self。

    Traceback (most recent call last):

    File "test.py", line 6, in

    dog = Animal()# 创建动物类的实例

    TypeError: __init__() takes 0 positional arguments but 1 was given

    self的作用

    在编写类方法时,必须多加一个self在参数列表开头,但是我们却没有为这个参数赋值,那么这个self的作用是什么呢?其实,self是一种特定的变量,它引用的是对象本身。按照惯例,它被赋予self这一名称。当然也可以使用其他的名字,但是强烈推荐使用self这一名称,因为任何一个程序员都可以一眼认出它。

    那么 Python 是如何给 self 赋值的?我们依然以动物类为例。在Animal类创建了2个实例dog和cat 。当调用这个对象的方法,如 dog.info()时,Python将会自动将其转换成Animal.info(dog) 。同理,当调用cat.info()时,Python将会自动将其转换成Animal.info(cat) 。如果info()函数还有参数,例如:

    def info(self,arg1,arg2):

    pass

    那么调用dog.info(arg1,arg2)时,Python将会自动将其转换成Animal.info(dog,arg1,arg2) 。

    类属性和实例属性

    前面介绍了类中的方法,本节来介绍一下类中的另一个成员——属性。所谓的属性,可以理解为在类中定义的变量。根据定义位置,又可以分为类属性和实例属性。

    1.类属性

    类属性是指定义在类中,并且在函数体外的属性。类属性可以在类的所有实例之间共享值,也就是在所有实例化的对象中公用。

    例如,定义一个Animal类,然后定义1个类属性,代码如下:

    class Animal:

    age = 5

    可以通过2种方式来调用类属性:类名.属性名和实例名.属性名。例如:

    class Animal:

    age = 5

    print(Animal.age)# 类名调用属性,输出结果为 5

    dog =Animal()# 实例化Animal类

    print(dog.age)# 实例名调用属性,输出结果为 5

    cat =Animal()# 实例化Animal类

    print(cat.age)# 实例名调用属性,输出结果为 5

    上述代码中,Animal类有一个属性age,所以Animal.age的值为5。但是Animal的2个实例dog和cat为什么也能够获取age属性呢?这就是属性继承搜索的功能。下面的类树图可以很好的解释这个问题。

    上图中,Animal类有一个属性age,并且通过Animal类创建了2个实例dog和cat。当调用dog.age时,由于age实例的age属性不存在,所以会继续向上搜索到Animal,发现Animal类中有age属性,所以,dog.age为5。

    注意:如果调用dog.name,由于age实例和Animal类都没有这个属性,所以提示如下错误信息:AttributeError:'Animal' object has no attribute 'name'

    创建一个Animal类,该类有一个属性age,并创建2个实例dog和cat,如果更改Animal类age属性的初始值,那么,dog.age和cat.age是否会发生变化呢?下面通过一个例子来学习一下。

    class Animal:

    age = 5

    dog = Animal() # 实例化Animal类

    cat = Animal() # 实例化Animal类

    dog.age = 10

    Animal.age = 8

    print('Animal类的age属性值为%s' % Animal.age) # 类名调用属性,输出结果为 8

    print('dog实例的age属性值为%s' % dog.age) # 实例名调用属性,输出结果为 10

    print('cat实例的age属性值为%s' % cat.age) # 实例名调用属性,输出结果为 8

    运行结果如下。

    Animal类的age属性值为8

    dog实例的age属性值为10

    cat实例的age属性值为8

    上述代码中,对于dog实例,动态为其添加属性age,dog.age在查找时,优先查找dog实例的age,所以结果为10。接下来,动态更改Animal类的age属性值,所以Animal.age的值变为8。而由于cat实例没有age属性,所以向上查找Animal类的属性,此时,由于Animal的age属性值已经变为8,所以cat.age为8。通过下图可以更好理解赋值过程。

    2.实例属性

    实例属性是指定义在类的方法中的属性,只作用于当前实例中。在18.2.4节创建__init__()方法时,就使用了实例属性。

    创建一个Animal类,该类中有3个方法,__init__()初始化方法,info()输出详细信息方法以及get_gender()判断动物性别的方法。

    在__init__()初始化方法中,使用self.name、self.gender和self.weight设置实例属性。

    class Animal: # 创建动物类

    '''动物类'''

    def __init__(self,name,gender,weight):

    '''初始化赋值'''

    self.name = name

    self.gender = gender

    self.weight = weight

    def info(self):

    '''输出基本信息'''

    gender = self.get_gender(self.gender)

    print('这只%s是%s的' % (self.name,gender))

    print('%s的重量是%s千克' % (self.name , self.weight))

    def get_gender(self,gender):

    '''判断动物的性别'''

    if self.gender == 'male':

    gender = '公'

    else :

    gender = '母'

    return gender

    dog = Animal('腊肠狗','male',10) # 创建动物类的实例

    dog.info() # 调用实例的info方法,输出基本信息

    cat = Animal('波斯猫','female',5) # 创建动物类的实例

    cat.info() # 调用实例的info方法,输出基本信息

    运行结果如下所示。

    这只腊肠狗是公的

    腊肠狗的重量是10千克

    这只波斯猫是母的

    波斯猫的重量是5千克

    相关文章

      网友评论

        本文标题:面向对象进阶版!从入门到大牛!Python就是这么牛逼!

        本文链接:https://www.haomeiwen.com/subject/kwcagftx.html