zoukankan      html  css  js  c++  java
  • 【CF1181D】Irrigation

    题目大意:给定 M 个城市,每年会选出一个城市举办比赛,现给出前 N 年城市举办比赛的情况。在接下来的年份中,每年会在举办比赛次数最小的城市举办比赛,如果有很多城市举办次数均为最小值,则在编号最小的城市举办比赛。现给出 Q 个询问,每次询问第 K 年在哪个城市举办比赛。

    题解:
    首先,记录下每个城市前 N 年举办比赛的次数,并按照举办比赛的次数为第一关键字,城市编号为第二关键字进行从小到大排序。
    发现排序后城市举办比赛的情况如下图所示

    那么若当前矩形高度小于下一个小矩形高度时,一定是当前矩形高度增加,且对于大矩形内的小矩形高度呈现周期性增加。
    因此,可以采用将询问离线,并将问题划分为前 i 个高度均为 h[i] 的矩形到前 i+1 个高度均为 h[i+1] 的矩形的时间。对于每个划分出来的子问题,i 个小矩形高度呈现周期性变化,可以利用取模操作得到第几个矩形高度增加。需要注意的是,这里面的第 i 个矩形为相对下标,还需要转化成具体是第几个城市,这里可以采用权值线段树上二分的操作,求出第 k 个数对应的下标即可。

    代码如下

    #include <bits/stdc++.h>
    using namespace std;
    const int maxn=5e5+10;
    typedef long long LL;
    
    int n,m,Q,sz[maxn<<2],ans[maxn];
    struct city{int cnt,id;}c[maxn];
    struct qry{int id;LL year;}q[maxn];
    void insert(int o,int l,int r,int pos){
    	if(l==r){++sz[o];return;}
    	int mid=l+r>>1;
    	if(pos<=mid)insert(o<<1,l,mid,pos);
    	else insert(o<<1|1,mid+1,r,pos);
    	sz[o]=sz[o<<1]+sz[o<<1|1];
    }
    int kth(int o,int l,int r,int k){
    	if(l==r)return l;
    	int mid=l+r>>1;
    	if(k<=sz[o<<1])return kth(o<<1,l,mid,k);
    	else return kth(o<<1|1,mid+1,r,k-sz[o<<1]);
    }
    
    void read_and_parse(){
    	scanf("%d%d%d",&n,&m,&Q);
    	for(int i=1;i<=n;i++){
    		int x;scanf("%d",&x);
    		++c[x].cnt;
    	}
    	for(int i=1;i<=m;i++)c[i].id=i;
    	for(int i=1;i<=Q;i++){
    		LL year;scanf("%lld",&year);
    		q[i].id=i,q[i].year=year;
    	}
    	sort(c+1,c+m+1,[](const city &a,const city &b){
    		return a.cnt==b.cnt?a.id<b.id:a.cnt<b.cnt;
    	});
    	sort(q+1,q+Q+1,[](const qry &a,const qry &b){
    		return a.year<b.year;
    	});
    }
    void solve(){
    	LL now=n;
    	for(int i=1,j=1;i<=m;i++){
    		LL nxt=i==m?2e18:(LL)(c[i+1].cnt-c[i].cnt)*i+now;
    		insert(1,1,m,c[i].id);
    		while(j<=Q&&q[j].year<=nxt){
    			LL year=q[j].year;
    			int idx=(year-now)%i;
    			if(idx==0)idx=i;
    			ans[q[j].id]=kth(1,1,m,idx);
    			++j;
    		}
    		now=nxt;
    	}
    	for(int i=1;i<=Q;i++)printf("%d
    ",ans[i]);
    }
    int main(){
    	read_and_parse();
    	solve();
    	return 0;	
    } 
    
  • 相关阅读:
    Spring MVC 入门示例讲解
    使用Word2013,发布博文到博客园
    走过2103,迈向2014
    C#中的字符串拼接@,$
    ORA-22922:nonexistent LOB value问题及listagg()函数
    证明一个数是不是存在于该数组中
    论java中System.arrayCopy()与Arrays.copyOf()的区别
    JSP中获取各种路径的方法
    JavaScript中变量声明有var和没var的区别
    JavaScript:理解事件循环
  • 原文地址:https://www.cnblogs.com/wzj-xhjbk/p/11039510.html
Copyright © 2011-2022 走看看