如何使用继承来升级角色

时间:2020-12-19 21:44:22

标签: python-3.x oop object inheritance

# Import Modules
from Dice import dice
d10 = dice(10,1) #I use dice, but you could just as easily use random.randint()
d4 = dice(4,1)
d20 = dice(20,1)

# Assign Classes
class Player_Character:
    inventory = []
    def __init__(self, HP, MaxHP, AC, ToHitAdjustment, Surprise_Adjustment, 
                 Initiative_Adjustment, Exp, \
                 MaxExp, Gold, Damage):
        self.HP = int(HP)
        self.MaxHP = int(MaxHP)
        self.AC = int(AC)
        self.ToHitAdjustment = int(ToHitAdjustment)
        self.Surprise_Adjustment = int(Surprise_Adjustment)
        self.Initiative_Adjustment = int(Initiative_Adjustment)
        self.Exp = int(Exp)
        self.MaxExp = int(MaxExp)
        self.Gold = int(Gold)
        self.Damage = int(Damage)

下一节可能是错误所在,尽管我对此表示怀疑,因为当我运行代码时,角色被分配了 exp 和 gold 并且它们似乎以正确的速度增长。

    def attack(self, goblin, Player_Character):
        Player_attack_roll = d20.die_roll() + Player_Character.ToHitAdjustment
        if (Player_attack_roll >= goblin.AC):
            print('you did', Player_Character.Damage, 'damage')
            goblin.HP -= Player_Character.Damage
            if (goblin.HP <= 0):
                print("congratulations you killed the goblin")
                Player_Character.Exp += goblin.ExpReward
                if (Player_Character.Exp >= Player_Character.MaxExp):
                    print("Congratulations you have leveled up")
                    Player_Character.LevelUp()
                print('you have', Player_Character.Exp, 'exp ')
                Player_Character.Gold += goblin.Gold
                print("you have", Player_Character.Gold, 'gold ')
                del goblin
        else:
            print("You miss ")

    def flee(self):
        print("you run away ")
        quit()

    def heal(self, Player_Character):
        Player_Character.HP += d10.die_roll()
        if Player_Character.HP >= Player_Character.MaxHP:
            Player_Character.HP = Player_Character.MaxHP
            print("You have reached max HP ")

这可能是错误所在,我希望它将之前分配给 Player_Character 的变量名称 'MrHezy' 分配给 FighterLvl1,但是当我运行代码时,它不断说我正在升级,我不确定如果这意味着角色仍然是 Player_Character,或者还有其他错误

    def LevelUp(self):
        MrHezy = FighterLvl1(25, 25, 15, 10, 2, 2, 25, 75, 20, d10.die_roll()+5)

下一节的代码是我的继承问题发挥作用的地方。即使 FighterLvl1 是从 (Player_Character) 继承的,我的 IDE 也会说“FighterLvl1 的未解析属性引用 'Exp'”之类的内容,而且我不确定在 FighterLvl1 下添加 'attack' 方法是否有任何作用;我希望它覆盖 Player_Character 的 'attack' 方法

class FighterLvl1(Player_Character):
    def attack(self, goblin, Player_Character):
        Player_attack_roll = d20.die_roll() + Player_Character.ToHitAdjustment
        if (Player_attack_roll >= goblin.AC):
            print('you did', Player_Character.Damage, 'damage')
            goblin.HP -= Player_Character.Damage
            if (goblin.HP <= 0):
                print("congratulations you killed the goblin")
                FighterLvl1.Exp += goblin.ExpReward
                if (FighterLvl1.Exp >= FighterLvl1.MaxExp):
                    print("Congratulations you have leveled up")
                    FighterLvl1.LevelUp()
                print('you have', FighterLvl1.Exp, 'exp ')
                FighterLvl1.Gold += goblin.Gold
                print("you have", FighterLvl1.Gold, 'gold ')
                del goblin
    def LevelUp(self):
        MrHezy = FighterLvl2(Player_Character)

class FighterLvl2(Player_Character):
    def LevelUp(self):
        MrHezy = FighterLvl3(Player_Character)

