Skip to content

Latest commit

 

History

History
150 lines (123 loc) · 5.89 KB

3-gathering-collectables.md

File metadata and controls

150 lines (123 loc) · 5.89 KB

Gathering Collectables

The Plan

You need to gather collectable items in order to spawn new players to increase the size of your army; with more players you can occupy more of the map and reach new collectable items before your opponents, thus increasing the size of your army and (hopefully) eventually eliminating all opponents.

Code Updates

One Collectable Per Player

It makes sense to ensure that each collectable is only assigned to one player, so add a Map to manage these assignments, again making it a local variable and passing it to the relevant methods:

Map<Player, Position> assignedPlayerDestinations = new HashMap<>();

Collectables are Higher Priority

Given the importance of growing your army, let's assign players to gather collectables before assigning the remaining players to exploration duties. Add the following method:

private List<Move> doCollect(final GameState gameState, final Map<Player, Position> assignedPlayerDestinations, final List<Position> nextPositions) {
    List<Move> collectMoves = new ArrayList<>();
    System.out.println(collectMoves.size() + " players collecting");
    return collectMoves;
}

And then add a call to doCollect above the call to doExplore:

moves.addAll(doCollect(gameState, assignedPlayerDestinations, nextPositions));

Track Collectables and Players

First things first, you need to know where the collectables are and also a list of your players, so let's add some code to the doCollect method, after the list initialisation, for those two tasks:

Set<Position> collectablePositions = gameState.getCollectables().stream()
        .map(collectable -> collectable.getPosition())
        .collect(Collectors.toSet());
Set<Player> players = gameState.getPlayers().stream()
        .filter(player -> isMyPlayer(player))
        .collect(Collectors.toSet());

Calculate Distances and Routes

Next you want to assign a player to gather each collectable, it makes sense to assign the player closest to each collectable to that task so you need to calculate the distances from all players to all collectables using another utility method map.distance(from, to). Add the following code next:

List<Route> collectableRoutes = new ArrayList<>();
for (Position collectablePosition : collectablePositions) {
    for (Player player : players) {
        int distance = gameState.getMap().distance(player.getPosition(), collectablePosition);
        Route route = new Route(player, collectablePosition, distance);
        collectableRoutes.add(route);
    }
}

The Route data structure is given below:

public class Route implements Comparable<Route> {
    private final Player player;
    private final Position destination;
    private final int distance;
    public Route(Player player, Position destination, int distance) {
        this.player = player;
        this.destination = destination;
        this.distance = distance;
    }
    public Player getPlayer() {
        return player;
    }
    public Position getDestination() {
        return destination;
    }
    public int getDistance() {
        return distance;
    }
    @Override
    public int compareTo(Route o) {
        return distance - o.getDistance();
    }
}

This class implements the Comparable interface which means you can make use of the Collections.sort utility method to order the routes based on distance:

Collections.sort(collectableRoutes);

N.B. A more sophisticated Route implementation can be found here: com.scottlogic.hackathon.game.Route.

Now everything is in place so that you can work through the routes and assign your players accordingly using yet another utility method: map.directionsTowards(from, to), this returns a Stream of Directions that a player could make that will reduce the distance between the starting position and the destination.

for (Route route : collectableRoutes) {
    if (!assignedPlayerDestinations.containsKey(route.getPlayer())
            && !assignedPlayerDestinations.containsValue(route.getDestination())) {
        Optional<Direction> direction = gameState.getMap().directionsTowards(route.getPlayer().getPosition(), route.getDestination()).findFirst();
        if (direction.isPresent() && canMove(gameState, nextPositions, route.getPlayer(), direction.get())) {
            collectMoves.add(new MoveImpl(route.getPlayer().getId(), direction.get()));
            assignedPlayerDestinations.put(route.getPlayer(), route.getDestination());
        }
    }
}

Notice the double check on the assignedPlayerDestinations map, once to check the keys for the current player and then again to check the values for the current collectable. If both checks are negative, i.e. neither the player nor the collectable have already been assigned, then a new assignment is added to the map and the relevant first move is added to the list of collection moves.

Testing

Again you're ready to send your upgraded Bot into battle, so run another game.

Windows command prompt:

gradlew run -P mainClass=<your_bot_class_fully_qualified_name>

Unix shell:

./gradlew run -P mainClass=<your_bot_class_fully_qualified_name>

Your Bot should now survive to the TURN_LIMIT_REACHED end condition every time, and if you're lucky you might even see the LONE_SURVIVOR end condition with just your Bot remaining: congratulations, you've just won a game! But your players are now doing a very good impression of a swarm of bees, you can fix that by attempting to explore the map.

Extra Credit

Calculation Efficiency

Does it make sense to recalculate the routes every time? Or is there some scope for improving the efficiency of the algorithm?

Routing Efficiency

Conversely it might make sense to separately manage the assignment of collectables that have just appeared and reassign players to gather these items if the new collectable is nearer than the originally assigned destination.