标签:
题意:
第1行:2个数N, Q中间用空格分隔,其中N为岛的数量,Q为查询的数量(1 <= N, Q <= 50000)。 第2 - N + 1行,每行1个数,对应N个岛屿的高度(1 <= A[i] <= 10^9)。 第N + 2 - N + Q + 1行,每行一个数,对应查询的海平面高度(1 <= Q[i] <= 10^9)。
输出共Q行,对应每个查询的岛屿数量。
5 4 2 1 3 2 3 0 1 3 2
1 2 0 2
这个题目用了一些小技巧
可以看作1-n周围的已经被淹没 cnt = 2;
那么 cnt -1 就是答案
当一个岛被淹没 如果他的左右有一块被淹没的区域 那么cnt不变
当周围没被淹 那么cnt++
如果周围都被淹没 那么淹没的区域为cnt--
#include<stdio.h> #include<iostream> #include<algorithm> #include<cstring> #include<stack> #include<climits> #include<vector> using namespace std; const int N = 5e4+10; int min_h = INT_MAX,max_h = INT_MIN; struct data { int h,id; /* data */ }Q[N]; bool vis[N]; vector<int>v[N],num; int H[N]; int ans[N]; int getid(int x) { x++; return lower_bound(num.begin(),num.end(),x)-num.begin()-1; } bool cmp(data a,data b) { return a.h<b.h; } int cnt = 2; void check(int pos) { vis[pos] = true; if(vis[pos+1]&&vis[pos-1]) { cnt--; } else{ if(vis[pos+1]||vis[pos-1]) { ; } else{ cnt++; } } } int main() { int n,q,t; scanf("%d%d",&n,&q); for(int i=1;i<=n;i++) { scanf("%d",&t); num.push_back(t); H[i] = t; max_h = max(max_h,t); min_h = min(min_h,t); } sort(num.begin(),num.end()); num.erase(unique(num.begin(),num.end()),num.end() ); for(int i=1;i<=n;i++) { //cout<<getid(H[i])<<endl; v[getid(H[i])].push_back(i); } vis[0] = vis[n+1] = true; for(int i=0;i<q;i++) { scanf("%d",&Q[i].h); Q[i].id = i; } sort(Q,Q+q,cmp); int pre = -1; for(int i=0;i<q;i++) { if(Q[i].h>=max_h) { ans[Q[i].id] = 0; continue; } if(Q[i].h<min_h) { ans[Q[i].id] = 1; continue; } int now = getid(Q[i].h); //cout<<endl<<"H: "<<Q[i].h<<" id: "<<now<<endl; for(int j = pre+1;j<=now;j++) { int len = v[j].size(); for(int k=0;k<len;k++) { check(v[j][k]); //cout<<v[j][k]<<" "; } } pre = now; ans[Q[i].id] = cnt - 1; } for(int i=0;i<q;i++) { printf("%d\n",ans[i]); } return 0; }
标签:
原文地址:http://www.cnblogs.com/Geek-xiyang/p/5847521.html