第九章 类 (1)

前言

面向对象的编程是最新有效的软件编写方法之一。在面向对象编程中,你编写表示现实世界中的事物和情景的类,并基于这些类来创建对象。编写类时,我们定义的一个类都有的通用行为。基于类创建对象时,每个对象都具有自动具备这种通用行为,然后可以根据需要赋予每个对象独特的个性。使用面向对象编程课模拟现实情境,

9.1创建和使用类

9.1.1 创建dog类

class Dog:
    """一次模拟小狗的简单尝试"""
    
    def _——init__(self, name, age):
        """初始化属性 name和age"""
        self.name = name
        self.age = age
        
    def sit(self):
        """模拟小狗收到命令时蹲下"""
        print(f"{self.name} is now sitting.")
    
    def roll_over(self):
        """模拟小狗收到命令时打滚"""
        print(f"{self.name} rolled over!")

开头定义了一个名为Dog的类,根据约定,在python中,首字母大写的名称指的是类,这个类定义中没有圆括号,因为要从空白创建这个类。
方法 __ init__ ()
类中的函数称为方法,我们在之前学到的有关函数的一切都适用于方法,就目前而言唯一重要的差别是调用方法的方式__ init__ ()是一个特殊的方法,每当我们根据Dog类创建新的实例的时候,python就会自动运行它。在这个方法名称中,开头和结尾有两个下划线这是一种约定,旨在避免python默认方法于普通方法发生名称冲突。务必确保__ init __()的两边都有两个下划线,否则当使用类来创建实例的时候,将不会自动调用这个方法,进而引发难以发现的错误。

我们将方法 __ init __ ()定义成包含三个形参:self、name和age。在这个方法的定义中,形参self是必不可少的,且必须位于其他形参的前面。为何必须在方法定义中包含形参self呢?因为python调用这个方法来创建Dog实例的时候,将自动传入实参self,每个与实例想关联的方法调用都自动传入实参self,它是一个指向实例本身的引用,让实例相关联的方法调用都自动传递实参self,它是一个指向实例本身的引用,让实例能够访问类中的属性和方法。创建Dog实例的时候,python将调用Dog类的方法 __ init __ ()。我们将通过实参向Dog()传递名字和年龄,self会自动传递它。每当根据Dog类创建实例的时候,都只需要给最后两个形参(name和age)提供值。

self.name;self.age处定义的两个变量都有前缀self。以self为前缀的变量可以提供类中的所有方法的使用,可以通过类的任何实例来访问。self.name = name 获取与形参name相关联的值,并将其赋给变量name,然后该变量被关联到当前创建的实例。像这样可以通过实例访问的变量成为属性。

Dog类还定义了另外两个方法:sit()和roll_over()。这些方法执行时不需要额外的信息,因此它们只有一个形参self

9.1.2 根据类创建实例

my_dog = Dog('Willie',6)

print(f"My dog's name is {my_dog.name}.")
print(f"My dog is {my_dog.age} years old")
My dog's name is Willie.
My dog is 6 years old

1.访问属性
要访问实例的属性,可使用句点表示法。
句点表示法在python中很常用,这种语法演示了python如何获悉属性的值。在这里,python先找到实例my_dog,再查找与该实例相关联的属性name。

2.调用方法

my_dog.sit()
my_dog.roll_over()
Willie is now sitting.
Willie rolled over!

3.创建多个实例
可按照需求根据类创建任意数量的实例。

my_dog = Dog('Willie',6)
your_dog = Dog('Lucy',3)

print(f"My dog's name is {my_dog.name}:")
print(f"My dog is {my_dog.age} years old")
my_dog.sit()

print(f"Your dog's name is {your_dog.name}:")
print(f"Your dog is {your_dog.age} years old")
your_dog.sit()
My dog's name is Willie:
My dog is 6 years old
Willie is now sitting.
Your dog's name is Lucy:
Your dog is 3 years old
Lucy is now sitting.

9.2.1 Car类

class Car:
    """一次模拟汽车的简单尝试"""
    def __init__(self,make,model,year):
        """初始化描述汽车的属性"""
        self.make = make
        self.model = model
        self.year = year
        
    def get_descriptive_name(self):
        """返回整洁的描述信息"""
        long_name = f"{self.year} {self.make} {self.model}"
        return long_name.title()
my_new_car = Car('audi','a4',2019)
print(my_new_car.get_descriptive_name())
2019 Audi A4

9.2.2给属性指定默认值

创建实例的时候,有些属性无法通过形参来定义,可在方法 __ init __ ()中为其指定默认值。

class Car:
    """一次模拟汽车的简单尝试"""
    def __init__(self,make,model,year):
        """初始化描述汽车的属性"""
        self.make = make
        self.model = model
        self.year = year
        self.odomter_reading = 0
        
    def get_descriptive_name(self):
        """返回整洁的描述信息"""
        long_name = f"{self.year} {self.make} {self.model}"
        return long_name.title()
    def read_odmeter(self):
        """打印一条指出汽车里程的消息"""
        print(f"This car has {self.odomter_reading} miles on it.")
        
