题目大意:平面上给出n个点,每个点有一个颜色,你可以选择一条线段以上或以下的所有点,并且这些点不能包含所有颜色,问最多能选多少点
假设找一条线段以下的
首先我们可以按y坐标排序,然后假设这条线段所在直线在最高处,这时答案等于相邻两个相同颜色之间的最大间隔点数
然后当这条线段所在直线逐渐下移时,相当于逐渐删掉这些点,每删掉一个就用这个点之前相邻的和之后相邻的之间的点数更新一下答案
这一步可以用链表+树状数组完成
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<map>
#define N 100010
using namespace std;
struct ppp{long long x,y,c,num;}a[N];
bool cmpx(ppp x,ppp y){return x.x<y.x;}
bool cmpy(ppp x,ppp y){return x.y<y.y;}
long long bef[N],nxt[N],pre[N];
long long ax[N];
long long cntx,cnty;
long long X[N],Y[N];
long long c[N];
map<int,int>p;
void change(long long x,long long v)
{
for(;x<=cntx;x+=x&-x)
c[x]+=v;
}
long long check(long long x)
{
long long ret=0;
for(;x;x-=x&-x)
ret+=c[x];
return ret;
}
void del(long long x)
{
bef[nxt[x]]=bef[x];
nxt[bef[x]]=nxt[x];
}
void doit()
{
cntx=0;cnty=0;
memset(pre,0,sizeof(pre));
long long n,k;
scanf("%lld%lld",&n,&k);
long long i,j,x,y,l;
for(i=1;i<=n;i++)
{
scanf("%lld%lld%lld",&a[i].x,&a[i].y,&a[i].c);
a[i].num=i;
}
ax[0]=707185547;
for(i=1;i<=n;i++)
ax[i]=a[i].x;
sort(ax+1,ax+n+1);
for(i=1;i<=n;i++)
if(ax[i]!=ax[i-1])
{
cntx++;
p[ax[i]]=cntx;
}
for(i=1;i<=n;i++)
a[i].x=p[a[i].x];
for(i=1;i<=n;i++)
ax[i]=a[i].y;
sort(ax+1,ax+n+1);
for(i=1;i<=n;i++)
if(ax[i]!=ax[i-1])
{
cnty++;
p[ax[i]]=cnty;
}
for(i=1;i<=n;i++)
a[i].y=p[a[i].y];
for(i=1;i<=n;i++)
X[i]=a[i].x,Y[i]=a[i].y;
sort(a+1,a+n+1,cmpx);
for(i=1;i<=n;i++)
{
nxt[pre[a[i].c]]=a[i].num;
bef[a[i].num]=pre[a[i].c];
// cout<<a[i].num<<' '<<bef[a[i].num]<<' '<<a[i].c<<'!'<<endl;
pre[a[i].c]=a[i].num;
}
for(i=1;i<=k;i++)
nxt[pre[i]]=n+1;
/* for(i=1;i<=n;i++)
cout<<a[i].num<<' '<<bef[i]<<' '<<nxt[i]<<endl;*/
X[n+1]=cntx+1;
long long ans=0;
for(i=1;i<=n;i++)
change(a[i].x,1);
for(i=1;i<=n;i++)
{
ans=max(ans,check(X[nxt[i]]-1)-check(X[i]));
ans=max(ans,check(X[i]-1)-check(X[bef[i]]));
}
// cout<<ans;
sort(a+1,a+n+1,cmpy);
j=1;
for(i=1;i<=n;i++)
{
while(a[j].y==a[i].y)
{
change(a[j].x,-1);
// cout<<a[j].x<<'!';
del(a[j].num);
j++;
}
l=a[i].num;
ans=max(ans,check(X[nxt[l]]-1)-check(X[bef[l]]));
// cout<<check(X[nxt[l]]-1)<<'@'<<ans<<' ';
}
memset(pre,0,sizeof(pre));
sort(a+1,a+n+1,cmpx);
for(i=1;i<=n;i++)
{
nxt[pre[a[i].c]]=a[i].num;
bef[a[i].num]=pre[a[i].c];
pre[a[i].c]=a[i].num;
}
for(i=1;i<=k;i++)
{
nxt[pre[i]]=n+1;
if(pre[i]==0) ans=n;
}
X[n+1]=cntx+1;
for(i=1;i<=n;i++)
change(a[i].x,1);
sort(a+1,a+n+1,cmpy);
j=n;
for(i=n;i>=1;i--)
{
while(a[j].y==a[i].y)
{
change(a[j].x,-1);
del(a[j].num);
j--;
}
l=a[i].num;
ans=max(ans,check(X[nxt[l]]-1)-check(X[bef[l]]));
}
printf("%lld\n",ans);
}
int main()
{
long long T;
scanf("%lld",&T);
while(T--)
doit();
}
平面点集颜色选择算法

本文介绍了一种解决平面上给定多个带颜色点的问题的算法。目标是找到一条线段,使得该线段一侧的点不包含所有颜色,并最大化这一侧的点的数量。文章详细解释了使用链表和树状数组进行高效计算的方法。
1458

被折叠的 条评论
为什么被折叠?



