Java实现 蓝桥杯VIP 算法训练 会议中心

算法训练 会议中心
时间限制:2.0s 内存限制:512.0MB
  会议中心  Siruseri政府建造了一座新的会议中心。许多公司对租借会议中心的会堂很感兴趣,他们希望能够在里面举行会议。
  对于一个客户而言,仅当在开会时能够独自占用整个会堂,他才会租借会堂。会议中心的销售主管认为:最好的策略应该是将会堂租借给尽可能多的客户。显然,有可能存在不止一种满足要求的策略。
  例如下面的例子。总共有4个公司。他们对租借会堂发出了请求,并提出了他们所需占用会堂的起止日期(如下表所示)。
在这里插入图片描述

上例中,最多将会堂租借给两家公司。租借策略分别是租给公司1和公司3,或是公司2和公司3,也可以是公司1和公司4。注意会议中心一天最多租借给一个公司,所以公司1和公司2不能同时租借会议中心,因为他们在第九天重合了。
  销售主管为了公平起见,决定按照如下的程序来确定选择何种租借策略:首先,将租借给客户数量最多的策略作为候选,将所有的公司按照他们发出请求的顺序编号。对于候选策略,将策略中的每家公司的编号按升序排列。最后,选出其中字典序最小[1]的候选策略作为最终的策略。
  例中,会堂最终将被租借给公司1和公司3:3个候选策略是{(1,3),(2,3),(1,4)}。而在字典序中(1,3)<(1,4)<(2,3)。
  你的任务是帮助销售主管确定应该将会堂租借给哪些公司。
输入格式
  输入的第一行有一个整数N,表示发出租借会堂申请的公司的个数。第2到第N+1行每行有2个整数。第i+1行的整数表示第i家公司申请租借的起始和终止日期。对于每个公司的申请,起始日期为不小于1的整数,终止日期为不大于109的整数。
输出格式
  输出的第一行应有一个整数M,表示最多可以租借给多少家公司。第二行应列出M个数,表示最终将会堂租借给哪些公司。
数据规模和约定
  对于50%的输入,N≤3000。在所有输入中,N≤200000。
样例输入
4
4 9
9 11
13 19
10 17
样例输出
2
1 3

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;


