Alabama Aviation College Refactoring in Eclipse Computer Programming Questions
Note: this work can only be done in eclipse
Instructions
Background
Q.1
When we think of computer programs and applications today, we think of graphical interfaces with buttons and expandable menus and other kinds of displays. But the consumer computers started finding their ways into homes 5 years before the technology for what we would think of as “graphics” debuted, and it would take another 5 years for those first graphics cards to become commonplace. So for a solid decade, almost all software had to display output in a text format, and collect input from users through text-based commands.
Some of the first games for these home computers then were similar to Choose-Your-Own-Adventure books, and defined the “adventure” genre accordingly (sometimes the genre name is attributed to the game Colossal Cave Adventure, a pioneer of the format) . In these kinds of games, the player is presented with a description of a place or an event, and has to type in a command to do something. The game parses this command, and then prints some new output accordingly. Generally, the player is navigating through different “rooms” or areas using cardinal directions to explore the environment.
For a good example of this, check out the adventure game Zork, which you can play from the link in the reference section below. You’ll find that while the complete list of commands was only printed in the manual, many of them are intuitive – open mailbox, take leaflet, read leaflet, go south, look…
For this project, you will be building a similar kind of application, but rather than an adventure game we’ll just use it to present an interactive “tour” of a home – maybe your current home, maybe your dream home. In any case, the pattern will be exactly the same as other text-based games:
- Display prompt
- Collect input
- Process input as a command
- Print output of command
Here’s an example of what your game might look like:
Instructions
For now, we’re just going to create the shell for our project, based on what we understand of the requirements. Later, as your skills improve and you continue to learn, you’ll probably make changes to these classes or their structure – a process called refactoring – to make your project easier to expand.
- In Eclipse, create a project named HomeTour
- In the HomeTour project, create the following packages and classes:
fixtures
fixtures.Fixture
(abstract)fixtures.Room
game
game.Main
game.Player
game.RoomManager
These classes will work as follows:
fixtures.Fixture:
This abstract class will be used as a base for anything that can be looked at or interacted with. This class should define (at least) the following properties:
String name
: a short name / title for the fixtureString shortDescription
: a one-sentence-long description of a fixture, used to briefly mention the fixtureString longDescription
: a paragraph-long description of the thing, displayed when the player investigates the fixture thoroughly (looks at it, or enters a room)
Here’s an example of what your game might look like, so these make sense:
fixtures.Room
:
This class represents a room in the house. It will extend fixtures.Fixture, and so will inherit the descriptive properties. The Room will also have the following properties:
Room[] exits
: the rooms adjacent to this one. You might decide that a room in a particular direction always uses a certain index, e.g. a north exit always goes in index 0, an east exit always goes in index 1, etc. If so, then the size of this array depends on how many directions you want to support.
The Room class should also have a constructor that accepts a name, shortDescription, and longDescription. You might also find it convenient to create a getter not just for all the exits, but for a particular exit given a direction:
public Room(String name, String shortDescription, String longDescription) {
super(name, shortDescription, longDescription);
this.exits = new Room[?]; // size is your choice
}
public Room[] getExits() {
}
public Room getExit(String direction) {
}
game.Main
:
This class will store the main(String[])
method for our game (and of course, it will be the only class that has a main(String[])
method). This is where the game-loop will go, where we’ll display a prompt, collect input, and parse that input
The printRoom(Player)
method will print a prompt to the console for the player’s current room, similar to the above image.
The collectInput()
method will use a Scanner object to collect console input from the user, and then will divide that input into multiple parts. Generally those parts will look like this:
- An action
- The target of an action (if any)
For example, “go east” -> “go” is the command, “east” is the target. This method will break the input into a String[], and return that.
The parse(String[], Player)
method will take the output of the above collectInput()
method and a player object, and will resolve that command. This can actually be simpler than it sounds – the first index of the passed-in String[] should be the action, so you can switch
on that and handle the target differently for each case
. The Player object is there so you can modify it if needed (like changing the Player’s currentRoom based on the direction moved)
public static void main(String[] args) {
}
private static void printRoom(Player player) {
}
private static String[] collectInput() {
}
private static void parse(String[] command, Player player) {
}
game.Player
:
This class represents the player moving through these rooms. The Player class has these properties:
Room currentRoom
: the room the player is currently in.
game.RoomManager
:
This class will be responsible for “loading” our rooms into memory. When game.Main
is executed, it will invoke the init()
method in this class that will instantiate all our Room objects, link them together as exits, and designate a startingRoom.
Room startingRoom
: the room a player should start in.Room[] rooms
: all the rooms in the house.
public void init() {
Room foyer = new Room(
"The Foyer",
"a small foyer",
"The small entryway of a neo-colonial house. A dining room is open to the south, where a large table can be seen." + "n"
+ "The hardwood floor leads west into doorway, next to a staircase that leads up to a second floor." + "n"
+ "To the north is a small room, where you can see a piano.");
this.rooms[0] = foyer;
this.startingRoom = foyer;
}
Instructions
Up until now, you have been tracking the exits in each Room of your Home Tour application using a Room[]. This might have been functional, but it's not optimal, and you have probably noticed how clunky it feels. We have to set an arbitrary standard ahead of time, known only to those possessing documentation - exits[0] is north, exits[1] is east, etc. We also have to decide during development the maximum number of exits a room might have, and in which directions those exits might be.
So what do we know about room exits? We know that the exit of one Room is another Room, and we know those exits are situated in a certain direction from the current room. For example:
foyer:
north: library
This scenario seems like a good application of a Map, doesn't it? Instead of a Room[], you could have a Map<String, Room> where the key is the direction of an exit and the value is the room in that direction.
The process of rewriting code to improve efficiency or the overall design, is called Refactoring. Having to refactor your code is not a sign of ineptitude or bad design. It's an extremely common practice for developers to implement the first solution that comes to mind, so that a Minimum Viable Product (MVP) can be demonstrated and tested. Having an MVP allows you to demonstrate that a problem can be solved, that you are capable of completing the application that will do so. Once the MVP is in hand, the design can be refined to squash bugs, improve efficiency and stability, and to add new features. There are entire books written about the best way to go about refactoring your code, and tools to assist in it.
Take some time to refactor your Room class to use this Map for its exits, instead of Arrays. This will also entail refactoring other classes accordingly, like Main.parse(String[], Player). Instead of accessing rooms by index, they have to use the get(String) method.
You could also probably refactor RoomManager to use a collection for its collection of Rooms, instead of a Room[]. This will make it slightly easier to add new rooms in the future.