[AGC018B] Sports Festival

题目大意

\(n\) 个人 \(m\) 个活动,告诉你每一个人对于活动喜欢程度的排序,你可以鸽掉一些活动。如果一个最喜欢的活动被鸽了,那么他就会参加次喜欢的,依次类推直到参加为止。

求参加人数最多的那个项目,参加人数最少是多少。

思路

做法

首先模拟出每一个活动的参加人数,依次将人数最多的活动鸽掉,一遍鸽一遍统计最小值。

理由

假设你已经鸽掉了一些活动,你么现在总能找到一个参加人数最多的项目,假设这个项目是第 \(x\) 个项目。如果将非 \(x\) 的项目鸽掉,那么参加第 \(x\) 个项目的人就并不会减少反而还会增加,这对将整体的最大值减小是没有任何效果的。

假设你鸽掉了第 \(x\) 个项目,那么你肯定就又可以找到一个参加人数是最多的项目,假设是第 \(y\) 个项目。那么如果不鸽掉第 \(y\) 个项目与前面鸽掉第 \(x\) 个项目的效果其实是一样的。所以在这次操作就只能将 \(y\) 鸽掉,否则就没有意义。

AC Code

#include<bits/stdc++.h>
#define int long long
using namespace std;
const int N=305;
int n,m,a[N][N],s[N],top[N],ans=1e18;
signed main(){
	ios::sync_with_stdio(false);
	cin.tie(nullptr);
	cin>>n>>m;
	for(int i=1;i<=n;i++){
		for(int j=1;j<=m;j++){
			cin>>a[i][j];
			top[i]=1;
		}
	}
	for(int i=1;i<=n;i++){
		s[a[i][1]]++;
	}
	for(int i=1,p,mmax;i<=m;i++){
		mmax=0;
		for(int j=1;j<=m;j++){
			if(s[j]>mmax){
				mmax=s[j];
				p=j;
			}
		}
		ans=min(ans,mmax);
		s[p]=-1;
		for(int j=1;j<=n;j++){
			bool flag=0;
			while(s[a[j][top[j]]]==-1){
				top[j]++;
				flag=1;
			}
			if(flag){
				s[a[j][top[j]]]++;
			}
		}
	}
	cout<<ans<<'\n';
	return 0;
}
posted @ 2024-07-14 13:42  未抑郁的刘大狗  阅读(3)  评论(0编辑  收藏  举报