#include #include #include #include #include using namespace std; // Input Information int N, T; int A[3009], B[3009], C[3009], D[3009]; int CurrentMoney; int CurrentCorps; // Other Variables int Build[1009]; int Money[2000009]; int Cost[199][199]; int D1[15][16]; int D2[16][15]; vector> tmp; // Initialize Money void Initialize() { for (int i = 0; i <= 1000; i++) { for (int j = 0; j < 1000; j++) { Money[1000 * i + 223 * j] = 60 * j; } } for (int i = 1; i <= 1000; i++) { Build[i] = (int)(10000000.0 / sqrt(1.0 * i)); } } // Get Total Earned Money int GetEarnedMoney() { for (int i = 0; i < 196; i++) { for (int j = 0; j < 196; j++) Cost[i][j] = 1000000; Cost[i][i] = 0; } // Initialize Cost for (int i = 0; i < 13; i++) { for (int j = 0; j < 14; j++) { int a1 = (i + 0) * 14 + (j + 0); int a2 = (i + 1) * 14 + (j + 0); if (D1[i][j] == 1) { Cost[a1][a2] = 223; Cost[a2][a1] = 223; } else { Cost[a1][a2] = 1000; Cost[a2][a1] = 1000; } } } for (int i = 0; i < 14; i++) { for (int j = 0; j < 13; j++) { int a1 = (i + 0) * 14 + (j + 0); int a2 = (i + 0) * 14 + (j + 1); if (D2[i][j] == 1) { Cost[a1][a2] = 223; Cost[a2][a1] = 223; } else { Cost[a1][a2] = 1000; Cost[a2][a1] = 1000; } } } // Warshall-Floyd for (int k = 0; k < 196; k++) { for (int i = 0; i < 196; i++) { for (int j = 0; j < 196; j++) Cost[i][j] = min(Cost[i][j], Cost[i][k] + Cost[k][j]); } } int sum = 0; for (int i = 1; i <= N; i++) { int pos1 = (A[i] - 1) * 14 + (B[i] - 1); int pos2 = (C[i] - 1) * 14 + (D[i] - 1); sum += Money[Cost[pos1][pos2]]; } return sum; } // Set Rectangle: Case 1 void SetRect1(int L, int R) { for (int i = 0; i < 13; i++) { for (int j = 0; j < 14; j++) D1[i][j] = 0; } for (int i = 0; i < 14; i++) { for (int j = 0; j < 13; j++) D2[i][j] = 0; } tmp.clear(); vector ord = { 6, 5, 7, 4, 8, 3, 9, 2, 10, 1, 11, 0, 12 }; for (int i = 0; i <13; i++) { D1[ord[i]][L] = 1; tmp.push_back(make_tuple(ord[i], L, ord[i] + 1, L)); } for (int i = 0; i <13; i++) { D1[ord[i]][R] = 1; tmp.push_back(make_tuple(ord[i], R, ord[i] + 1, R)); } for (int i = L; i < R; i++) { D2[ 0][i] = 1; tmp.push_back(make_tuple( 0, i, 0, i + 1)); } for (int i = L; i < R; i++) { D2[13][i] = 1; tmp.push_back(make_tuple(13, i, 13, i + 1)); } } // Set Rectangle: Case 2 void SetRect2(int L, int R) { for (int i = 0; i < 13; i++) { for (int j = 0; j < 14; j++) D1[i][j] = 0; } for (int i = 0; i < 14; i++) { for (int j = 0; j < 13; j++) D2[i][j] = 0; } tmp.clear(); vector ord = { 6, 5, 7, 4, 8, 3, 9, 2, 10, 1, 11, 0, 12 }; for (int i = 0; i <13; i++) { D2[L][ord[i]] = 1; tmp.push_back(make_tuple(L, ord[i], L, ord[i] + 1)); } for (int i = 0; i <13; i++) { D2[R][ord[i]] = 1; tmp.push_back(make_tuple(R, ord[i], R, ord[i] + 1)); } for (int i = L; i < R; i++) { D1[i][ 0] = 1; tmp.push_back(make_tuple(i, 0, i + 1, 0)); } for (int i = L; i < R; i++) { D1[i][13] = 1; tmp.push_back(make_tuple(i, 13, i + 1, 13)); } } int main() { // Input Initialize(); cin >> N >> T; for (int i = 1; i <= N; i++) cin >> A[i] >> B[i] >> C[i] >> D[i]; // Brute Force Type 1 Rectangle int Maximum = 0; tuple MaxID = make_tuple(-1, -1, -1); for (int i = 0; i < 14; i++) { for (int j = i + 1; j < 14; j++) { if (rand() % 100 >= 25) continue; SetRect1(i, j); int Score = GetEarnedMoney(); if (Maximum < Score) { Maximum = Score; MaxID = make_tuple(1, i, j); } } } // Brute Force Type 2 Rectangle for (int i = 0; i < 14; i++) { for (int j = i + 1; j < 14; j++) { if (rand() % 100 >= 25) continue; SetRect2(i, j); int Score = GetEarnedMoney(); if (Maximum < Score) { Maximum = Score; MaxID = make_tuple(2, i, j); } } } // Get tmp int BuiltRoads = 0; if (get<0>(MaxID) == 1) SetRect1(get<1>(MaxID), get<2>(MaxID)); if (get<0>(MaxID) == 2) SetRect2(get<1>(MaxID), get<2>(MaxID)); GetEarnedMoney(); // Answer Query for (int i = 1; i <= T; i++) { cin >> CurrentMoney >> CurrentCorps; if (CurrentMoney == -1 && CurrentCorps == -1) break; fprintf(stderr, "Turn =% 4d, Money=% 10d, Corps =% 3d, Roads =% 3d/%d\n", i, CurrentMoney, CurrentCorps, BuiltRoads, (int)tmp.size()); // Get Acts if (i <= 50) { cout << "2" << endl; } else if (BuiltRoads == (int)tmp.size()) { cout << "3" << endl; } else if (BuiltRoads >= 3 && CurrentMoney < Build[CurrentCorps]) { cout << "2" << endl; } else if (BuiltRoads <= 2 && CurrentMoney < Build[CurrentCorps]) { cout << "3" << endl; } else { cout << "1 " << get<0>(tmp[BuiltRoads]) + 1 << " " << get<1>(tmp[BuiltRoads]) + 1 << " " << get<2>(tmp[BuiltRoads]) + 1 << " " << get<3>(tmp[BuiltRoads]) + 1 << endl; BuiltRoads += 1; } } return 0; }