试图了解使用对象和类的简单python游戏的流程和逻辑

时间:2013-12-31 00:22:37

标签: python oop

我正在学习“学习Python困难之路”教程。我已经做了一些练习,试图真正了解Zed的(作者)OOP示例游戏(练习43)。

谨慎,我认为我明白了 - 什么类和对象是什么。我理论上至少要了解鱼和鲑鱼,汽车和本田等之间的区别等。

我知道我们使用类和对象(而不是只按需编写函数)以重用代码(对吗?)。

我的思想链条现在很微妙,但我不能休息,睡觉或感觉满足,直到我真正点击它。

游戏中有许多场景(场景是游戏中的一个类)。例如。中央走廊,大桥,Escape Pod。

除了场景,游戏还使用引擎和地图(对象)。我完全不了解这些组件如何使游戏运行。

在剧本的最后,开始游戏的是代码块:

a_map = Map('central_corridor')
a_game = Engine(a_map)
a_game.play() 

具体问题:

  1. 游戏如何从一个场景移动到另一个场景?
  2. 引擎和地图类如何在游戏中发挥作用?即引擎和地图有什么作用?我无法完全遵循逻辑。理想情况下,这可以用一个例子来解释,例如从TheBridge转移到EscapePod的代码中的步骤(游戏流程中的最后两个场景)
  3. 我的陈述是否正确:“我理解我们使用类和对象(而不是仅按需编写函数)以重用代码。”
  4. 这是游戏:

    from sys import exit
    from random import randint
    
    class Scene(object):
        def enter(self):
            print "This scene is not yet configured. Subclass it and implement enter()."
    
    class Engine(object):
    
        def __init__(self, scene_map):
            print "Engine __init__ has scene_map", scene_map
            self.scene_map = scene_map
    
        def play(self):
            current_scene = self.scene_map.opening_scene()
            print "Play's first scene", current_scene
    
            while True:
                print "\n--------"
                next_scene_name = current_scene.enter()
                print "next scene", next_scene_name
                current_scene = self.scene_map.next_scene(next_scene_name)
                print "map returns new scene", current_scene
    
    class Death(Scene):
    
        quips = [
            "You died. You kinda suck at this.",
            "Your mom would be proud... if she were smarter",
            "Such a luser",
            "I have a small puppy that's better at this."
            ]
    
        def enter(self):
            print Death.quips[randint(0, len(self.quips)-1)]
            exit(1)
    
    class CentralCorridor(Scene):
    
        def enter(self):
            print "The Gothons of Planet Percal #25 have invaded your ship and destroyed"
            print "your entire crew.  You are the last surviving member and your last"
            print "mission is to get the neutron destruct bomb from the Weapons Armory,"
            print "put it in the bridge, and blow the ship up after getting into an "
            print "escape pod."
            print "\n"
            print "You're running down the central corridor to the Weapons Armory when"
            print "a Gothon jumps out, red scaly skin, dark grimy teeth, and evil clown costume"
            print "flowing around his hate filled body.  He's blocking the door to the"
            print "Armory and about to pull a weapon to blast you."
    
            action = raw_input("> ")
    
            if action == "shoot!":
                print "Quick on the draw you yank out your blaster and fire it at the Gothon."
                print "His clown costume is flowing and moving around his body, which throws"
                print "off your aim.  Your laser hits his costume but misses him entirely.  This"
                print "completely ruins his brand new costume his mother bought him, which"
                print "makes him fly into an insane rage and blast you repeatedly in the face until"
                print "you are dead.  Then he eats you."
                return 'death'
    
            elif action == "dodge!":
                print "Like a world class boxer you dodge, weave, slip and slide right"
                print "as the Gothon's blaster cranks a laser past your head."
                print "In the middle of your artful dodge your foot slips and you"
                print "bang your head on the metal wall and pass out."
                print "You wake up shortly after only to die as the Gothon stomps on"
                print "your head and eats you."
                return 'death'
    
            elif action == "tell a joke":
                print "Lucky for you they made you learn Gothon insults in the academy."
                print "You tell the one Gothon joke you know:"
                print "Lbhe zbgure vf fb sng, jura fur fvgf nebhaq gur ubhfr, fur fvgf nebhaq gur ubhfr."
                print "The Gothon stops, tries not to laugh, then busts out laughing and can't move."
                print "While he's laughing you run up and shoot him square in the head"
                print "putting him down, then jump through the Weapon Armory door."
                return 'laser_weapon_armory'
    
            else:
                print "DOES NOT COMPUTE!"
                return 'central_corridor'
    
    class LaserWeaponArmory(Scene):
    
        def enter(self):
            print "You do a dive roll into the Weapon Armory, crouch and scan the room"
            print "for more Gothons that might be hiding.  It's dead quiet, too quiet."
            print "You stand up and run to the far side of the room and find the"
            print "neutron bomb in its container.  There's a keypad lock on the box"
            print "and you need the code to get the bomb out.  If you get the code"
            print "wrong 10 times then the lock closes forever and you can't"
            print "get the bomb.  The code is 3 digits."
            code = "%d%d%d" % (randint(1,9), randint(1,9), randint(1,9))
            guess = raw_input("[keypad]> ")
            guesses = 0
    
            while guess != code and guesses < 10:
                print "BZZZZEDDD!"
                guesses += 1
                guess = raw_input("[keypad]> ")
    
            if guess == code:
                print "The container clicks open and the seal breaks, letting gas out."
                print "You grab the neutron bomb and run as fast as you can to the"
                print "bridge where you must place it in the right spot."
                return 'the_bridge'
            else:
                print "The lock buzzes one last time and then you hear a sickening"
                print "melting sound as the mechanism is fused together."
                print "You decide to sit there, and finally the Gothons blow up the"
                print "ship from their ship and you die."
                return 'death'
    
    class TheBridge(Scene):
    
        def enter(self):
            print "You burst onto the Bridge with the netron destruct bomb"
            print "under your arm and surprise 5 Gothons who are trying to"
            print "take control of the ship.  Each of them has an even uglier"
            print "clown costume than the last.  They haven't pulled their"
            print "weapons out yet, as they see the active bomb under your"
            print "arm and don't want to set it off."
    
            action = raw_input("> ")
    
            if action == "throw the bomb":
                print "In a panic you throw the bomb at the group of Gothons"
                print "and make a leap for the door.  Right as you drop it a"
                print "Gothon shoots you right in the back killing you."
                print "As you die you see another Gothon frantically try to disarm"
                print "the bomb. You die knowing they will probably blow up when"
                print "it goes off."
                return 'death'
    
            elif action == "slowly place the bomb":
                print "You point your blaster at the bomb under your arm"
                print "and the Gothons put their hands up and start to sweat."
                print "You inch backward to the door, open it, and then carefully"
                print "place the bomb on the floor, pointing your blaster at it."
                print "You then jump back through the door, punch the close button"
                print "and blast the lock so the Gothons can't get out."
                print "Now that the bomb is placed you run to the escape pod to"
                print "get off this tin can."
                return 'escape_pod'
            else:
                print "DOES NOT COMPUTE!"
                return "the_bridge"
    
    class EscapePod(Scene):
    
        def enter(self):
            print "You rush through the ship desperately trying to make it to"
            print "the escape pod before the whole ship explodes.  It seems like"
            print "hardly any Gothons are on the ship, so your run is clear of"
            print "interference.  You get to the chamber with the escape pods, and"
            print "now need to pick one to take.  Some of them could be damaged"
            print "but you don't have time to look.  There's 5 pods, which one"
            print "do you take?"
    
            good_pod = randint(1,5)
            guess = raw_input("[pod #]> ")
    
    
            if int(guess) != good_pod:
                print "You jump into pod %s and hit the eject button." % guess
                print "The pod escapes out into the void of space, then"
                print "implodes as the hull ruptures, crushing your body"
                print "into jam jelly."
                return 'death'
            else:
                print "You jump into pod %s and hit the eject button." % guess
                print "The pod easily slides out into space heading to"
                print "the planet below.  As it flies to the planet, you look"
                print "back and see your ship implode then explode like a"
                print "bright star, taking out the Gothon ship at the same"
                print "time.  You won!"
    
                return 'finished'
    
    class Map(object):
    
        scenes = {
            'central_corridor': CentralCorridor(),
            'laser_weapon_armory': LaserWeaponArmory(),
            'the_bridge': TheBridge(),
            'escape_pod': EscapePod(),
            'death': Death()
        }
    
        def __init__(self, start_scene):
            self.start_scene = start_scene
            print "start_scene in __init__", self.start_scene
    
        def next_scene(self, scene_name):
            print "start_scene in next_scene"
            val = Map.scenes.get(scene_name)
            print "next_scene returns", val
            return val
    
        def opening_scene(self):
            return self.next_scene(self.start_scene)
    
    a_map = Map('central_corridor')
    a_game = Engine(a_map)
    a_game.play()
    