class FighterLvl3(Player_Character):
    def MaxLevel(self):
        print("Congratulations you are level 3, this is the highest programmed level in the game 
               ")

class goblin:
    def __init__(self, HP, MaxHP, AC, ToHitAdjustment, Surprise_Adjustment, 
                 Initiative_Adjustment, \
                 ExpReward, Gold, Damage):
        self.HP = int(HP)
        self.MaxHP = int(MaxHP)
        self.AC = int(AC)
        self.ToHitAdjustment = int(ToHitAdjustment)
        self.Surprise_Adjustment = int(Surprise_Adjustment)
        self.Initiative_Adjustment = int(Initiative_Adjustment)
        self.ExpReward = int(ExpReward)
        self.Gold = int(Gold)
        self.Damage = int(Damage)

    def attack(self, Player_Character, goblin):
        if (goblin.HP <= 0):
            print("the goblin is inert")
        else:
            goblin_attack_roll = d20.die_roll() + goblin.ToHitAdjustment
            if (goblin_attack_roll >= Player_Character.AC):
                goblin_damage = d4.die_roll()
                print("you take", goblin_damage, 'damage')
                Player_Character.HP -= goblin_damage
                if (Player_Character.HP <= 0):
                    print("oh dear you have died")
                    del Player_Character
                    quit()
            else:
                print("the goblin misses ")

MrHezy = Player_Character(20, 20, 10, 5, 0, 0, 0, 25, 0, d10.die_roll())


def spawn_goblin(goblin):
    G1 = goblin(5, 10, 8, 2, 0, 0, 25, 5, d4.die_roll())
    return G1

goblin1 = spawn_goblin(goblin)

def battle(goblin1):

    # user input
    player_initiative_adjustment = MrHezy.Initiative_Adjustment
    monster_initiative_adjustment = goblin1.Initiative_Adjustment

    #define while loop for the battle
    while True:
        if (goblin1.HP <= 0):
            print('oh dear looks like we need a new goblin')
            break
        #use random.randint(a,b) to generate player and monster base initiative
        player_base_initiative = d10.die_roll()
        monster_base_initiative = d10.die_roll()

        #subtract the adjustment to get player and monster initiative
        player_initiative = player_base_initiative - player_initiative_adjustment
        monster_initiative = monster_base_initiative - monster_initiative_adjustment

        #compare the initiatives and display the results
        if (player_initiative < monster_initiative):
            attack_flee_heal = input("congratulations you go first. Would you like to attack, 
                                      flee, or heal?")
            if attack_flee_heal == 'attack':
                MrHezy.attack(goblin1, MrHezy)
            elif attack_flee_heal == 'heal':
                MrHezy.heal(MrHezy)
                print("the goblin attacks")
                goblin1.attack(MrHezy)
            elif attack_flee_heal == 'flee':
                MrHezy.flee()
        else:
            print("uhoh, the monsters go first, they attack!")
            goblin1.attack(MrHezy, goblin1)
            attack_flee_heal = input("Would you like to attack, flee, or heal? ")
            if attack_flee_heal == 'attack':
                MrHezy.attack(goblin1, MrHezy)
            elif attack_flee_heal == 'heal':
                MrHezy.heal(MrHezy)
                print("the goblin attacks")
                goblin1.attack(MrHezy, goblin1)
            elif attack_flee_heal == 'flee':
                MrHezy.flee()



#main game loop
while True:
    spawn_goblin(goblin)
    battle(goblin1)
    print("spawn another goblin")
    goblin1.HP = 0
    goblin1.HP += d10.die_roll()

此代码运行良好,请将其输入到您的 IDE 中。 我遇到的问题是一个逻辑错误,当一个地精被杀死而不是先将角色升级到 1 级,然后是 2 级,然后是 3 级,而不是角色一遍又一遍地升级到 1 级。

1 个答案:

答案 0 :(得分:0)

升级替换对象中的一个变量,你应该怎么做,如果你想每个级别使用一个类(不一定是一个好主意)是:

创建一个 Character 类。

Character 类中定义一个 implementation 字段。

初始化 implementation 字段,例如implementation = SomeSpeciesLevel1()

要“升级”:implementation = implementation.level_up()

这样你就有了一个不变的包装器,它不会改变它包装了级别变化的实现。

更好的设计是让一个类具有基于级别相关公式的功能。

class CharImpl:
    @abstractmethod
    def can_level_up() -> bool:
        return NotImplemented
    @abstractmethod
    def next_level() -> CharImpl:
        return NotImplemented
    pass

class Character:
    def __init__(initial_impl: CharImpl=None):
        self.impl = initial_impl.copy() if initial_impl else HumanLevel1()
        pass
    def level_up():
        if self.impl.can_level_up():
            self.impl = self.impl.next_level()
        pass
    pass

class Human(CharImpl):
    pass

class HumanLevel1(Human):
    def __init__(src: Human):
        # TODO: copy relevant fields from src
        pass
    def can_level_up() -> bool:
        return True
    def next_level() -> CharImpl:
        return HumanLevel2(self)
    pass

class HumanLevel2(Human):
    def __init__(src: Human):
        # TODO: copy relevant fields
        pass
    def can_level_up() -> bool:
        return False
    def next_level() -> CharImpl:
        return self
    pass