my_new_car = Car('audi','a4',2019)
print(my_new_car.get_descriptive_name())
my_new_car.read_odmeter()
2019 Audi A4
This car has 0 miles on it.

9.2.3 修改属性的值

我们能通过三种方式来修改属性的值:直接通过实例进行修改,通过方法进行设置,以及通过方法进行递增(增加特定的值),下面依次介绍这些方式:

1.直接修改属性的值

要修改属性的值,最简单的方式就是通过实例直接访问它。

my_new_car.odomter_reading = 23
my_new_car.read_odmeter()
This car has 23 miles on it.

使用句点表示法直接访问并设置汽车的属性odomete_reading,这行代码让python在实例my_new_car中找到属性oddometer_reading,并将其设置为23。

2.通过方法修改属性的值

如果有方法能替你更新属性,将大有裨益。这样就无须直接访问属性,而可将值传递给方法,由它在内部进行更新。

class Car:
    """一次模拟汽车的简单尝试"""
    def __init__(self,make,model,year):
        """初始化描述汽车的属性"""
        self.make = make
        self.model = model
        self.year = year
        self.odomter_reading = 0
        
    def get_descriptive_name(self):
        """返回整洁的描述信息"""
        long_name = f"{self.year} {self.make} {self.model}"
        return long_name.title()
    
    def read_odmeter(self):
        """打印一条指出汽车里程的消息"""
        print(f"This car has {self.odomter_reading} miles on it.")
        
    def update_odometer(self,mileage):
        """将里程表读数设置为指定的值"""
        self.odomter_reading = mileage
        
my_new_car = Car('audi','a4',2019)
print(my_new_car.get_descriptive_name())
my_new_car.update_odometer(23)
my_new_car.read_odmeter()
2019 Audi A4
This car has 23 miles on it.

对Car类所做的唯一修改是在类中加入了方法up_odometer().这个方法接受一个里程值,并将其赋给self.odometer_reading.我们在这里对update_odometer()进行拓展,使其在修改里程表读数时做一些额外的工作。下面添加一些逻辑,禁止任何人将里程表读数往回调:

class Car:
    """一次模拟汽车的简单尝试"""
    def __init__(self,make,model,year):
        """初始化描述汽车的属性"""
        self.make = make
        self.model = model
        self.year = year
        self.odomter_reading = 24
        
    def get_descriptive_name(self):
        """返回整洁的描述信息"""
        long_name = f"{self.year} {self.make} {self.model}"
        return long_name.title()
    
    def read_odmeter(self):
        """打印一条指出汽车里程的消息"""
        print(f"This car has {self.odomter_reading} miles on it.")
        
    def update_odometer(self,mileage):
        """将里程表读数设置为指定的值。
           禁止将里程表读数往回调
        """
        if mileage >= self.odomter_reading:
             self.odomter_reading = mileage
        else:
            print("You can't roll back and odometer!")
        
my_new_car = Car('audi','a4',2019)
print(my_new_car.get_descriptive_name())
my_new_car.update_odometer(25)
my_new_car.read_odmeter()
2019 Audi A4
This car has 25 miles on it.

3.通过方法对属性的值进行递增

有时候需要将属性值递增特定的值,而不是将其设置为全新的值。假设我们购买了一辆二手车,且从购买到登记期间增加了100英里的里程,下面的方法让我们能够传递这个增量,并相应地增大里程表读数。

class Car:
    """一次模拟汽车的简单尝试"""
    def __init__(self,make,model,year):
        """初始化描述汽车的属性"""
        self.make = make
        self.model = model
        self.year = year
        self.odomter_reading = 24
        
    def get_descriptive_name(self):
        """返回整洁的描述信息"""
        long_name = f"{self.year} {self.make} {self.model}"
        return long_name.title()
    
    def read_odmeter(self):
        """打印一条指出汽车里程的消息"""
        print(f"This car has {self.odomter_reading} miles on it.")
        
    def update_odometer(self,mileage):
        """将里程表读数设置为指定的值。
           禁止将里程表读数往回调
        """
        if mileage >= self.odomter_reading:
             self.odomter_reading = mileage
        else:
            print("You can't roll back and odometer!")
            
    def increment_odometer(self,miles):
        """将里程表读数增加到指定的量"""
        self.odomter_reading += miles
        
my_used_car = Car('audi','a4',2019)
print(my_new_car.get_descriptive_name())
my_used_car.update_odometer(23_500)
my_used_car.read_odmeter()
my_used_car.increment_odometer(100)
my_used_car.read_odmeter()
2019 Audi A4
This car has 23500 miles on it.
This car has 23600 miles on it.
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值