2 个答案:

答案 0 :(得分:5)

  1. 游戏的关键部分是这段代码:

    while True: 
        next_scene_name = current_scene.enter()  # <-- returns after interaction with user
        current_scene = self.scene_map.next_scene(next_scene_name)
    

    每个场景的enter方法返回要进入的下一个场景的名称(取决于玩家的动作)......这可能是同一个场景。然后next_scene类的Map方法返回与该名称匹配的场景实例,我们将其分配给current_scene变量...然后它循环并“进入”新的当前场景。这将无休止地继续,或直到您到达退出该计划的Death场景。

  2. 上面基本上解释了Engine类。恕我直言,没有必要成为一个类,只有一个简单的play函数,如同以下那样有效(并且更简单,因此更好):

    def play(scene_map):
        current_scene = scene_map.opening_scene()
        print "Play's first scene", current_scene
    
        while True:
            print "\n--------"
            next_scene_name = current_scene.enter()
            print "next scene", next_scene_name
            current_scene = scene_map.next_scene(next_scene_name)
            print "map returns new scene", current_scene
    

    并开始运行游戏:

    a_map = Map('central_corridor')
    play(a_map)
    

    Map类将场景对象实例与场景名称(如'central_corridor')相匹配......类的核心是包含实际映射的字典。您可以看到该类的next_scene方法只对dict键执行了get。实际上你可以通过消除Map类来简化代码,只需要一个简单的dict,将琐碎的起始场景逻辑移动到Engine类(或play方法中,如上所述)。

    然而,在Map类的情况下,我建议通过定义 API 来提供另一个有用的目的...换句话说,一组相关的方法使用游戏地图。这意味着如果你意识到你需要一个更复杂的Map逻辑实现而不仅仅是一个dict,那么在你的游戏开发之后,只要你保持API相同(即方法的名称和参数),你就不要我必须改变你的其余代码。

  3. 一个类定义了一组行为并为它们提供了一个接口。对象是类的实例,它是运行时状态的集合(即类的属性的特定值)。

    这不是关于代码重用的。代码重用意味着拥有从父级继承行为的子类(而不是必须复制和粘贴该行为),但是您可以通过使用类来获取值而不进行任何子类化。类就像用于创建具有共同行为的对象的模板。

    在目前的游戏代码中,一些类的使用是毫无意义的。例如,Engine类...您使用Map实例对其进行初始化并调用其play方法,但由于您只有Engine的单个实例没有必要在类中封装该状态(self.scene_map属性) - 您可以在调用它时将该状态传递给play方法。

    类似于Scene类及其子类。是的,Scene类定义了一个接口(“场景将有enter方法”)但由于每个场景都不包含任何唯一状态(即self上的属性),因此不需要有一个实例化的对象,你可以通过简单的函数更简单地实现同样的事情。

    希望我们的目的是在本教程的未来阶段将这些类开发成更有用的结构,从而证明使用类。

  4. 您可能希望进一步探讨这些术语的含义:
    http://en.wikipedia.org/wiki/Class_(computer_science)
    http://en.wikipedia.org/wiki/Object_(computer_science)

    关于SO的类似问题:
    Confused about classes in Learn Python the Hard Way ex43?

