zoukankan      html  css  js  c++  java
  • 【BZOJ 3196】 Tyvj 1730 二逼平衡树 分块

    分块 就是分块嘛!

    #include <cstdio>
    #include <iostream>
    #include <algorithm>
    #include <cstring>
    #include <cstdlib>
    #include <cmath>
    #define MAXN 1000000
    #define INF 1000000000
    #define MO 233333333
    #define LL long long
    using namespace std;
    int num[300][300];
    int init[50000+1];
    int n,m;
    int cnt,lenth;
    int Find(int tmp[300],int k)
    {
    	int L=1,R=tmp[0];
    	while(L<R)
    	{
    		int mid=(L+R)/2;
    		if(tmp[mid]>=k) R=mid;
    		else L=mid+1;
    	}
    	if(tmp[L]>=k)return L-1;
    	return L;
    }
    int Rank(int L,int R,int k)
    {
    	int ans=0;
    	int s=(L+lenth-1)/lenth,t=(R+lenth-1)/lenth;// cout<<lenth<<endl;
    	if(s!=t)
    	{
    		for(int i=L;i<=lenth*s;i++)
    			if(init[i]<k)
    				ans++;
    		s++;
    		for(int i=R;i>=lenth*(t-1)+1;i--)
    			if(init[i]<k)
    				ans++;
    		for(int i=s;i<t;i++)
    			ans+=Find(num[i],k);
    	}
    	else
    	{
    		for(int i=L;i<=R;i++)
    			if(init[i]<k) 
    				ans++;
    	}
    	return ans+1;
    }
    int Kth(int L,int R,int k)
    {
    	int s=0,t=100000000;
    	int ans=0;
    	while(s<t)
    	{
    		int mid=(s+t)/2;
    		if(Rank(L,R,mid)>k) t=mid;
    		else s=mid+1,ans=mid;
    	}
    	return ans;
    }
    void Change(int loc,int k)
    {
    	init[loc]=k;
    	int tmp=(loc+lenth-1)/lenth;
    	for(int i=1;i<=num[tmp][0];i++)
    		num[tmp][i]=init[(tmp-1)*lenth+i];
    	sort(num[tmp]+1,num[tmp]+num[tmp][0]+1);
    }
    int Before(int L,int R,int k)
    {
    	int s=(L+lenth-1)/lenth,t=(R+lenth-1)/lenth;
    	int mmax=-1;//cout<<s<<' '<<t<<endl;
    	if(s!=t)
    	{
    		for(int i=L;i<=s*lenth;i++)
    			if(init[i]<k)
    				mmax=max(mmax,init[i]);
    		s++;
    		for(int i=R;i>=(s-1)*lenth;i--)
    			if(init[i]<k)
    				mmax=max(mmax,init[i]);
    		for(int i=s;i<t;i++)
    		{
    			int x=Find(num[i],k);
    			if(x!=0)
    				mmax=max(num[i][x],mmax);
    		}
    	}
    	else
    	{
    		for(int i=L;i<=R;i++)
    			if(init[i]<k)
    				mmax=max(mmax,init[i]);
    	}
    	return mmax;
    }
    int Find_After(int tmp[300],int k)
    {
    	int L=1,R=tmp[0];
    	while(L<R)
    	{
    		int mid=(L+R)/2;
    		if(tmp[mid+1]<=k) L=mid+1;
    		else R=mid;
    	}
    	if(tmp[L]<=k) return L+1;
    	return L;
    }
    int After(int L,int R,int k)
    {
    	int s=(L+lenth-1)/lenth,t=(R+lenth-1)/lenth;
    	int mmin=100000000+1;
    	if(s!=t)
    	{
    		for(int i=L;i<=s*lenth;i++)
    			if(init[i]>k)
    				mmin=min(mmin,init[i]);
    		s++;
    		for(int i=R;i>=(s-1)*lenth;i--)
    			if(init[i]>k)
    				mmin=min(mmin,init[i]);
    		for(int i=s;i<t;i++)
    		{
    			int x=Find_After(num[i],k);
    			if(x!=num[i][0]+1)
    				mmin=min(num[i][x],mmin);
    		}
    	}
    	else
    	{
    		for(int i=L;i<=R;i++)
    			if(init[i]>k)
    				mmin=min(mmin,init[i]);
    	}
    	return mmin;
    }
    int main()
    {
    //	freopen("a.in","r",stdin);
    	cin>>n>>m; //cout<<n<<endl;
    	for(int i=1;i<=n;i++) scanf("%d",&init[i]);
    	lenth=sqrt(n)+2;
    	cnt=lenth;
    	for(int i=1;i<=cnt;i++)
    	{
    		for(int j=1;j<=lenth;j++)
    			num[i][j]=init[(i-1)*lenth+j];
    		sort(num[i]+1,num[i]+lenth+1);
    		num[i][0]=lenth;
    	}
    	if(n%lenth!=0)
    	{
    		cnt++;
    		num[cnt][0]=n%lenth;
    		for(int i=1;i<=num[cnt][0];i++)
    			num[cnt][i]=init[(cnt-1)*lenth+i];
    		sort(num[cnt]+1,num[cnt]+1+num[cnt][0]);
    	}
    	int L,R,pos,opt,k;
    	for(int i=1;i<=m;i++)
    	{
    		scanf("%d",&opt);
    		switch (opt)
    		{
    			case 1:
    				{
    					scanf("%d %d %d",&L,&R,&k); 
    					printf("%d
    ",Rank(L,R,k));
    					break;
    				}
    			case 2:
    				{
    					scanf("%d %d %d",&L,&R,&k);
    					printf("%d
    ",Kth(L,R,k));
    					break;
    				}
    			case 3:
    				{
    					scanf("%d %d",&pos,&k);
    					Change(pos,k);
    					break; 
    				}
    			case 4:
    				{
    					scanf("%d %d %d",&L,&R,&k);
    					printf("%d
    ",Before(L,R,k));
    					break;
    				}
    			case 5:
    				{
    					scanf("%d %d %d",&L,&R,&k);
    					printf("%d
    ",After(L,R,k)); 
    					break;
    				}
    		}
    
    
    	}
    
    
    	return 0;
    }
    
  • 相关阅读:
    从苏宁电器到卡巴斯基第13篇:我在苏宁电器当营业员 V
    从苏宁电器到卡巴斯基第12篇:我在苏宁电器当营业员 IV
    从苏宁电器到卡巴斯基第11篇:我在苏宁电器当营业员 III
    从苏宁电器到卡巴斯基第10篇:我在苏宁电器当营业员 II
    从苏宁电器到卡巴斯基第09篇:我在苏宁电器当营业员 I
    从苏宁电器到卡巴斯基第08篇:来到苏宁之前的过渡
    【目录】从苏宁电器到卡巴斯基
    从苏宁电器到卡巴斯基第07篇:我在佳木斯的日子(下)
    从苏宁电器到卡巴斯基第06篇:我在佳木斯的日子(中)
    从苏宁电器到卡巴斯基第05篇:我在佳木斯的日子(上)
  • 原文地址:https://www.cnblogs.com/ofsxb/p/5133145.html
Copyright © 2011-2022 走看看