联系方式

  • QQ:99515681
  • 邮箱:99515681@qq.com
  • 工作时间:8:00-21:00
  • 微信:codinghelp

您当前位置:首页 >> Java编程Java编程

日期:2022-02-26 12:08

Object Oriented Programming

Assignment 1 - Connect Four

Overview

This assignment is about practising the programming concepts taught during the first few weeks of

the course. To hone and demonstrate your skills, you will complete a series of steps to implement

the board game Connect Four (https://en.wikipedia.org/wiki/Connect_Four) in Java. A program

structure is provided with some incomplete methods for you to fill in. The following three sections

describe the steps you need to complete for a basic, intermediate and advanced solution.

Before you start, please read the following sections below:

• Restrictions

• Marking Criteria

• Submission, late submission & extensions

Setup

This section describes how to download the provided code, create a new project and run the game.

Download the provided code

• Go to Assignments → Assignment 1 (where you also found these instructions).

• Download ConnectFourStarter.zip and unpack the src and saves folders it contains.

Create a new project

• Create a new, empty Java project called “ConnectFour” in IntelliJ (or your preferred IDE).

• Copy the src and saves folders into your project directory (replacing the default src folder).

Explore the packages and classes

• Do not panic! Most of the classes have already been completed for you. Providing code is the

only way of letting you work on a larger, more interesting project before you’ve learned all

required concepts. You do not have to understand how everything works.

• You will only be editing Model.java and TextView.java in the game package. Depending on how

far you get, you will also be working on one or more classes in the players package.

Run the code

• Open ConnectFour.java (in the main package) and press your IDE’s run button. You should be

greeted with a welcome message and a menu. Selecting option 1 starts a new game, but until

you have completed some steps in the basic section, this will enter an infinite loop from which

you have to quit manually.

Basic requirements

This section describes the features required for a basic implementation of Connect Four. A good

solution will allow you to pass, without attempting any of the intermediate or advanced features.

The suggested sequence of steps is intended to guide you towards creating a playable game.

Step 1: Model the state of the board [Relevant file: Model.java]

Think about how to represent the state of the board. Add the appropriate field(s) to Model.java and

initialise these variables in its initNewGame method. This is called every time a new game is started

and should create an empty board. For now, you can ignore the settings parameter and assume a

standard board size with 6 rows and 7 columns. The top row is row 0, with row numbers increasing

downwards. The leftmost column is column 0, with column numbers increasing towards the right

(see illustration below). Next, implement the getPieceIn method to allow access to the state of the

board. It takes row and column indices as its parameters and should return the owner of the piece in

that cell of the board i.e. 1 for player 1 and 2 for player 2. Return 0 to indicate an empty cell.

Step 2: Allow the user to select moves [Relevant file: HumanPlayer.java]

You can already select different players from the main menu (option 4), but all of these players will

just concede the game on their first move. The goal is now to replace this behaviour with something

useful. To allow the user to make a move, implement the chooseMove method of HumanPlayer.java

(in the players package). This is called at the start of a player’s turn and should return the column

index into which the player wants to insert their next piece (or -1 if they really wish to concede). All

you need to do is print out a message asking the user to enter a move and return what they typed in

(use InputUtil.readIntFromUser for this). When you start a game, the program now waits until the

user has entered a move and pressed enter. After entering a few moves, you will realise that you are

still stuck in an infinite loop though. That is because we are not yet reacting to those moves. We will

return to this issue a few steps later. Note that HumanPlayer is selected by default for both players

so you do not need to change the players in the menu every time you start the program.

Step 3: Display the board [Relevant file: TextView.java]

Open TextView.java and complete the displayBoard method to print a text representation of the

board to standard output. Use characters ‘X’ and ‘O’ to represent pieces belonging to players 1 and

2 respectively and use ‘_’ (underscore) for empty slots. You can add a space between each column if

you like, but please do not add any other decorations. You can test this is working by starting a new

game. An empty board should be displayed, see left image below.

Empty board Board after several moves

Step 4: Store the moves in the model [Relevant file: Model.java]

Although the model receives these moves you enter (as the parameter to its makeMove method), it

has not yet been set up to put the pieces onto the board. As long as the moves you make are not

stored in memory, the board will remain empty. Correct this now, making sure each piece is inserted

in the correct row and column. Use the getActivePlayer method to determine the owner of the piece.

You should now be able to place some pieces on the board when you play, but since player 1 starts

the game and the active player is never changed, all pieces will be marked as ‘X’.

Step 5: Taking turns [Relevant file: Model.java]

The model also needs to keep track of whose turn it is. Currently, its getActivePlayer method

always returns 1. After putting the first piece on the board, player 2 should become active and

getActivePlayer should return 2. After that it’s back to player 1 and so on. Think about how to

represent whose turn it is and when to switch. In particular, getters (methods that return a value)

should never change anything so that you can call them as often as you like. Make sure player 1

always goes first when a new game starts. Players should now be able to take turns and, after a few

moves, the state of your board should be similar to that shown in the screenshot above on the right.

Step 6: Game over [Relevant file: Model.java]

The game should end either when the board is full or when a player concedes (by entering -1 as

their move). The model’s getGameStatus method is responsible for returning a code that indicates if

the game is still in progress or, if not, how it ended. A zero means the game is ongoing, a 1 means

victory for player 1, a 2 victory for player 2 and 3 indicates a tie. To make programs easier to read,

we often store such codes as special constants. That is why instead of returning 0, getGameStatus

returns IModel.GAME_STATUS_ONGOING. Change this to return IModel.GAME_STATUS_TIE

when the board has been filled up. If player 2 concedes, return GAME_STATUS_WIN_1 and

GAME_STATUS_WIN_2 if player 1 concedes. Hint: Think about what causes the game status to

change and where this should be recorded. Detecting the winner (first player to get four in a row) is

an intermediate feature and not required for a basic solution.

Intermediate features

Once you are satisfied that you have all basic features working, you can enhance your program by

adding the functionality described here. This will allow you to obtain a mark of up to 69% provided

that your code for all the basics is working well.

Variable game settings [Relevant files: TextView.java & Model.java ]

When the user selects “Change game settings” from the menu, the requestGameSettings method in

TextView.java is called. Edit this method to allow the user to enter a board height (number of rows)

and width (number of columns). You should also be able to choose the required streak length, so

you can play Connect X. It is important to ask for the settings in this order (rows, columns, streak).

The user should press enter after typing each value (use InputUtil.readIntFromUser for this).

Now, make any changes required for the model to support the variable game settings. The chosen

settings are passed to the initNewGame method and can be accessed using settings.nrRows etc.

Input validation [Relevant files: Model.java & TextView.java ]

So far, we have trusted the user to enter only sensible values. If they do not, they can crash the

program or make the game impossible to win. Input validation involves detecting and preventing

such issues.

Start by changing the model’s isMoveValid method to return false when the passed in move is

invalid (and true otherwise). Players will then automatically be asked to re-enter rejected moves.

The requestGameSettings method should repeat the request for each setting until an acceptable

value has been entered. The board should have no less than 3 rows or columns and no more than 10

(see constants for this in IModel).

Implement your first AI player [Relevant file: RoundRobinPlayer.java]

Complete the prepareForGameStart and chooseMove methods in RoundRobinPlayer.java in order

to create your first AI player. This player's first move (after a game is started/loaded) is always to

play the leftmost column that is not full. The column index then increases by 1 with each time, until

we reach the last column and return to column 0. A typical sequence of moves would therefore be:

0, 1, 2, 3, 4, 5, 6, 0, 1, … The only exception is when a column is full. In this case, skip ahead until

a valid column is found. Loading games is an advanced feature that you do not have to implement

here. However, please do not assume the board will always be empty when prepareForGameStart is

called. Your code should be compatible with variable game settings.

To test your AI player, choose option 4 from the game menu and then select one or both players to

be the RoundRobinPlayer (option 2). All combinations of players are possible, including AI vs AI.

Automatic win detection [Relevant file: Model.java]

Write code to detect when the game is won. The getGameStatus method in the model should then

return 1 or 2 to indicate the winner. Again, this should work with variable game settings.

Advanced features

Tasks described in this section are much more difficult and only required for a distinction.

Load saved game states from file [Relevant file: Model.java]

Implement a mechanism by which you can load a game state from a text file and resume play. This

involves implementing initSavedGame in the model. All save files are assumed to be in the “saves”

folder in the project hierarchy. The user is only prompted for the name of the file to be loaded (this

has already been implemented). Each file starts with 4 lines containing one integer denoting the

number of rows, columns, streak length and the active player (in that order from top to bottom). The

board state follows, with one line for each row of the board (top to bottom). Each line is a string

with as many characters as there are columns. Zeroes represent empty cells and filled cells are

represented by 1 or 2 depending on the player who owns the piece. You can assume all save files are

formatted correctly and that saved games are not yet over (there is no winner and the board is not

full). See the “Save.txt” and “AnnotatedSaveFile.txt” in the “saves” folder.

Implement the WinDetectingPlayer [Relevant files: WinDetectingPlayer.java & maybe Model.java]

If the player can win on their current turn, a winning move must be selected. Otherwise, choose

any valid move that does not let the opponent win next turn. If no such move exists, concede. This

player must be able to cope with variable game settings and games being loaded from file.

Implement the CompetitivePlayer [Relevant files: CompetitivePlayer.java & maybe Model.java]

Create the strongest AI player you can come up with. It will compete against a pretty strong AI of

our own. Points are awarded depending on the score over a number of games. Games in which your

AI submits invalid moves or tampers with the game state will be forfeited (i.e. counted as a loss).

This player must be able to cope with variable game settings and games being loaded from file.

Restrictions

Please note that you will only be able to submit the files you have been asked to change ! That

means you should not edit any other files, as your solution must be compatible with the supplied

versions of those files. Going against this can cause the automated tests to fail and will result in

you losing the associated marks!

You are permitted to use the Java API, but you must not use any other/external libraries. You cannot

upload libraries anyway, but you should also not copy library code into your the files you upload.

For your own learning, we recommend you do not use any functional language constructs such as

lambdas or streams. Those were taught in first semester, but this course is about imperative

programming. This is not enforced for assignment 1, but will be for assignment 3.


相关文章

版权所有:编程辅导网 2021 All Rights Reserved 联系方式:QQ:99515681 微信:codinghelp 电子信箱:99515681@qq.com
免责声明:本站部分内容从网络整理而来,只供参考!如有版权问题可联系本站删除。 站长地图

python代写
微信客服:codinghelp