public class 会议中心 {
	static int[] sqqs,sqzz,sqsx,sqcf,cxz,finz;
	static int[][] dgb;
	static int n=0,fin=0,max=0;
	public static void main(String[] args)throws IOException {  
		BufferedReader reader=new BufferedReader(new InputStreamReader(System.in));
	    String s=reader.readLine();
	    n=Integer.parseInt(s);
        sqqs=new int[n+1];
        sqzz=new int[n+1];
        sqcf=new int[n+1];
	for(int i=1;i<=n;i++){
		s=reader.readLine();
	    String[] sa = s.split(" ");
	    sqqs[i]=Integer.parseInt(sa[0]);
	    sqzz[i]=Integer.parseInt(sa[1]);
	    if (max<sqzz[i]) max=sqzz[i];
	}
	sqsx=new int[max+1];
	for (int i=1;i<=n;i++){
		if (sqsx[sqzz[i]]==0){
	    	sqsx[sqzz[i]]=i;
	    }else{
	    	if (sqqs[i]>sqqs[sqsx[sqzz[i]]]){    
	    	     for (int j=1;j<=n;j++){
	    	 	     if (sqcf[j]==0){
	    				  sqcf[j]=sqsx[sqzz[i]];
	    				  sqsx[sqzz[i]]=i;
	    	 	    	  break;
	    		     }
	    	     }
	    	}
	    }
	}
	dgb=new int[2][n+1];
	for (int i=1;i<=max;i++) {
		if (sqsx[i]!=0) {
			getcs(sqsx[i]);
		}
	}
	for (int i=1;i<=n;i++){
		if (fin<dgb[0][i]) fin=dgb[0][i];
	}
	for (int k=max;k>=1;k--){
		if (sqsx[k]==0) continue;
		int i=sqsx[k];
		if (dgb[0][i]==fin) continue;
		for (int j=sqzz[i]+1;j<=max;j++){
			if (j==max){
			    dgb[0][i]=0;
			    dgb[1][i]=0;
			    continue;
			}
			if (sqsx[j]==0) continue;
			if (dgb[0][sqsx[j]]==0) continue;
			if ((dgb[0][sqsx[j]]==dgb[0][i]+1)&(sqqs[sqsx[j]]>sqzz[i])) break;
			if (dgb[0][sqsx[j]]>dgb[0][i]+1) {
				dgb[0][i]=0;
				dgb[1][i]=0;
				break;
			}
		}
		
	}
	for (int i=1;i<=n;i++) {
		if (sqcf[i]!=0) {
			getcs1(sqcf[i]);
		} else break;
	}
	
	cxz=new int[fin+1];
	finz=new int[fin+1];
	int xh=1;
	for (int i=1;i<=n;i++){
		if (dgb[0][i]==0) continue;
		if (cxz[dgb[0][i]]!=0) continue;
		if (fincheck(i)==1){
		  	finz[xh]=i;
			xh++;
			cxz[dgb[0][i]]=i;
			dealsq(i);
		}
	}
    System.out.println(fin);
   
    for (int i=1;i<=fin;i++){
    	System.out.print(""+finz[i]+" ");
    }
   System.out.println();
	}
static void getcs(int a){
	int cs=0;
	for (int i=sqqs[a]-1;i>0;i--){
		if (sqsx[i]!=0){
			cs=dgb[0][sqsx[i]];
			if (cs!=0) {
				break;
			}
		}
	}
	for (int i=sqqs[a];i<sqzz[a];i++){
		if (sqsx[i]!=0){
			if (dgb[0][sqsx[i]]>cs+1) return;
		}
	}
	dgb[0][a]=cs+1;
	dgb[1][a]=1;
}
static void getcs1(int a){
	int cs=0;
	int b=dgb[0][sqsx[sqzz[a]]];
	if (b==0) return;
	for (int i=sqqs[a]-1;i>0;i--){
		if (sqsx[i]!=0){
			cs=dgb[0][sqsx[i]];
			if (cs!=0) {
				break;
			}
		}
	}
	if (cs+1<b) return;
	dgb[0][a]=cs+1;
	dgb[1][a]=1;
}
static int fincheck(int a){
	int[] lsst=new int[fin+1];
	int b=0,c=0,sc=0,ec=0;
	int cs=dgb[0][a];
	for (int i=cs-1;i>=1;i--){
		if (cxz[i]!=0) {
			b=cxz[i];
			sc=i;
			break;
		}
	}
	if (b!=0){
		lsst[sc]=sqzz[b];
		for (int i=sqzz[b]+1;i<sqqs[a];i++){
			if (sqsx[i]==0) continue;
			if (dgb[0][sqsx[i]]==0) continue;
			if (lsst[dgb[0][sqsx[i]]]!=0) continue;
			if (sqqs[sqsx[i]]<=lsst[dgb[0][sqsx[i]]-1]) continue;
			lsst[dgb[0][sqsx[i]]]=sqzz[sqsx[i]];
		}
		if (lsst[cs-1]==0) return 0;
		if (lsst[cs-1]>=sqqs[a]) return 0;
	}
	for (int i=cs+1;i<=fin;i++){
		if (cxz[i]!=0){
			c=cxz[i];
			ec=i;
			break;
		}
	}
	if (c!=0){
		lsst[ec]=sqqs[c];
		for (int i=sqqs[c]-1;i>sqzz[a];i--){
			if (sqsx[i]==0) continue;
			if (dgb[0][sqsx[i]]==0) continue;
			if (i>=lsst[dgb[0][sqsx[i]]+1]) continue;
			lsst[dgb[0][sqsx[i]]]=Math.max(sqqs[sqsx[i]],lsst[dgb[0][sqsx[i]]]);
		}
		if (lsst[cs+1]==0) return 0;
		if (lsst[cs+1]<=sqzz[a]) return 0;
	}
	return 1;
}
static void dealsq(int a){
	int cs=dgb[0][a];
	for (int i=sqzz[a]-1;i>=1;i--){
		if (sqsx[i]==0) continue;
		if (dgb[0][sqsx[i]]==0) continue;
		int dcs=dgb[0][sqsx[i]];
		if (dcs==cs) {
			dgb[0][sqsx[i]]=0;
			dgb[1][sqsx[i]]=0;
			continue;
		}
		if (dcs==cs-1) {
			if (i>=sqqs[a]) {
				dgb[0][sqsx[i]]=0;
				dgb[1][sqsx[i]]=0;
				continue;
			}
		}
		if (dcs==cs-2) break;
	}
	for (int i=sqzz[a]+1;i<=max;i++){
		if (sqsx[i]==0) continue;
		if (dgb[0][sqsx[i]]==0) continue;
		int dcs=dgb[0][sqsx[i]];
		if (dcs==cs) {
			dgb[0][sqsx[i]]=0;
			dgb[1][sqsx[i]]=0;
			continue;
		}
		if (dcs==cs+1) {
			if (sqzz[a]>=sqqs[sqsx[i]]) {
				dgb[0][sqsx[i]]=0;
				dgb[1][sqsx[i]]=0;
				continue;
			}
		}
		if (dcs==cs+2) break;
	}
}

}

原文地址:https://www.cnblogs.com/a1439775520/p/13078452.html