美文网首页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