(Analysis by Nick Wu and Brian Dean)

The critical observation for this problem is that an optimal rectangle must have a Holstein on each of its four sides. This motivates the following line-sweep solution:

For every pair of horizontal lines that contains a Holstein, remove all cows that aren't between those two lines. Sweep from left-to-right, keeping track of how many Holsteins have been seen so far without a Guernsey.

If we pre-sort all the cows, then the sweeping process takes linear-time, giving us an $O(n^3)$ solution, illustrated in the code below. This was fast enough to obtain full credit for the problem (the problem was intended to the "easier" of the gold problems on this contest).

Faster solutions are possible. Here is a short description of one (of several) ways to achieve an $O(n^2 \log n)$ running time. Recall that the four sides of the optimal rectangle must contain Holsteins; let's denote these by $H_t$, $H_b$, $H_l$, and $H_r$, with $t$ meaning "top", $b$ meaning "bottom", $l$ meaning "left" and $r$ meaning "right". We first iterate over all possible choices for $H_b$, contributing $O(n)$ to our running time. Having now fixed $H_b = (x_b, y_b)$, we scan upward (having pre-sorted all points on $y$), adding all $H$'s to an STL set, $S$, keyed on $x$ coordinate. We also keep track of the $G$ with maximum $x$ coordinate less than $x_b$ (call it $g_l$) and the $G$ with minimum $x$ coordinate larger than $x_b$ (call it $g_r$). We use these values to restrict the entries in $S$ so they belong to the range $[g_l, g_r]$, by deleting the min or max from $S$ whenever these fall outside the range. Now for each $H$ we encouter, if it lies in the $x$ range $[g_l, g_r]$, we test the rectangle with this $H$ as $H_t$, and with the min and max entries in $S$ as $H_l$ and $H_r$. The best rectangle overall is returned. The total scanning time is $O(n \log n)$, for a total running time of $O(n^2 \log n)$.

import java.io.*;
import java.util.*;
public class rectangle {
public static void main(String[] args) throws IOException {
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
PrintWriter pw = new PrintWriter(new BufferedWriter(new OutputStreamWriter(System.out)));
int n = Integer.parseInt(br.readLine());
State[] list = new State[n];
TreeSet<Integer> ys = new TreeSet<Integer>();
for(int a = 0; a < n; a++) {
StringTokenizer st = new StringTokenizer(br.readLine());
int x = Integer.parseInt(st.nextToken());
int y = Integer.parseInt(st.nextToken());
list[a] = new State(x, y, st.nextToken().equals("H"));
ys.add(y);
}
Arrays.sort(list);
ArrayList<Integer> ysArray = new ArrayList<Integer>();
for(int y: ys) {
ysArray.add(y);
}
int most = 0;
int area = 0;
for(int i = 0; i < ysArray.size(); i++) {
for(int j = i+1; j < ysArray.size(); j++) {
boolean valid = false;
int lastX = -1;
int now = 0;
for(int a = 0; a < n;) {
int b = a;
int red = 0;
int blue = 0;
while(b < n && list[a].x == list[b].x) {
if(list[b].y >= ysArray.get(i) && list[b].y <= ysArray.get(j)) {
if(list[b].red) {
red++;
}
else {
blue++;
}
}
b++;
}
if(blue > 0) {
valid = false;
now = 0;
}
else if(red + blue > 0) {
if(!valid) {
valid = true;
lastX = list[a].x;
}
now += red;
int currArea = (ysArray.get(j) - ysArray.get(i)) * (list[a].x - lastX);
if(now > most || (now == most && currArea < area)) {
most = now;
area = currArea;
}
}
a = b;
}
}
}
pw.println(most);
pw.println(area);
pw.close();
}
static class State implements Comparable<State> {
public int x,y;
public boolean red;
public State(int a, int b, boolean c) {
x=a;
y=b;
red=c;
}
public int compareTo(State s) {
return x - s.x;
}
}
}