zoukankan      html  css  js  c++  java
  • [BZOJ1818][CQOI2010]内部白点

    题目链接:

    BZOJ1818

    首先,题目根本不会有(-1)的情况,且所有节点变色只发生在第一秒。

    证明?如果一个节点((x,y))在第二秒变色,那么一定有一个节点会在第一秒内于((x,y))的四周生成。

    假设在左边(其他方向也一样),则设坐标为((x`,y))

    那么因为((x`,y))出现了,说明((x`,y))是个内部节点,那么在此之前((x`,y))的左边也一定有一个黑色节点,否则此节点不能变色。

    那么((x,y))就可以在第一秒变色。

    则题目变成求线段(两个(x)(y)轴坐标相同点的连线)的交点数量。

    那么将所有线段离散化,从左向右扫描竖线,对于横线用树状数组维护当前哪些地方有横向线段。

    若碰到线段左端,此线段对(y)坐标做出(1)的贡献,反之(-1)

    对于如((1,1),(3,1),(5,1))三个点,应该拆成两条线段,防止覆盖。

    看了看别人的代码,感觉自己的好麻烦。。还慢

    时间复杂度 (O(nlog_2n))

    #include <cstdio>
    #include <vector>
    #include <cstring>
    #include <algorithm>
    
    int n,Horc,Verc;
    int xs[100005],ys[100005];
    int as[100005];
    std::vector<int> cs[100005],St[100005],Ed[100005];
    
    struct Segment
    {
    	int l,r,p;
    }Hor[100005],Ver[100005];//线段,Hor为横向
    
    struct Binary_Indexed_Tree
    {
    	int c[100005];
    	
    	inline void Modify(int x,int v)
    	{
    		for(;x<=n;x+=x&-x)c[x]+=v;
    	}
    	
    	inline int Query(int x)
    	{
    		int Res=0;
    		for(;x;x^=x&-x)Res+=c[x];
    		return Res;
    	}
    }BIT;//树状数组
    
    void Simplify(int *a)//离散化
    {
    	memcpy(as,a,sizeof as);
    	std::sort(as+1,as+n+1);
    	int nl=std::unique(as+1,as+n+1)-as-1;
    	for(int i=1;i<=n;++i)
    		a[i]=std::lower_bound(as+1,as+nl+1,a[i])-as;
    }
    
    void Getseg(int *a,int *b,int &Cnt,Segment *Tar)//求线段
    {
    	for(int i=1;i<=n;++i)
    		cs[i].clear();
    	for(int i=1;i<=n;++i)
    		cs[b[i]].push_back(a[i]);
    	for(int i=1;i<=n;++i)
    		std::sort(cs[i].begin(),cs[i].end());
    	for(int i=1;i<=n;++i)
    		for(int j=1;j<(int)cs[i].size();++j)
    			if(cs[i][j]-cs[i][j-1]>=2)
    				Tar[++Cnt]=(Segment){cs[i][j-1]+1,cs[i][j]-1,i};
    }
    
    int main()
    {
    	scanf("%d",&n);
    	for(int i=1;i<=n;++i)
    		scanf("%d%d",&xs[i],&ys[i]);
    	Simplify(xs);
    	Simplify(ys);
    	Getseg(xs,ys,Horc,Hor);
    	Getseg(ys,xs,Verc,Ver);
    	for(int i=1;i<=Horc;++i)
    	{
    		St[Hor[i].l].push_back(Hor[i].p);
    		Ed[Hor[i].r].push_back(Hor[i].p);
    	}
    	int Sv=1;
    	long long Ans=0;
    	for(int i=1;i<=n;++i)
    	{
    		for(int j=0;j<(int)St[i].size();++j)
    			BIT.Modify(St[i][j],+1);//碰到左端端点
    		for(;Sv<=Verc&&Ver[Sv].p==i;++Sv)
    			Ans+=BIT.Query(Ver[Sv].r)-BIT.Query(Ver[Sv].l-1);
    		for(int j=0;j<(int)Ed[i].size();++j)
    			BIT.Modify(Ed[i][j],-1);//碰到右端端点
    	}
    	printf("%lld
    ",Ans+n);//还要加上原来的黑点
    	return 0;
    }
    
  • 相关阅读:
    [SCOI2015]国旗计划
    [SCOI2015]小凸玩矩阵
    点分治复习笔记
    [HNOI2014]米特运输
    [HNOI2015]接水果
    [HEOI2016/TJOI2016]游戏
    为什么验证集的loss会小于训练集的loss?
    转载GPU并行计算
    深度学习图像标注工具汇总(转载)
    你理解了吗?
  • 原文地址:https://www.cnblogs.com/LanrTabe/p/10176166.html
Copyright © 2011-2022 走看看