组合

  给一个类的对象封装一个属性,这个属性是另一个类的对象,这样我们在调用这个属性时就指向了另一个类的对象,这样我们就可以调用另一个类的方法。

模拟英雄联盟写一个游戏人物的类
# 要求:
# (1)创建一个 Game_role的类.
# (2) 构造方法中给对象封装name,ad(攻击力),hp(血量).三个属性.
# (3) 创建一个attack方法,此方法是实例化两个对象,互相攻击的功能:
# 例: 实例化一个对象 盖伦,ad为10, hp为100
# 实例化另个一个对象 剑豪 ad为20, hp为80
# 盖伦通过attack方法攻击剑豪,此方法要完成 '谁攻击谁,谁掉了多少血, 还剩多少血'的提示功能.

class LolRole:

    def __init__(self, name, ad, hp):  # 初始化对象属性
        self.name = name
        self.ad = ad
        self.hp = hp

    def attack(self, enemy):  # 攻击方法,除了攻击者本身还要有敌人(enemy)对象作为参数传入
        enemy.hp = enemy.hp - self.ad
        print('{0}攻击了{1},{1}还剩{2}血'.format(self.name, enemy.name, enemy.hp))


gailun = LolRole('盖伦', 10, 100)  # 实例化盖伦对象,包括名字,攻击力,血量三个属性

yasuo = LolRole('亚索', 20, 80)  # 实例化亚索对象

gailun.attack(yasuo)  # 盖伦攻击亚索

# 运行结果:盖伦攻击了亚索,亚索还剩70血

现在我有一个weapen武器的类,需要给亚索和盖伦装备上,用武器攻击对方掉的血量则是武器的攻击力

class LolRole:

    def __init__(self, name, ad, hp):  # 初始化对象属性
        self.name = name
        self.ad = ad
        self.hp = hp

    def attack(self, enemy):  # 攻击方法,除了攻击者本身还要有敌人(enemy)对象作为参数传入
        enemy.hp = enemy.hp - self.ad
        print('{0}攻击了{1},{1}还剩{2}血'.format(self.name, enemy.name, enemy.hp))


gailun = LolRole('盖伦', 10, 100)  # 实例化盖伦对象,包括名字,攻击力,血量三个属性

yasuo = LolRole('亚索', 20, 80)  # 实例化亚索对象

gailun.attack(yasuo)  # 盖伦攻击亚索


# 运行结果:盖伦攻击了亚索,亚索还剩70血

class Weapen:
    def __init__(self, name, ad):  # 实例化武器
        self.name = name
        self.ad = ad

    def weapen_attack(self, role, enmey):  # 拿着武器攻击
        enmey.hp = enmey.hp - self.ad
        print('{0}拿着{1}攻击了{2},{2}掉了{3}血,还剩{4}血'
              .format(role.name, self.name, enmey.name, self.ad, enmey.hp))


sword = Weapen('大保剑', 30)  # 实例化武器
katana = Weapen('草雉剑', 50)

sword.weapen_attack(gailun, yasuo)

上面定了weapen类,但是你会发现我们用武器攻击人时,行动的主体是武器而不是人,人的类是调用不了武器的方法,那么我们要如何实现是人拿着武器去攻击呢,这就用到组合的概念,我们可以将weapen实例出的对象作为属性添加到我人类中,这样我人的类在调用这个属性时,就对应了weapen的对象,而武器的对象可以调用武器类的方法,这样我们就间接的让人类的对象使用了武器类的方法。

实现代码如下:

class LolRole:

    def __init__(self, name, ad, hp):  # 初始化对象属性
        self.name = name
        self.ad = ad
        self.hp = hp

    def attack(self, enemy):  # 攻击方法,除了攻击者本身还要有敌人(enemy)对象作为参数传入
        enemy.hp = enemy.hp - self.ad
        print('{0}攻击了{1},{1}还剩{2}血'.format(self.name, enemy.name, enemy.hp))

    def equipment_weapen(self, w):  # w接受实例化的武器,这样相当于在人实例化的对象中添加了weapen这个属性这个属性对应武器的对象
        self.weapen = w


class Weapen:
    def __init__(self, name, ad):  # 实例化武器
        self.name = name
        self.ad = ad

    def weapen_attack(self, role, enmey):  # 拿着武器攻击
        enmey.hp = enmey.hp - self.ad
        print('{0}拿着{1}攻击了{2},{2}掉了{3}血,还剩{4}血'
              .format(role.name, self.name, enmey.name, self.ad, enmey.hp))


sword = Weapen('大保剑', 30)  # 实例化武器
katana = Weapen('草雉剑', 50)
gailun
= LolRole('盖伦', 10, 100) # 实例化盖伦对象,包括名字,攻击力,血量三个属性 yasuo = LolRole('亚索', 20, 80) # 实例化亚索对象

print(gailun.__dict__) # {'name': '盖伦', 'ad': 10, 'hp': 100}

# 将大保健交给盖伦 gailun.equipment_weapen(sword)
print(gailun.__dict__) # {'name': '盖伦', 'ad': 10, 'hp': 100, 'weapen': <__main__.Weapen object at 0x05934450>} # 可以看见此时的gailun这个对象已经添加了这个weapen这个属性,这个属性就是实例化的武器的对象大宝剑 # 这样我们调用 gailun.weapen就等于sword。

gailun.weapen.weapen_attack(gailun, yasuo) # 等价于sword.weapen_attack(gailun,yasuo) # 盖伦拿着大保剑攻击了亚索,亚索掉了30血,还剩50血
03-16 14:28