答案 1 :(得分:4)

<强> 1。游戏如何从一个场景移动到另一个场景?

<强> 2。引擎和地图类如何在游戏中运行?即引擎和地图有什么作用?

可能最好一起回答这两个问题。

Engine通过play()方法设置游戏流程。在该方法中while循环的每次迭代,您可能会认为在游戏术语中是单个“转弯”或“回合”。在每个不同的术语/回合中,以下事项按以下顺序发生:

  • 印有虚假横向规则;
  • 下一个场景的名称由当前场景的enter()方法决定;
  • 打印描述性行;
  • 下一个场景的名称将提供给scene_map以返回相应的;
  • 打印描述性的行。

while循环的第一,第三和第五行只是为了方便您查看游戏过程中发生的事情。真正的肉在第二和第四行。

游戏如何从一个场景移动到另一个场景?好吧,Engine类只知道Scene会给它一个名字,它会查看scene_map以查看哪个用这个名字。 scene_map基本上是一个字典,其中添加了一些功能,可以向你发送“我正在做什么”,同时它为您提供了由 name 标识的Engine无需了解Scene期间发生的情况,或用户选择下一个Scene的情况;它所做的只是让你沿着每一轮重复的五个步骤前进。

我们如何从一个场景移动到另一个场景取决于我们当前所处的场景。当您编写一个继承自Scene的类时,您必须定义enter()方法,并且你真正需要做的就是确保编写一个功能场景,确保你的enter()方法在某个时刻返回(到Engine)下一个场景的名称。阅读一些场景,当用户提供不同的输入时,您会注意到它们会返回不同的场景名称

当然,如果您的功能场景没有出现在scene_map中,并且没有其他场景可能会返回其名称,从而指示Engine调用其enter()方法,您的功能场景无法访问。

我很快就输入了这个,因为这是一天的结束,但希望我能以一种你认为有用的方式说出一些事情......祝你好运!