Untitled

 avatar
unknown
java
a year ago
5.6 kB
11
Indexable
package comp1110.lab8;

import java.util.*;

public class CropRotation {
    /**
     * Each Vegetable is assigned to one of four groups.
     */
    public enum Group {
        LEGUME, BRASSICA, ALLIUM, FRUITING
    }

    public static class Vegetable {
        String name;
        Group group;

        public Vegetable(String name, Group group) {
            this.name = name;
            this.group = group;
        }

        @Override
        public String toString() {
            return name + " (" + group.name().toLowerCase() + ")";
        }
    }

    /**
     * Get all valid crop rotations that can be created from the provided
     * set of vegetable crops for the given number of seasons.
     * A crop rotation is a sequence of vegetable crops to plant.
     * <p>
     * One crop is planted per season, and any crop may be planted at most once.
     * Crops must be planted in order of their Group according to the following
     * rules:
     * - a LEGUME may only be followed by a BRASSICA;
     * - a BRASSICA may only be followed by an ALLIUM;
     * - an ALLIUM may only be followed by a FRUITING crop; and
     * - a FRUITING crop may only be followed by a LEGUME.
     * <p>
     * For example, the call
     * getAllRotations([Tomato (fruiting), Onion (allium)], 2)
     * returns a set containing a single rotation:
     * - [Onion (allium), Tomato (fruiting)]
     * because an ALLIUM may only be followed by a FRUITING crop.
     * <p>
     * The call
     * getAllRotations([Tomato (fruiting), Kale (brassica), Onion (allium), Pea (legume)], 4)
     * returns a set containing four rotations:
     * - [Kale (brassica), Onion (allium), Tomato (fruiting), Pea (legume)]
     * - [Onion (allium), Tomato (fruiting), Pea (legume), Kale (brassica)]
     * - [Pea (legume), Kale (brassica), Onion (allium), Tomato (fruiting)]
     * - [Tomato (fruiting), Pea (legume), Kale (brassica), Onion (allium)]
     * <p>
     * If no valid crop rotation can be found, an empty list is returned.
     * For example, the call:
     * getAllRotations([Tomato (fruiting), Gai Lan (brassica)], 2)
     * returns an empty set, because a FRUITING crop cannot be followed by
     * a BRASSICA, and a BRASSICA cannot be followed by a FRUITING crop.
     *
     * @param crops   the set of vegetable crops from which to construct a rotation
     * @param seasons the number of seasons
     *
     * @return the set of all possible rotations of the provided crops for the
     * given number of seasons. If there are no crops or no seasons or the number
     * of seasons is greater than the number of crops, return empty list.
     */
    public static Set<List<Vegetable>> getAllRotations(Set<Vegetable> crops, int seasons) {
        List<Vegetable> used = new ArrayList<>();          // vegetables used so far in a given search
        Set<List<Vegetable>> rotations = new HashSet<>();  // rotations found so far

        for (Vegetable firstVegetable : crops) {
            used.add(firstVegetable);
            getFixedRotation(crops, seasons, used, rotations);
            used.remove(used.size() - 1);
        }
        return rotations;
    }

    /**
     * Recursive method to find all rotations given a starting crop.
     * <p>
     * Note that this method does not have a return type. Instead, it modifies
     * `rotations` by adding crop rotations to `rotations` as it finds them in the
     * search space.
     *
     * @param crops     the set of vegetable crops from which to construct a rotation.
     *                  Note that this is all possible crops that can be used, not just
     *                  the crops that have already been used or that are remaining. You
     *                  can think of this as the domain of the problem if you'd like.
     * @param seasons   the number of seasons
     * @param used      the crops that already been used (in the order in which they are used)
     * @param rotations all rotations found so far
     */
    private static void getFixedRotation(Set<Vegetable> crops, int seasons, List<Vegetable> used,
                                         Set<List<Vegetable>> rotations) {
        if (used.size() == seasons) {
            List<Vegetable> copyOfUsed = new ArrayList<>(used);
            rotations.add(copyOfUsed);
            return;
        }
        Vegetable lastVegetable = used.get(used.size() - 1);
        for (Vegetable nextVegetable : crops) {
            if (canFollow(lastVegetable, nextVegetable) && !used.contains(nextVegetable)) {
                used.add(nextVegetable);
                getFixedRotation(crops, seasons, used, rotations);
                used.remove(used.size() - 1);
            }
        }
    }

    /**
     * Determine whether one vegetable can follow another.
     *
     * @param first the first vegetable
     * @param next  the next vegetable
     *
     * @return true if next can follow first
     */
    private static boolean canFollow(Vegetable first, Vegetable next) {
        if (first.group == Group.LEGUME && next.group == Group.BRASSICA) {
            return true;
        }
        if (first.group == Group.BRASSICA && next.group == Group.ALLIUM) {
            return true;
        }
        if (first.group == Group.ALLIUM && next.group == Group.FRUITING) {
            return true;
        }
        if (first.group == Group.FRUITING && next.group == Group.LEGUME) {
            return true;
        }
        return false;
    }
}
Editor is loading...