メンバー > SGHR > topcoder > ApplesAndPears

問題


ジャンル

全探索

解説

格子状のテーブルの上のりんご(Apple)となし(Pear)の並びが与えられ、K回以内の並べ替えでりんごまたはなしまたは空で埋め尽くせる最大の長方形の面積を求めよ、という問題。
例えば下のような初期状態でK=3のとき
P P
P P A
P A P
3回の並べ替えで
P P P
P P P
A A
とするのが最大で、このときなしで埋め尽くされた長方形の面積は6となる。

一見どこから手をつければいいかわからない問題であるがアルゴリズムは単純で、考えうる全ての長方形について
  1. そこに含まれるりんご、なし、空の数を数える。
  2. K回以内の並べ替えでりんご、なし、空で埋め尽くせるか判定する。
  3. 埋め尽くせる場合はその長方形の面積をSmaxとして更新する。
とすればよく、計算量はO(n^4)である。

見た目は複雑そうだがアルゴリズムは単純な良問。

コード

  1. #include<cmath>
  2. #include<cstdlib>
  3. #include<sstream>
  4. #include<string>
  5. #include<vector>
  6. #include<iostream>
  7. #include<queue>
  8. #include<deque>
  9. #include<stack>
  10. #include<list>
  11. #include<map>
  12. #include<algorithm>
  13. using namespace std;
  14.  
  15. #define MAX_N 51
  16.  
  17. class ApplesAndPears{
  18. public:
  19. map<char, vector< vector<int> > > ref;
  20.  
  21. int rect(char c, int x0, int y0, int x1, int y1){
  22. vector< vector<int> > table = ref[c];
  23. return table[x1][y1] - table[x1][y0] - table[x0][y1] + table[x0][y0];
  24. }
  25.  
  26. int getArea(vector <string> board, int K){
  27. int n = board.size();
  28. vector< vector<int> > table;
  29. table.resize(MAX_N);
  30. for(int i=0; i<MAX_N; i++) table[i].resize(MAX_N);
  31. ref.insert(map<char, vector< vector<int> > >::value_type('A', table));
  32. ref.insert(map<char, vector< vector<int> > >::value_type('P', table));
  33. ref.insert(map<char, vector< vector<int> > >::value_type('.', table));
  34.  
  35. for(int x=0; x<=n; x++){
  36. int a = 0;
  37. int p = 0;
  38. int e = 0;
  39. for(int y=0; y<=n; y++){
  40. if(x == 0 || y == 0){
  41. ref['A'][x][y] = 0;
  42. ref['P'][x][y] = 0;
  43. ref['.'][x][y] = 0;
  44. }
  45. else{
  46. if(board[x-1][y-1] == 'A') a++;
  47. if(board[x-1][y-1] == 'P') p++;
  48. if(board[x-1][y-1] == '.') e++;
  49. ref['A'][x][y] = a + ref['A'][x-1][y];
  50. ref['P'][x][y] = p + ref['P'][x-1][y];
  51. ref['.'][x][y] = e + ref['.'][x-1][y];
  52. }
  53. }
  54. }
  55.  
  56. int A = rect('A', 0, 0, n, n);
  57. int P = rect('P', 0, 0, n, n);
  58. int E = rect('.', 0, 0, n, n);
  59. int ans = 0;
  60. for(int x0=0; x0<=n-1; x0++){
  61. for(int y0=0; y0<=n-1; y0++){
  62. for(int x1=x0+1; x1<=n; x1++){
  63. for(int y1=y0+1; y1<=n; y1++){
  64. int S = (x1 - x0) * (y1 - y0);
  65. int a = rect('A', x0, y0, x1, y1);
  66. int p = rect('P', x0, y0, x1, y1);
  67. int e = rect('.', x0, y0, x1, y1);
  68. if(S <= A && 2*p+e <= K && (p == 0 || E > 0)) ans = max(ans, S);
  69. if(S <= P && 2*a+e <= K && (a == 0 || E > 0)) ans = max(ans, S);
  70. if(S <= E && a+p <= K) ans = max(ans, S);
  71. }
  72. }
  73. }
  74. }
  75. return ans;
  76. }
  77. };
  78.  
最終更新:2014年06月19日 14:42