题意:有一个N*N的网格,左下角的坐标是(0,0),网格里有两个纵坐标相同的A,B两个宾馆,以及M个餐厅,A,B中各有一个宾馆,编号1,2,还有3~M的餐厅,你先在需要开一家新的餐厅,那么好的位置的标记是:对于已有的每个餐厅q使得,
dis(q,A)>diis(p,A)或dis(q,B)>dis(q,B),问有几个好的位置
思路:首先先分别求出只考虑相对于A,B每个横坐标的纵坐标与y1的距离大于等于d的时候是不再有好位置了,那么这个的关键在于:我们考虑当前横坐标的时候,首先要考虑到它之前一个的结果,因为我们可能是不能大于它的结果的+1,要不然就与它的重叠了 然后在依次相加就是答案,注意纵坐标的大小可能会影响相加时候的可能性
#include <iostream> #include <cstdio> #include <cstring> #include <algorithm> using namespace std; const int MAXN = 60001; const int INF = 0x3f3f3f3f; int miny[MAXN],maxy[MAXN],h[MAXN]; int n,m; int main(){ int t,x1,y1,x2,y2; scanf("%d",&t); while (t--){ scanf("%d%d",&n,&m); scanf("%d%d%d%d",&x1,&y1,&x2,&y2); if (x1 > x2){ swap(x1,x2); swap(y1,y2); } for (int i = x1+1; i < x2; i++){ miny[i] = INF; maxy[i] = -INF; } for (int i = 2; i < m; i++){ int ita,itb; scanf("%d%d",&ita,&itb); if (itb >= y1) //得出每个横坐标的最接近A,B的纵坐标 miny[ita] = min(miny[ita],itb); if (itb <= y1) maxy[ita] = max(maxy[ita],itb); } for (int i = x1; i <= x2; i++) //最后得出最近的值 h[i] = min(y1-maxy[i],miny[i]-y1); h[x1] = 0; for (int i = x1+1; i < x2; i++) h[i] = min(h[i],h[i-1]+1); h[x2] = 0; for (int i = x2-1; i > x1; i--) h[i] = min(h[i],h[i+1]+1); long long ans = 0; for (int i = x1; i <= x2; i++) if (h[i]){ // 每次安上去之后也算在q里面了 ans++; ans += min(h[i]-1,y1); ans += min(h[i]-1,n-y1-1); } printf("%lld\n",ans); } return 0; }