zoukankan      html  css  js  c++  java
  • 【CF875E】Delivery Club 二分+线段树

    【CF875E】Delivery Club

    题意:有n个快递需要依次接收,这n个快递分部在x轴上,第i个快递的位置是xi。有两个快递员,一开始分别在s0,s1,你可以任意安排哪个人收哪个快递,前提是一个快递员收快递是另一个快递员不能移动(也就是说他只有在收快递时能移动),并且要保证任何时候两人的距离不超过k。问你k最小是多少。

    n<=10^5,xi<=10^9

    题解:二分是显然的。我们可以用f[i][a][b]表示收第i个快递时,两个快递员一个在a,一个在b是否可行,又因为a或b一定等于i,所以我们可以省掉一维。我们还可以用线段树再省一维。因为在收第i+1个快递时,要么是在i处的快递员走到i+1,此时与i+1距离超过k的位置a都变成了不合法的,可以用线段树区间清零搞定;要么是在a处的快递员走到i+1。用线段树很容易维护这些东西。

    #include <cstdio>
    #include <cstring>
    #include <iostream>
    #include <algorithm>
    #define lson x<<1
    #define rson x<<1|1
    using namespace std;
    const int maxn=100010;
    int n;
    int v[maxn],p[maxn],rnk[maxn];
    bool s[maxn<<2];
    inline int Abs(const int &a) {return a>0?a:-a;}
    bool cmp(const int &a,const int &b)
    {
    	return v[a]<v[b];
    }
    inline void pushdown(int x)
    {
    	if(!s[x])	s[lson]=s[rson]=0;
    }
    void modify(int l,int r,int x,int a)
    {
    	if(l==r)
    	{
    		s[x]=1;
    		return ;
    	}
    	pushdown(x);
    	int mid=(l+r)>>1;
    	if(a<=mid)	modify(l,mid,lson,a);
    	else	modify(mid+1,r,rson,a);
    	s[x]=s[lson]|s[rson];
    }
    void updata(int l,int r,int x,int a,int b)
    {
    	if(a>b)	return ;
    	if(a<=l&&r<=b)
    	{
    		s[x]=0;
    		return ;
    	}
    	pushdown(x);
    	int mid=(l+r)>>1;
    	if(a<=mid)	updata(l,mid,lson,a,b);
    	if(b>mid)	updata(mid+1,r,rson,a,b);
    	s[x]=s[lson]|s[rson];
    }
    bool check(int len)
    {
    	int i,l,r,mid;
    	s[1]=0;
    	modify(1,n,1,rnk[1]);
    	for(i=3;i<=n;i++)
    	{
    		l=1,r=rnk[i];
    		while(l<r)
    		{
    			mid=(l+r)>>1;
    			if(v[p[mid]]<v[i]-len)	l=mid+1;
    			else	r=mid;
    		}
    		updata(1,n,1,1,l-1);
    		l=rnk[i],r=n;
    		while(l<r)
    		{
    			mid=(l+r)>>1;
    			if(v[p[mid]]<=v[i]+len)	l=mid+1;
    			else	r=mid;
    		}
    		updata(1,n,1,l,n);
    		if(Abs(v[i]-v[i-1])<=len)	modify(1,n,1,rnk[i-1]);
    		if(!s[1])	return 0;
    	}
    	return 1;
    }
    inline int rd()
    {
    	int ret=0,f=1;	char gc=getchar();
    	while(gc<'0'||gc>'9')	{if(gc=='-')	f=-f;	gc=getchar();}
    	while(gc>='0'&&gc<='9')	ret=ret*10+(gc^'0'),gc=getchar();
    	return ret*f;
    }
    int main()
    {
    	n=rd()+2;
    	int i,l=0,r=0,mid;
    	for(i=1;i<=n;i++)	v[i]=rd(),r=max(r,v[i]),p[i]=i;
    	sort(p+1,p+n+1,cmp);
    	for(i=1;i<=n;i++)	rnk[p[i]]=i;
    	v[0]=-1000000000,v[n+1]=1000000000;
    	l=Abs(v[2]-v[1]);
    	while(l<r)
    	{
    		mid=(l+r)>>1;
    		if(check(mid))	r=mid;
    		else	l=mid+1;
    	}
    	printf("%d",r);
    	return 0;
    }
  • 相关阅读:
    Vector 、ArrayList、LinkedList比较
    MySQL主从复制
    多浏览器兼容flv视频播放HTML
    [转]javascript对联广告、漂浮广告封装类,多浏览器兼容
    ASP获取json天气信息
    IIS6的SSL配置,如何配置SSL到登陆页,如何将SSL证书设置成受信任的证书
    Web Service初探
    算法之逆序对
    算法之数组和问题
    重读算法导论之算法基础
  • 原文地址:https://www.cnblogs.com/CQzhangyu/p/8157512.html
Copyright © 2011-2022 走看看