CS200: Computer Science, Spring 2002 |
Problem Set 6: Adventures in Charlottansville Out: 8 March 2002
Due: 25 March 2002, before class
Turn-in Checklist: On March 25, bring to class a stapled turn in containing:
- Your written answers to Question 1, Question 3 and Question 8 (including a sample interaction).
- All the code you wrote or modified for this problem set. Be sure to clearly mark the code for each question. Do not turn in long printouts of code you did not modify.
Collaboration Policy - Read Carefully
For this problem set, you may either work alone and turn in a problem set with just your name on it, or work with one other student in the class of your choice. If you work with a partner, you and your partner should turn in one assignment with both of your names on it. For the final question (Question 8), you may combine efforts with as many students as you wish.
Regardless of whether you work alone or with a partner, you are encouraged to discuss this assignment with other students in the class and ask and provide help in useful ways. You may consult any outside resources you wish including books, papers, web sites and people. If you use resources other than the class materials, indicate what you used along with your answer.
Purpose
Downloads
- Program with objects
- Develop and use methods
- Understand and use inheritance
- Make an adventure game
- listprocs.ss — the list procedures we have defined so far in class
- objects.ss — code that defines objects. Defines classes for people, places and things.
- adventure.ss — code describing the imaginary world of the game and playing
Background
In the 1961, Digital Equipment Corporation (now Compaq) donated a PDP-1 mini-computer to MIT, hoping they would use it to solve important scientific problems. Instead, Steve Russell invented the first computer game: Spacewar!. Ten years later, Will Crowther produced the first interactive fiction game, Colossal Cave Adventure (or Adventure for short). Inspired by Adventure, a group of MIT students created Zork and released a version for the Apple II. The game became wildly popular, and was the cause of many students failing out of school.For this problem set, you will create your own adventure game. Our game takes place in a far-off fictional land known as “The University” in Charlottansville, East Virginia. Set on bucolic grounds of rolling hills, grassy fields, and red-brick buildings, the characters in our adventure game will take us to imaginary places like the Cabal Hall, Cdrs Hill, the Recursa, and Oldbrushe Hall. Programming an adventure game involves modeling objects in a fictional world. Hence, you will build your adventure game using techniques known as object-oriented programming.
Q: So some of the locations in the game are based on real places?
A: Except for a few. As far as I know, there is no eldricth altar at which students are sacrificed to nameless gods. But then, I was never a professor, so I can't be sure.Dave Lebling, on The Lurking Horror adventure game Note: All of the places and characters in our game are purely fictional (although most are not purely functional, since they use mutation). Any similarity to persons or places real or imagined is purely coincidental.
Object-Oriented Programming
By the word operation, we mean any process which alters the mutual relation of two or more things, be this relation of what kind it may. This is the most general definition, and would include all subjects in the universe. Again, it might act upon other things besides number, were objects found whose mutual fundamental relations could be expressed by those of the abstract science of operations, and which should be also susceptible of adaptations to the action of the operating notation and mechanism of the engine... Supposing, for instance, that the fundamental relations of pitched sounds in the science of harmony and of musical composition were susceptible of such expression and adaptations, the engine might compose elaborate and scientific pieces of music of any degree of complexity or extent.Ada, Countess of Lovelace, around 1830 (from Lecture 1)
In all of the previous assignments, we solved problems by dividing them into procedures that could be combined to solve the problem. This is known as procedural programming.
Another way of solving problems it to model them using objects. This approach is particularly well-suited to problems that involve simulating a real (or imagined) world such as a graphical use interface (a simulation of a desktop with objects like files and documents), an astrophysics experiment (a simulation of the universe with objects like stars and galaxies), or an adventure game (a simulation of a fictional world with objects like people and donuts).
As in the real world, an object is an entity that packages state and procedures. We call the state that is part of an object instance variables, and the procedures that are part of an object methods. Methods may give information about the state of an object, modify the state of an object, or direct the object to interact with other objects.
We also need procedures for creating objects. We call these procedures constructors. Once an object is created, it is only manipulated by using the object's methods. We call a method of an object by sending the object a message.
For our game, we need to represent three basic kinds of things: people, places and things. Every object we make will be a person, a place or a thing. We call kinds of objects classes.
We have provided a constructor procedure for creating an object of each class. All objects in our fictional world have names (rumors about a Nameless Field are as yet unsubstantiated), so each of our constructor procedures will take a name parameter. We use quoted symbols for names (see SICP, p. 142 for details on quotation). The three constructors are:
- (make-place name) — create a place with the name symbol. Places cannot move.
- (make-thing name) — create a thing with the name name. Things can move and act, but they are not sentient.
- (make-person name) — create a person with the name name.
To put a new thing or person in our fictional world, we must install it in a place. The procedure (install-object Object Place) installs object Object (which must not be a place) in the place Place (which must be a place).
Once an object is installed, we interact with it using ask to send a message to an object that invokes a method:
All objects in the system are manipulated using message-accepting procedures. These are details in the objects.ss file.
- (ask Object Message Arguments) — invoke the method identified by Message on the Object. The Arguments may be zero of more arguments to the method. These are passed to the method as parameters.
Different objects handle different messages. Here is a partial list of messages you can use. The ask procedure is used to implement each of these. Note that some of the messages have parameters.
In adventure.ss we define some places in our imaginary world. The procedure set-up-charlantansville installs those places and sets up connections between them. For example, the definitions
- (ask Person 'look) — Report on the immediate surroundings.
- (ask Person 'say String) — Say the words in the String (something in double quotes) parameter.
- (ask Person 'have-fit) — Throw a temper tantrum.
- (ask Place 'exits) — Report the directions you can move from a given place.
- (ask Person 'go Direction) — Move in the indicated direction. The Direction is a symbol such as 'north, 'west, 'up or 'down (the valid directions depend on the place the person is currently).
make two places, Cabal-Hall and Bart-Statue. In set-up-charalatansville, we use can-go-both-ways (defined in adventure.ss) to connect Cabal-Hall and Bart-Statue:(define Cabal-Hall (make-place 'Cabal-Hall)) (define Bart-Statue (make-place 'Bart-Statue))(can-go-both-ways Bart-Statue 'south 'north Cabal-Hall)We can experiment with our world by evaluating (set-up-charlottansville) and then asking objects in our world to do things. (The provided file ps6.ss does this when you load it.) For example:> (set-up-charlottansville)
welcome-to-charlottansville
> (ask Cabal-Hall 'exits)
(down north)
> (ask Cabal-Hall 'name)
cabal-hall
> (ask Cabal-Hall 'neighbor-towards 'down)
#<procedure>
> (ask (ask Cabal-Hall 'neighbor-towards 'down) 'name)
steam-tunnel
Question 1: Why does (ask Cabal-Hall 'neighbor-towards 'down) evaluate to a procedure? Our world needs some people in it, so let's create one:
(define JT (make-person 'Jeffus-Thomasson))We need to install the new object in our world:We can also make things and add them to our world. For example, let's create a donut and place it in Cabal-Hall (where JT is now).> (install-object JT Cabal-Hall)
Installing jeffus-thomasson at cabal-hall
installed
> (define donut (make-thing 'donut))
> (install-object donut Cabal-Hall)
Installing donut at cabal-hall
installed
Next, we'll have Mr. Thomasson look around, sees the donut and take it.
> (ask JT 'look)
At cabal-hall: jeffus-thomasson says -- I see donut
At cabal-hall: jeffus-thomasson says -- I can go down north(donut)
> (ask JT 'take donut)
At cabal-hall : jeffus-thomasson says -- I take donut
#t
Question 2: Play. Load ps6.ss in DrScheme. Then, in the interactions window, start making people and moving them around. You don't need to turn anything in for this question, but get a feel for how objects work before moving on. Defining Objects
All the people, places and things in our adventure are objects. The procedure make-object creates a new object. Here is make-object without using any syntactic sugar:(define make-object (lambda (name) (lambda (message) (if (eq? message 'object?) (lambda (self) #t) (if (eq? message 'class) (lambda (self) 'object) (if (eq? message 'name) (lambda (self) name) (if (eq? message 'say) (lambda (self list-of-stuff) (if (not (null? list-of-stuff)) (display-message list-of-stuff)) 'nuf-said) (if (eq? message 'install) (lambda (self . args) 'installed) #f))))))))The name make-object is a procedure that takes one parameter, name. It evaluates to a procedure that takes one parameter, message. When that procedure is applied to a message, what does it evaluate to?
Question 3: For each of the Scheme expressions below, predict what they will evaluate to. Then, try evaluating them in your interactions window. Write an explanation that explains clearly why evaluate the way they do. In particular, if an expression evaluates to #<procedure>, you should explain what procedure it is. a. (make-object 'book)
b. ((make-object 'book) 'name)
c. ((make-object 'book) 'fly)
d. (((make-object 'book) 'name) (make-object 'donkey))
e. (((make-object 'book) 'say) (make-object 'donkey) (list 'hello))
Is the book talking or the donkey?
f. (eq? (make-object 'book) (make-object 'book))All those (if (eq? ...)) expressions in make-object get pretty hard to read, so Scheme provides the case syntactic sugar to write this more conveniently (see the DrScheme help for the details on case, but you can probably figure out what you need to know from this example). Here is the sugary version of make-object — it means exactly the same thing as the previous definition:
(define no-method #f) (define (make-object-sugared name) (lambda (message) (case message ((object?) (lambda (self) #t)) ((class) (lambda (self) 'object)) ((name) (lambda (self) name)) ((say) (lambda (self list-of-stuff) (if (not (null? list-of-stuff)) (display-message list-of-stuff)) 'nuf-said)) ((install) (lambda (self . args) 'installed)) (else no-method))))Interacting with objects by calling them is awkward, so we want to define an ask procedure that sends a message to an object. You should be able to figure out these definitions yourself:(define (get-method object message) (object message)) ;;; Send a message to an object (with optional arguments) ;;; (The ask in object.ss is a bit different to produce better error messages.) (define (ask object message . args) ;;; The . args means to allow zero or ;;; more parameters here, and use ;;; args to refer to all of them. (apply (get-method object message) object args))
Question 4: The say method of make-object takes a list as its second parameter and says everything in that list. Instead of saying a whole list, we might want a method that says just one thing. Define an utter method of make-object that behaves like this: > (define dog (make-object 'spot))You can use (display sym) to output one symbol. The output in this example would be produced by (display 'wuff).
> (ask dog 'utter 'wuff)
wuff
Inheritance
Generic people, places and things are okay, but for an interesting game we need to have people and things that can do special things.Our basic object (procedures produced by applying make-object) provides a say method:
> (define dean (make-speaker))
> (ask dean 'say '(okay people, two lines!))
(okay people, two lines!)
What if we have lots of different kinds of speakers — lecturers, evangelical preachers, rambling drunkards? We want to make them speak different ways. For instance, a lecturer is a kind of speaker, except that she can lecture as well as say. When lecturing, the lecturer follows every comment with “you should be taking notes”.
We can make a lecture a special kind of object:(define (make-lecturer name) (let ((super (make-object name))) (lambda (message) (if (eq? message 'lecture) (lambda (self stuff) (ask self 'say stuff) (ask self 'say (list "you should be taking notes"))) (get-method super message)))))When a message is sent to an object created by make-lecturer, it first checks if the message is 'lecture. If it is, it evaluates to the procedure that lectures. If it is not, it evaluates (get-method super message) to get the method in the superclass. The name super is a place in the frame defined by the let (which desugars to lambda) that has the value of (make-object name).
Question 5: A professor is even more arrogant than a lecturer. Define a procedure (make-professor name) that produces a professor object. It should inherit from make-lecturer, so it will be able to respond to the lecture message. It should also have a method profess that is like lecturing, but precedes every statement with “it is intuitively obvious that”. Your professor should work like this:
Note that the lecture method is inherited from lecturer and the name method is inherited from lecturer, which inherits it from object. Your make-professor procedure should fit on 8 reasonably short lines.> (define ed (make-professor 'Evan-Davis))
> (ask ed 'profess (list "(lambda (n) ((lambda (f) (f f n)) (lambda (f k) (if (= k 1) 1 (* k (f f (- k 1))))))) is a familiar function"))
it is intuitively obvious that
(lambda (n) ((lambda (f) (f f n)) (lambda (f k) (if (= k 1) 1 (* k (f f (- k 1))))))) is a familiar function
you should be taking notesnuf-said
> (ask ed 'lecture (list "smalltalk is cool"))
smalltalk is cool
you should be taking notesnuf-said
> (ask ed 'name)
evan-davis
State
Objects package procedures and state. The method you added in Question 4 and the professor class you defined in Question 5 didn't use any state. Every time we do (ask ed 'lecture (list "smalltalk is cool")) the same thing should happen.Recall the counter from Lecture 20:
Remember that the let is just syntactic sugar for an application of a lambda.(define (make-counter) (let ((count 0)) (lambda (message) (if (eq? message 'reset) (set! count 0) (if (eq? message 'next) (set! count (+ 1 count)) (if (eq? message 'how-many) count))))) )The counter object packages the state count with the methods reset, next and how-many. We call places associated with objects instance variables. Each time we evaluate (make-counter) a new object is created. We call objects created by constructors instances of the class. Hence, (define mycounter (make-counter)) defines mycounter as an object that is an instance of the counter class. The object state variables are known as instance variables because they name a place associated with a particular instance of the class. That is, if we do:
both counter1 and counter2 will be counter objects. Each will have its own frame, containing a place named count. Hence:> (define counter1 (make-counter)) > (define counter2 (make-counter))> (ask counter1 'next) > (ask counter2 'how-many) 0More interesting objects in our game will need to use state to keep track of things that might change during an execution. Look at the make-person procedure defined in objects.ss. Its pretty long because a person has many methods. Here we show some of the code, but leave out some methods:(define (make-person name) (let ( ;; person inherits from mobile-object (super (make-mobile-object name)) ;; Instance variables (possessions '()) ;;; What the person is carrying (a list of Objects that are things (restlessness 0) ;;; How likely the person is to move randomly ) (lambda (message) (case message ((person?) (lambda (self) #t)) ((install) (lambda (self where) (ask super 'install where))) ((get-possessions) (lambda (self) possessions)) ;;; Many other methods not shown (else (get-method super message))))))A person has an instance variable possessions that is a list of objects the person is carrying (we'll get to the restlessness instance variable later). The method get-possessions can be used to see what a person is holding. Other methods use (set! possessions ...) to change the possesions a person is holding.
Question 6: A student is a special kind of person (this doesn't mean all students are special or kind). Define a procedure make-student that creates a student object. It should inherit from person. A student has an instance variable is-dressesed that indicates whether or not the student is clothed. Initally, all students are dressed. A student has a method get-undressed that changes the state of is-dressed to #f. A student also has a method get-dressed that changes the state of is-dressed to #t. A student also has a method is-dressed? that evaluates to #t if the student is dressed and #f otherwise.
Your student should work like this:
> (define alyssa (make-student 'alyssa-p-hacker))
> (ask alyssa 'is-dressed?)
#t
> (ask alyssa 'name)
alyssa-p-hacker
> (ask alyssa 'get-undressed)
At not-yet-installed: alyssa-p-hacker says -- brrrrr...its cold!
> (ask alyssa 'is-dressed?)
#f
> (ask alyssa 'get-dressed)
At not-yet-installed: alyssa-p-hacker says -- I feel much better now.
> (ask alyssa 'is-dressed?)
#t
Automating Objects
This kind of world doesn't make for a very challenging or interesting game. All the people and things only do what the player tells them to. For a more interesting game, we need to have people and things that act autonomously.We do this by creating a list of all the characters to be moved by the computer and by simulating the passage of time with an world-clock object. The make-world-clock procedure (defined in objects.ss) creates a world-clock. It has two instance variables: global-time, for keeping track of the time, and clock-list for keeping track of the objects that should be sent clock-tick messages when the clock advances.
The methods add and remove that have object parameters and add or remove objects from the clock-list. When the clock receives a clock-tick message it means time has passed. It sends a clock-tick message to each object in its clock-list. This doesn't necessarily do something every time, but for some objects it will lead to an action.
In adventure.ss, we create a clock and add all objects installed using install-object to the clock:
We can advance the clock by doing (ask clock 'tick).(define clock (make-world-clock)) (define (install-object object place) (ask object 'install place) (ask clock 'add object))People hang about idly until they get bored enough to do something. To account for this, we give people a restlessness instance variable that indicates how likely they are to get bored enough to move randomly. If restlessness is not #f, a person will move in a random direction with 1/restlessness probability with each clock tick. For example, if restlessness is 1, the person will move randomly every clock tick. A person object has a method make-restless that take a parameter and sets the restlessness instance variable to that value.
The University administration does not condone streaking (see these October 19, 1999 and March 24, 2000 articles in The Cavalier Daily). The have begun strategically placing police officers on the Lawn to apprehend streakers. Coincidentally, the fictional University in our simulation game also has a similar policy!
Question 7: Define a procedure make-police-officer that makes a special kind of person. It should inherit from a person — but have an additional method: In addition, you should make the clock-tick method for a police officer automatically arrest anyone streaking in the place where the police officer is. If no one is streaking, the police officer will act like a normal person — that is, it should invoke the super class (person) clock-tick method. You may find the other-people-at-place procedure (defined in objects.ss) useful.
- (ask 'arrest Person) — A police officer can arrest a person if they are in the same place. The arrestee is sent to jail (you can do this with (ask Person 'move-to Jail)) and the police officer takes all of the arrestees. possessions.
At this point, you should spend some more time playing the game. We have provided a procedure (play-interactively-as character) that provides a better interface to playing the game. The play-game procedure (defined in ps6.ss) installs two students and one restless police officier in our world, and starts playing interactively as one of the students.
Here's what a typical game might look like:
> (play-game)
At not-yet-installed: alyssa-p-hacker says -- Installing alyssa-p-hacker at green
At not-yet-installed: ben-bitdiddle says -- Installing ben-bitdiddle at cdrs-hill
At not-yet-installed: officer-krumpke says -- Installing officer-krumpke at bart-statue
what now? > name
< Result: alyssa-p-hacker>
[Clock] Tick 1
At bart-statue: officer-krumpke says -- No one to arrest. Must find donuts.
officer-krumpke moves from bart-statue to steam-tunnel
officer-krumpke is no longer at bart-statue
what now? > get-undressed
At green: alyssa-p-hacker says -- brrrrr...its cold!
[Clock] Tick 2
At steam-tunnel: officer-krumpke says -- No one to arrest. Must find donuts.
ben-bitdiddle moves from cdrs-hill to cricket-street
what now? > look
At green: alyssa-p-hacker says -- I see nothing
At green: alyssa-p-hacker says -- I can go down west north south
< Result: ()>
[Clock] Tick 3
At steam-tunnel: officer-krumpke says -- No one to arrest. Must find donuts.
what now? > go north
alyssa-p-hacker moves from green to recursa
< Result: #t>
[Clock] Tick 4
At steam-tunnel: officer-krumpke says -- No one to arrest. Must find donuts.
officer-krumpke moves from steam-tunnel to bart-statue
ben-bitdiddle moves from cricket-street to cdrs-hill
what now? > go south
alyssa-p-hacker moves from recursa to green
< Result: #t>
[Clock] Tick 5
At bart-statue: officer-krumpke says -- No one to arrest. Must find donuts.
what now? > go south
alyssa-p-hacker moves from green to bart-statue
At bart-statue: alyssa-p-hacker says -- Hi officer-krumpke
< Result: #t>
[Clock] Tick 6
At bart-statue: officer-krumpke says -- alyssa-p-hacker, you are under arrest!
alyssa-p-hacker moves from bart-statue to jail
At bart-statue: officer-krumpke says -- You have the right to remain silent, call methods and mutate instance variables.
what now? > look
At jail: alyssa-p-hacker says -- I see nothing
At jail: alyssa-p-hacker says -- I can go
< Result: ()>
[Clock] Tick 7
At bart-statue: officer-krumpke says -- No one to arrest. Must find donuts.
officer-krumpke moves from bart-statue to green
what now? > quit
Better luck next time. Play again soon!
Extensions
With a full command of the awesome powers of message-accepting procedures, object-oriented programming, and inheritance, you’re ready to start making an award-winning game. Keep in mind that, historically, computer games have been a colossal waste of time for humankind. This is no exception. Trust us, as simple as this game is, it's easy to get lost (especially in the steam tunnels). Spend your time on the problems, not the game. You have been warned.
Question 8: Design a non-trivial extension to this simulated world. You can do what you want (so long as it is in good taste, of course). The extension can be as elaborate as you like, but don’t go overboard — this is meant to be a problem set, not a term paper, after all.
Try to base your extended simulation around a theme. Possibilities include food, classes, or bicycles. Use your imagination!
Whatever you choose to do, your simulation should include at least two new kinds of persons, places, or things, using inheritance. For example, you might implement a classroom as a new kind of place. Your new objects should have some special methods or special properties in relation to other objects. To continue the previous example, you might make new kinds of people called lazy-students, who go to sleep when they enter classrooms.
In answering this problem, you should turn in:
- One or two paragraphs explaining the "story" behind your simulation. Describe your new objects and their behaviors.
- An inheritance diagram showing your new classes.
- Listings of any new procedures you write, or old procedures that you modify.
- A transcript that shows your simulation in action.
An award will be given for the best adventure game. Games will be judged on both technical merit and creativity. Note that it's more impressive to implement a simple, elegant idea, than to just amass tons of new objects and places. You may work with as many people as you want for this question, but there is only one award, so you will have to figure out how to divide it if you work in a group.
That's it! Now invite a friend over and teach them how to play. But, remember that this is a fictional world. Don't try anything from the game at home. If you do get a chance to visit Charlottansville, make sure to see Monty's Viola, the favorite instrument of the founder of the University and author of the influential 1976 treatise, The Ultimate Declarative, which lead many Schemers to revolt.
Credits: This problem set based on a problem set used in MIT 6.001 Fall 1989, and subsequently in many other years including 1997, on which we based our code. This problem set was developed by Portman Wills with help from David Evans. Portman is solely responsible for all the streaking references, however. The MIT version of the adventure game involved Deans smashing beer and party-troll's eating Deans. A course at UC Berkeley also had an adventure game problem set. Their acknowledgment was, This assignment is loosely based on an MIT homework assignment in their version of this course. But since this is Berkeley we've changed it to be politically correct; instead of killing each other, the characters go around eating gourmet food all the time. N.B.: Unless you are a diehard yuppie you may feel that eating gourmet food does not express appropriate sensitivity to the plight of the homeless. But it's a start.We like our Deans much more than they do at MIT, and UVA students don't eat much gourmet food. We thought about making our adventure game about eating Berkeley students, but were worried that might improve their student-faculty ratio and ruin our chances of ever catching them in the US News rankings again.
University of Virginia
Charlottansville, East Virginia
CS 200: Computer ScienceDavid Evans
evans@cs.virginia.edu