1127: [POI2008]KUP

1127: [POI2008]KUP

https://lydsy.com/JudgeOnline/problem.php?id=1127

分析:

  如果存在一个点大于等于k,小于等于2k的话,直接输出。

  否则把点分成两类,一类是<k的,另一类是大于2k的,大于2k的一定没用。

  然后找一个全部由小于2k的点中组成一个的矩形(悬线法),这个矩形有三种情况:1、<k,没用;2、大于等于k,小于等于2k,输出;3、大于2k,它的子矩阵中一定存在一个合法的矩阵(因为每个元素都是<k的,所以增加一个元素不可能直接使面积从小于k变成大于等于2k)。

  考虑如何对一个大于等于2k的矩形找到它的合法的子矩阵。每次删掉一行或者一列一定可以找到。

  

代码:

 1 #include<cstdio>
 2 #include<algorithm>
 3 #include<cstring>
 4 #include<iostream>
 5 #include<cmath>
 6 #include<cctype>
 7 #include<set>
 8 #include<queue>
 9 #include<vector>
10 #include<map>
11 #include<cstdlib>
12 using namespace std;
13 typedef long long LL;
14 
15 inline int read() {
16     int x=0,f=1;char ch=getchar();for(;!isdigit(ch);ch=getchar())if(ch=='-')f=-1;
17     for(;isdigit(ch);ch=getchar())x=x*10+ch-'0';return x*f;
18 }
19 
20 const int N = 2005;
21 
22 LL sum[N][N], k;
23 int a[N][N], U[N][N], L[N], R[N], n;
24 
25 void pd(int u,int d,int l,int r) {
26     if (u > d || l > r) return ;
27     LL now = sum[d][r] - sum[d][l - 1] - sum[u - 1][r] + sum[u - 1][l - 1];
28     if (now < k) return ;
29     if (now >= k && now <= k + k) {
30         printf("%d %d %d %d",l, u, r, d); exit(0);
31     }
32     if (d - u > r - l) {
33         pd(u + 1, d, l, r); pd(u, d - 1, l, r);
34         pd(u, d, l + 1, r); pd(u, d, l, r - 1);
35     }
36     else {
37         pd(u, d, l + 1, r); pd(u, d, l, r - 1);
38         pd(u + 1, d, l, r); pd(u, d - 1, l, r);
39     }
40 }
41 
42 int main() { 
43     k = read(), n = read();
44     for (int i = 1; i <= n; ++i) 
45         for (int j = 1; j <= n; ++j) {
46             a[i][j] = read();
47             sum[i][j] = a[i][j] + sum[i][j - 1] + sum[i - 1][j] - sum[i - 1][j - 1];
48             if (a[i][j] >= k && a[i][j] <= k + k) { printf("%d %d %d %d\n",j, i, j, i); return 0; }
49         }
50     for (int i = 1; i <= n; ++i)
51         for (int j = 1; j <= n; ++j) 
52              U[i][j] = a[i][j] <= k + k ? U[i - 1][j] + 1 : 0;
53     for (int i = 1; i <= n; ++i) L[i] = 0, R[i] = n + 1;
54     for (int i = 1; i <= n; ++i) {
55         int last = 0;
56         for (int j = 1; j <= n; ++j) {
57              if (a[i][j] <= k + k) L[j] = max(L[j], last + 1);
58              else last = j, L[j] = 0;
59         }
60         last = n + 1;
61         for (int j = n; j >= 1; --j) {
62             if (a[i][j] <= k + k) R[j] = min(R[j], last - 1);
63             else last = j, R[j] = n + 1;
64         }
65         for (int j = 1; j <= n; ++j) 
66             if (U[i][j] && L[j] >= 1 && R[j] <= n) pd(i - U[i][j] + 1, i, L[j], R[j]);
67     }
68     puts("NIE");
69     return 0;
70 }

 

posted @ 2018-12-02 09:42  MJT12044  阅读(198)  评论(0编辑  收藏  举报