#include #include #include #include #include #include using namespace std; class DSU { public: int n; vector parent_or_size; DSU(int N) { n = N; parent_or_size.resize(N, -1); } int leader(int a) { assert(0 <= a && a < n); if (parent_or_size[a] < 0) { return a; } return parent_or_size[a] = leader(parent_or_size[a]); } int merge(int a, int b) { assert(0 <= a && a < n); assert(0 <= b && b < n); int x = leader(a); int y = leader(b); if (x == y) { return x; } if (-parent_or_size[x] < -parent_or_size[y]) { swap(x, y); } parent_or_size[x] += parent_or_size[y]; parent_or_size[y] = x; return x; } bool same(int a, int b) { assert(0 <= a && a < n); assert(0 <= b && b < n); return leader(a) == leader(b); } int size(int a) { assert(0 <= a && a < n); return -parent_or_size[leader(a)]; } vector> groups() { vector leader_buf(n); vector group_size(n, 0); for (int i = 0; i < n; ++i) { leader_buf[i] = leader(i); ++group_size[leader_buf[i]]; } vector> result(n); for (int i = 0; i < n; ++i) { result[leader_buf[i]].push_back(i); } vector> result2; for (int i = 0; i < n; ++i) { if (!result[i].empty()) { result2.push_back(result[i]); } } return result2; } }; int main() { int H, W; cin >> H >> W; vector> A(H, vector(W)); for (int i = 0; i < H; ++i) { for (int j = 0; j < W; ++j) { cin >> A[i][j]; } } set C; vector> q; int Q; cin >> Q; for (int i = 0; i < Q; ++i) { int a, b, c, d; cin >> a >> b >> c >> d; q.push_back({(a - 1) * W + (b - 1), (c - 1) * W + (d - 1)}); C.insert(i); } vector>> L(H * W + 1); for (int i = 0; i < H; ++i) { for (int j = 0; j < W; ++j) { if (i < H - 1) { int x = A[i][j]; int y = A[i + 1][j]; int z = max(x, y); L[z].push_back({i * W + j, (i + 1) * W + j}); } if (j < W - 1) { int x = A[i][j]; int y = A[i][j + 1]; int z = max(x, y); L[z].push_back({i * W + j, i * W + j + 1}); } } } vector result(Q, 0); DSU Z(H * W); for (int x = 1; x <= H * W; ++x) { if (L[x].empty()) continue; for (auto& B : L[x]) { int pos1 = B.first, pos2 = B.second; Z.merge(pos1, pos2); } vector l; for (int k : C) { int p1 = q[k].first, p2 = q[k].second; if (Z.same(p1, p2)) { l.push_back(k); result[k] = x; } } for (int k : l) { C.erase(k); } } for (int i = 0; i < Q; ++i) { cout << result[i] << endl; } return 0; }